/** FILE: OpenLayers/SingleFile.js **/ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for * full list of 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 OpenLayers = { /** * Constant: VERSION_NUMBER */ VERSION_NUMBER: "Release 2.13 dev", /** * Constant: singleFile * TODO: remove this in 3.0 when we stop supporting build profiles that * include OpenLayers.js */ singleFile: true, /** * Method: _getScriptLocation * Return the path to this script. This is also implemented in * OpenLayers.js * * Returns: * {String} Path to this script */ _getScriptLocation: (function() { var r = new RegExp("(^|(.*?\\/))(OpenLayers[^\\/]*?\\.js)(\\?|$)"), s = document.getElementsByTagName('script'), src, m, l = ""; for(var i=0, len=s.length; i * * (end code) * * Please remember that when your OpenLayers script is not named * "OpenLayers.js" you will have to make sure that the default theme is * loaded into the page by including an appropriate -tag, * e.g.: * * (code) * * (end code) */ ImgPath : '' }; /** FILE: OpenLayers/BaseTypes/Class.js **/ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for * full list of contributors). Published 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 */ /** * Constructor: OpenLayers.Class * Base class used to construct all other classes. Includes support for * multiple inheritance. * * This constructor is new in OpenLayers 2.5. At OpenLayers 3.0, the old * syntax for creating classes and dealing with inheritance * will be removed. * * To create a new OpenLayers-style class, use the following syntax: * (code) * var MyClass = OpenLayers.Class(prototype); * (end) * * To create a new OpenLayers-style class with multiple inheritance, use the * following syntax: * (code) * var MyClass = OpenLayers.Class(Class1, Class2, prototype); * (end) * * Note that instanceof reflection will only reveal Class1 as superclass. * */ OpenLayers.Class = function() { var len = arguments.length; var P = arguments[0]; var F = arguments[len-1]; var C = typeof F.initialize == "function" ? F.initialize : function(){ P.prototype.initialize.apply(this, arguments); }; if (len > 1) { var newArgs = [C, P].concat( Array.prototype.slice.call(arguments).slice(1, len-1), F); OpenLayers.inherit.apply(null, newArgs); } else { C.prototype = F; } return C; }; /** * Function: OpenLayers.inherit * * Parameters: * C - {Object} the class that inherits * P - {Object} the superclass to inherit from * * In addition to the mandatory C and P parameters, an arbitrary number of * objects can be passed, which will extend C. */ OpenLayers.inherit = function(C, P) { var F = function() {}; F.prototype = P.prototype; C.prototype = new F; var i, l, o; for(i=2, l=arguments.length; i replacement = context[a]; // 1 -> replacement = context[a][b]; // 2 -> replacement = context[a][b][c]; var subs = match.split(/\.+/); for (var i=0; i< subs.length; i++) { if (i == 0) { replacement = context; } if (replacement === undefined) { break; } replacement = replacement[subs[i]]; } if(typeof replacement == "function") { replacement = args ? replacement.apply(null, args) : replacement(); } // If replacement is undefined, return the string 'undefined'. // This is a workaround for a bugs in browsers not properly // dealing with non-participating groups in regular expressions: // http://blog.stevenlevithan.com/archives/npcg-javascript if (typeof replacement == 'undefined') { return 'undefined'; } else { return replacement; } }; return template.replace(OpenLayers.String.tokenRegEx, replacer); }, /** * Property: tokenRegEx * Used to find tokens in a string. * Examples: ${a}, ${a.b.c}, ${a-b}, ${5} */ tokenRegEx: /\$\{([\w.]+?)\}/g, /** * Property: numberRegEx * Used to test strings as numbers. */ numberRegEx: /^([+-]?)(?=\d|\.\d)\d*(\.\d*)?([Ee]([+-]?\d+))?$/, /** * APIFunction: isNumeric * Determine whether a string contains only a numeric value. * * Examples: * (code) * OpenLayers.String.isNumeric("6.02e23") // true * OpenLayers.String.isNumeric("12 dozen") // false * OpenLayers.String.isNumeric("4") // true * OpenLayers.String.isNumeric(" 4 ") // false * (end) * * Returns: * {Boolean} String contains only a number. */ isNumeric: function(value) { return OpenLayers.String.numberRegEx.test(value); }, /** * APIFunction: numericIf * Converts a string that appears to be a numeric value into a number. * * Parameters: * value - {String} * trimWhitespace - {Boolean} * * Returns: * {Number|String} a Number if the passed value is a number, a String * otherwise. */ numericIf: function(value, trimWhitespace) { var originalValue = value; if (trimWhitespace === true && value != null && value.replace) { value = value.replace(/^\s*|\s*$/g, ""); } return OpenLayers.String.isNumeric(value) ? parseFloat(value) : originalValue; } }; /** * Namespace: OpenLayers.Number * Contains convenience functions for manipulating numbers. */ OpenLayers.Number = { /** * Property: decimalSeparator * Decimal separator to use when formatting numbers. */ decimalSeparator: ".", /** * Property: thousandsSeparator * Thousands separator to use when formatting numbers. */ thousandsSeparator: ",", /** * APIFunction: limitSigDigs * Limit the number of significant digits on a float. * * Parameters: * num - {Float} * sig - {Integer} * * Returns: * {Float} The number, rounded to the specified number of significant * digits. */ limitSigDigs: function(num, sig) { var fig = 0; if (sig > 0) { fig = parseFloat(num.toPrecision(sig)); } return fig; }, /** * APIFunction: format * Formats a number for output. * * Parameters: * num - {Float} * dec - {Integer} Number of decimal places to round to. * Defaults to 0. Set to null to leave decimal places unchanged. * tsep - {String} Thousands separator. * Default is ",". * dsep - {String} Decimal separator. * Default is ".". * * Returns: * {String} A string representing the formatted number. */ format: function(num, dec, tsep, dsep) { dec = (typeof dec != "undefined") ? dec : 0; tsep = (typeof tsep != "undefined") ? tsep : OpenLayers.Number.thousandsSeparator; dsep = (typeof dsep != "undefined") ? dsep : OpenLayers.Number.decimalSeparator; if (dec != null) { num = parseFloat(num.toFixed(dec)); } var parts = num.toString().split("."); if (parts.length == 1 && dec == null) { // integer where we do not want to touch the decimals dec = 0; } var integer = parts[0]; if (tsep) { var thousands = /(-?[0-9]+)([0-9]{3})/; while(thousands.test(integer)) { integer = integer.replace(thousands, "$1" + tsep + "$2"); } } var str; if (dec == 0) { str = integer; } else { var rem = parts.length > 1 ? parts[1] : "0"; if (dec != null) { rem = rem + new Array(dec - rem.length + 1).join("0"); } str = integer + dsep + rem; } return str; }, /** * Method: zeroPad * Create a zero padded string optionally with a radix for casting numbers. * * Parameters: * num - {Number} The number to be zero padded. * len - {Number} The length of the string to be returned. * radix - {Number} An integer between 2 and 36 specifying the base to use * for representing numeric values. */ zeroPad: function(num, len, radix) { var str = num.toString(radix || 10); while (str.length < len) { str = "0" + str; } return str; } }; /** * Namespace: OpenLayers.Function * Contains convenience functions for function manipulation. */ OpenLayers.Function = { /** * APIFunction: bind * Bind a function to an object. Method to easily create closures with * 'this' altered. * * Parameters: * func - {Function} Input function. * object - {Object} The object to bind to the input function (as this). * * Returns: * {Function} A closure with 'this' set to the passed in object. */ bind: function(func, object) { // create a reference to all arguments past the second one var args = Array.prototype.slice.apply(arguments, [2]); return function() { // Push on any additional arguments from the actual function call. // These will come after those sent to the bind call. var newArgs = args.concat( Array.prototype.slice.apply(arguments, [0]) ); return func.apply(object, newArgs); }; }, /** * APIFunction: bindAsEventListener * Bind a function to an object, and configure it to receive the event * object as first parameter when called. * * Parameters: * func - {Function} Input function to serve as an event listener. * object - {Object} A reference to this. * * Returns: * {Function} */ bindAsEventListener: function(func, object) { return function(event) { return func.call(object, event || window.event); }; }, /** * APIFunction: False * A simple function to that just does "return false". We use this to * avoid attaching anonymous functions to DOM event handlers, which * causes "issues" on IE<8. * * Usage: * document.onclick = OpenLayers.Function.False; * * Returns: * {Boolean} */ False : function() { return false; }, /** * APIFunction: True * A simple function to that just does "return true". We use this to * avoid attaching anonymous functions to DOM event handlers, which * causes "issues" on IE<8. * * Usage: * document.onclick = OpenLayers.Function.True; * * Returns: * {Boolean} */ True : function() { return true; }, /** * APIFunction: Void * A reusable function that returns ``undefined``. * * Returns: * {undefined} */ Void: function() {} }; /** * Namespace: OpenLayers.Array * Contains convenience functions for array manipulation. */ OpenLayers.Array = { /** * APIMethod: filter * Filter an array. Provides the functionality of the * Array.prototype.filter extension to the ECMA-262 standard. Where * available, Array.prototype.filter will be used. * * Based on well known example from http://developer.mozilla.org/en/Core_JavaScript_1.5_Reference/Global_Objects/Array/filter * * Parameters: * array - {Array} The array to be filtered. This array is not mutated. * Elements added to this array by the callback will not be visited. * callback - {Function} A function that is called for each element in * the array. If this function returns true, the element will be * included in the return. The function will be called with three * arguments: the element in the array, the index of that element, and * the array itself. If the optional caller parameter is specified * the callback will be called with this set to caller. * caller - {Object} Optional object to be set as this when the callback * is called. * * Returns: * {Array} An array of elements from the passed in array for which the * callback returns true. */ filter: function(array, callback, caller) { var selected = []; if (Array.prototype.filter) { selected = array.filter(callback, caller); } else { var len = array.length; if (typeof callback != "function") { throw new TypeError(); } for(var i=0; i} A cached center location. This should not be * accessed directly. Use instead. */ centerLonLat: null, /** * Constructor: OpenLayers.Bounds * Construct a new bounds object. Coordinates can either be passed as four * arguments, or as a single argument. * * Parameters (four arguments): * left - {Number} The left bounds of the box. Note that for width * calculations, this is assumed to be less than the right value. * bottom - {Number} The bottom bounds of the box. Note that for height * calculations, this is assumed to be less than the top value. * right - {Number} The right bounds. * top - {Number} The top bounds. * * Parameters (single argument): * bounds - {Array(Number)} [left, bottom, right, top] */ initialize: function(left, bottom, right, top) { if (OpenLayers.Util.isArray(left)) { top = left[3]; right = left[2]; bottom = left[1]; left = left[0]; } if (left != null) { this.left = OpenLayers.Util.toFloat(left); } if (bottom != null) { this.bottom = OpenLayers.Util.toFloat(bottom); } if (right != null) { this.right = OpenLayers.Util.toFloat(right); } if (top != null) { this.top = OpenLayers.Util.toFloat(top); } }, /** * Method: clone * Create a cloned instance of this bounds. * * Returns: * {} A fresh copy of the bounds */ clone:function() { return new OpenLayers.Bounds(this.left, this.bottom, this.right, this.top); }, /** * Method: equals * Test a two bounds for equivalence. * * Parameters: * bounds - {} * * Returns: * {Boolean} The passed-in bounds object has the same left, * right, top, bottom components as this. Note that if bounds * passed in is null, returns false. */ equals:function(bounds) { var equals = false; if (bounds != null) { equals = ((this.left == bounds.left) && (this.right == bounds.right) && (this.top == bounds.top) && (this.bottom == bounds.bottom)); } return equals; }, /** * APIMethod: toString * Returns a string representation of the bounds object. * * Returns: * {String} String representation of bounds object. */ toString:function() { return [this.left, this.bottom, this.right, this.top].join(","); }, /** * APIMethod: toArray * Returns an array representation of the bounds object. * * Returns an array of left, bottom, right, top properties, or -- when the * optional parameter is true -- an array of the bottom, left, top, * right properties. * * Parameters: * reverseAxisOrder - {Boolean} Should we reverse the axis order? * * Returns: * {Array} array of left, bottom, right, top */ toArray: function(reverseAxisOrder) { if (reverseAxisOrder === true) { return [this.bottom, this.left, this.top, this.right]; } else { return [this.left, this.bottom, this.right, this.top]; } }, /** * APIMethod: toBBOX * Returns a boundingbox-string representation of the bounds object. * * Parameters: * decimal - {Integer} How many significant digits in the bbox coords? * Default is 6 * reverseAxisOrder - {Boolean} Should we reverse the axis order? * * Returns: * {String} Simple String representation of bounds object. * (e.g. "5,42,10,45") */ toBBOX:function(decimal, reverseAxisOrder) { if (decimal== null) { decimal = 6; } var mult = Math.pow(10, decimal); var xmin = Math.round(this.left * mult) / mult; var ymin = Math.round(this.bottom * mult) / mult; var xmax = Math.round(this.right * mult) / mult; var ymax = Math.round(this.top * mult) / mult; if (reverseAxisOrder === true) { return ymin + "," + xmin + "," + ymax + "," + xmax; } else { return xmin + "," + ymin + "," + xmax + "," + ymax; } }, /** * APIMethod: toGeometry * Create a new polygon geometry based on this bounds. * * Returns: * {} A new polygon with the coordinates * of this bounds. */ toGeometry: function() { return new OpenLayers.Geometry.Polygon([ new OpenLayers.Geometry.LinearRing([ new OpenLayers.Geometry.Point(this.left, this.bottom), new OpenLayers.Geometry.Point(this.right, this.bottom), new OpenLayers.Geometry.Point(this.right, this.top), new OpenLayers.Geometry.Point(this.left, this.top) ]) ]); }, /** * APIMethod: getWidth * Returns the width of the bounds. * * Returns: * {Float} The width of the bounds (right minus left). */ getWidth:function() { return (this.right - this.left); }, /** * APIMethod: getHeight * Returns the height of the bounds. * * Returns: * {Float} The height of the bounds (top minus bottom). */ getHeight:function() { return (this.top - this.bottom); }, /** * APIMethod: getSize * Returns an object of the bounds. * * Returns: * {} The size of the bounds. */ getSize:function() { return new OpenLayers.Size(this.getWidth(), this.getHeight()); }, /** * APIMethod: getCenterPixel * Returns the object which represents the center of the * bounds. * * Returns: * {} The center of the bounds in pixel space. */ getCenterPixel:function() { return new OpenLayers.Pixel( (this.left + this.right) / 2, (this.bottom + this.top) / 2); }, /** * APIMethod: getCenterLonLat * Returns the object which represents the center of the * bounds. * * Returns: * {} The center of the bounds in map space. */ getCenterLonLat:function() { if(!this.centerLonLat) { this.centerLonLat = new OpenLayers.LonLat( (this.left + this.right) / 2, (this.bottom + this.top) / 2 ); } return this.centerLonLat; }, /** * APIMethod: scale * Scales the bounds around a pixel or lonlat. Note that the new * bounds may return non-integer properties, even if a pixel * is passed. * * Parameters: * ratio - {Float} * origin - { or } * Default is center. * * Returns: * {} A new bounds that is scaled by ratio * from origin. */ scale: function(ratio, origin){ if(origin == null){ origin = this.getCenterLonLat(); } var origx,origy; // get origin coordinates if(origin.CLASS_NAME == "OpenLayers.LonLat"){ origx = origin.lon; origy = origin.lat; } else { origx = origin.x; origy = origin.y; } var left = (this.left - origx) * ratio + origx; var bottom = (this.bottom - origy) * ratio + origy; var right = (this.right - origx) * ratio + origx; var top = (this.top - origy) * ratio + origy; return new OpenLayers.Bounds(left, bottom, right, top); }, /** * APIMethod: add * Shifts the coordinates of the bound by the given horizontal and vertical * deltas. * * (start code) * var bounds = new OpenLayers.Bounds(0, 0, 10, 10); * bounds.toString(); * // => "0,0,10,10" * * bounds.add(-1.5, 4).toString(); * // => "-1.5,4,8.5,14" * (end) * * This method will throw a TypeError if it is passed null as an argument. * * Parameters: * x - {Float} horizontal delta * y - {Float} vertical delta * * Returns: * {} A new bounds whose coordinates are the same as * this, but shifted by the passed-in x and y values. */ add:function(x, y) { if ( (x == null) || (y == null) ) { throw new TypeError('Bounds.add cannot receive null values'); } return new OpenLayers.Bounds(this.left + x, this.bottom + y, this.right + x, this.top + y); }, /** * APIMethod: extend * Extend the bounds to include the , * or specified. * * Please note that this function assumes that left < right and * bottom < top. * * Parameters: * object - {, or * } The object to be included in the new bounds * object. */ extend:function(object) { if (object) { switch(object.CLASS_NAME) { case "OpenLayers.LonLat": this.extendXY(object.lon, object.lat); break; case "OpenLayers.Geometry.Point": this.extendXY(object.x, object.y); break; case "OpenLayers.Bounds": // clear cached center location this.centerLonLat = null; if ( (this.left == null) || (object.left < this.left)) { this.left = object.left; } if ( (this.bottom == null) || (object.bottom < this.bottom) ) { this.bottom = object.bottom; } if ( (this.right == null) || (object.right > this.right) ) { this.right = object.right; } if ( (this.top == null) || (object.top > this.top) ) { this.top = object.top; } break; } } }, /** * APIMethod: extendXY * Extend the bounds to include the XY coordinate specified. * * Parameters: * x - {number} The X part of the the coordinate. * y - {number} The Y part of the the coordinate. */ extendXY:function(x, y) { // clear cached center location this.centerLonLat = null; if ((this.left == null) || (x < this.left)) { this.left = x; } if ((this.bottom == null) || (y < this.bottom)) { this.bottom = y; } if ((this.right == null) || (x > this.right)) { this.right = x; } if ((this.top == null) || (y > this.top)) { this.top = y; } }, /** * APIMethod: containsLonLat * Returns whether the bounds object contains the given . * * Parameters: * ll - {|Object} OpenLayers.LonLat or an * object with a 'lon' and 'lat' properties. * options - {Object} Optional parameters * * Acceptable options: * inclusive - {Boolean} Whether or not to include the border. * Default is true. * worldBounds - {} If a worldBounds is provided, the * ll will be considered as contained if it exceeds the world bounds, * but can be wrapped around the dateline so it is contained by this * bounds. * * Returns: * {Boolean} The passed-in lonlat is within this bounds. */ containsLonLat: function(ll, options) { if (typeof options === "boolean") { options = {inclusive: options}; } options = options || {}; var contains = this.contains(ll.lon, ll.lat, options.inclusive), worldBounds = options.worldBounds; if (worldBounds && !contains) { var worldWidth = worldBounds.getWidth(); var worldCenterX = (worldBounds.left + worldBounds.right) / 2; var worldsAway = Math.round((ll.lon - worldCenterX) / worldWidth); contains = this.containsLonLat({ lon: ll.lon - worldsAway * worldWidth, lat: ll.lat }, {inclusive: options.inclusive}); } return contains; }, /** * APIMethod: containsPixel * Returns whether the bounds object contains the given . * * Parameters: * px - {} * inclusive - {Boolean} Whether or not to include the border. Default is * true. * * Returns: * {Boolean} The passed-in pixel is within this bounds. */ containsPixel:function(px, inclusive) { return this.contains(px.x, px.y, inclusive); }, /** * APIMethod: contains * Returns whether the bounds object contains the given x and y. * * Parameters: * x - {Float} * y - {Float} * inclusive - {Boolean} Whether or not to include the border. Default is * true. * * Returns: * {Boolean} Whether or not the passed-in coordinates are within this * bounds. */ contains:function(x, y, inclusive) { //set default if (inclusive == null) { inclusive = true; } if (x == null || y == null) { return false; } x = OpenLayers.Util.toFloat(x); y = OpenLayers.Util.toFloat(y); var contains = false; if (inclusive) { contains = ((x >= this.left) && (x <= this.right) && (y >= this.bottom) && (y <= this.top)); } else { contains = ((x > this.left) && (x < this.right) && (y > this.bottom) && (y < this.top)); } return contains; }, /** * APIMethod: intersectsBounds * Determine whether the target bounds intersects this bounds. Bounds are * considered intersecting if any of their edges intersect or if one * bounds contains the other. * * Parameters: * bounds - {} The target bounds. * options - {Object} Optional parameters. * * Acceptable options: * inclusive - {Boolean} Treat coincident borders as intersecting. Default * is true. If false, bounds that do not overlap but only touch at the * border will not be considered as intersecting. * worldBounds - {} If a worldBounds is provided, two * bounds will be considered as intersecting if they intersect when * shifted to within the world bounds. This applies only to bounds that * cross or are completely outside the world bounds. * * Returns: * {Boolean} The passed-in bounds object intersects this bounds. */ intersectsBounds:function(bounds, options) { if (typeof options === "boolean") { options = {inclusive: options}; } options = options || {}; if (options.worldBounds) { var self = this.wrapDateLine(options.worldBounds); bounds = bounds.wrapDateLine(options.worldBounds); } else { self = this; } if (options.inclusive == null) { options.inclusive = true; } var intersects = false; var mightTouch = ( self.left == bounds.right || self.right == bounds.left || self.top == bounds.bottom || self.bottom == bounds.top ); // if the two bounds only touch at an edge, and inclusive is false, // then the bounds don't *really* intersect. if (options.inclusive || !mightTouch) { // otherwise, if one of the boundaries even partially contains another, // inclusive of the edges, then they do intersect. var inBottom = ( ((bounds.bottom >= self.bottom) && (bounds.bottom <= self.top)) || ((self.bottom >= bounds.bottom) && (self.bottom <= bounds.top)) ); var inTop = ( ((bounds.top >= self.bottom) && (bounds.top <= self.top)) || ((self.top > bounds.bottom) && (self.top < bounds.top)) ); var inLeft = ( ((bounds.left >= self.left) && (bounds.left <= self.right)) || ((self.left >= bounds.left) && (self.left <= bounds.right)) ); var inRight = ( ((bounds.right >= self.left) && (bounds.right <= self.right)) || ((self.right >= bounds.left) && (self.right <= bounds.right)) ); intersects = ((inBottom || inTop) && (inLeft || inRight)); } // document me if (options.worldBounds && !intersects) { var world = options.worldBounds; var width = world.getWidth(); var selfCrosses = !world.containsBounds(self); var boundsCrosses = !world.containsBounds(bounds); if (selfCrosses && !boundsCrosses) { bounds = bounds.add(-width, 0); intersects = self.intersectsBounds(bounds, {inclusive: options.inclusive}); } else if (boundsCrosses && !selfCrosses) { self = self.add(-width, 0); intersects = bounds.intersectsBounds(self, {inclusive: options.inclusive}); } } return intersects; }, /** * APIMethod: containsBounds * Returns whether the bounds object contains the given . * * bounds - {} The target bounds. * partial - {Boolean} If any of the target corners is within this bounds * consider the bounds contained. Default is false. If false, the * entire target bounds must be contained within this bounds. * inclusive - {Boolean} Treat shared edges as contained. Default is * true. * * Returns: * {Boolean} The passed-in bounds object is contained within this bounds. */ containsBounds:function(bounds, partial, inclusive) { if (partial == null) { partial = false; } if (inclusive == null) { inclusive = true; } var bottomLeft = this.contains(bounds.left, bounds.bottom, inclusive); var bottomRight = this.contains(bounds.right, bounds.bottom, inclusive); var topLeft = this.contains(bounds.left, bounds.top, inclusive); var topRight = this.contains(bounds.right, bounds.top, inclusive); return (partial) ? (bottomLeft || bottomRight || topLeft || topRight) : (bottomLeft && bottomRight && topLeft && topRight); }, /** * APIMethod: determineQuadrant * Returns the the quadrant ("br", "tr", "tl", "bl") in which the given * lies. * * Parameters: * lonlat - {} * * Returns: * {String} The quadrant ("br" "tr" "tl" "bl") of the bounds in which the * coordinate lies. */ determineQuadrant: function(lonlat) { var quadrant = ""; var center = this.getCenterLonLat(); quadrant += (lonlat.lat < center.lat) ? "b" : "t"; quadrant += (lonlat.lon < center.lon) ? "l" : "r"; return quadrant; }, /** * APIMethod: transform * Transform the Bounds object from source to dest. * * Parameters: * source - {} Source projection. * dest - {} Destination projection. * * Returns: * {} Itself, for use in chaining operations. */ transform: function(source, dest) { // clear cached center location this.centerLonLat = null; var ll = OpenLayers.Projection.transform( {'x': this.left, 'y': this.bottom}, source, dest); var lr = OpenLayers.Projection.transform( {'x': this.right, 'y': this.bottom}, source, dest); var ul = OpenLayers.Projection.transform( {'x': this.left, 'y': this.top}, source, dest); var ur = OpenLayers.Projection.transform( {'x': this.right, 'y': this.top}, source, dest); this.left = Math.min(ll.x, ul.x); this.bottom = Math.min(ll.y, lr.y); this.right = Math.max(lr.x, ur.x); this.top = Math.max(ul.y, ur.y); return this; }, /** * APIMethod: wrapDateLine * Wraps the bounds object around the dateline. * * Parameters: * maxExtent - {} * options - {Object} Some possible options are: * * Allowed Options: * leftTolerance - {float} Allow for a margin of error * with the 'left' value of this * bound. * Default is 0. * rightTolerance - {float} Allow for a margin of error * with the 'right' value of * this bound. * Default is 0. * * Returns: * {} A copy of this bounds, but wrapped around the * "dateline" (as specified by the borders of * maxExtent). Note that this function only returns * a different bounds value if this bounds is * *entirely* outside of the maxExtent. If this * bounds straddles the dateline (is part in/part * out of maxExtent), the returned bounds will always * cross the left edge of the given maxExtent. *. */ wrapDateLine: function(maxExtent, options) { options = options || {}; var leftTolerance = options.leftTolerance || 0; var rightTolerance = options.rightTolerance || 0; var newBounds = this.clone(); if (maxExtent) { var width = maxExtent.getWidth(); //shift right? while (newBounds.left < maxExtent.left && newBounds.right - rightTolerance <= maxExtent.left ) { newBounds = newBounds.add(width, 0); } //shift left? while (newBounds.left + leftTolerance >= maxExtent.right && newBounds.right > maxExtent.right ) { newBounds = newBounds.add(-width, 0); } // crosses right only? force left var newLeft = newBounds.left + leftTolerance; if (newLeft < maxExtent.right && newLeft > maxExtent.left && newBounds.right - rightTolerance > maxExtent.right) { newBounds = newBounds.add(-width, 0); } } return newBounds; }, CLASS_NAME: "OpenLayers.Bounds" }); /** * APIFunction: fromString * Alternative constructor that builds a new OpenLayers.Bounds from a * parameter string. * * (begin code) * OpenLayers.Bounds.fromString("5,42,10,45"); * // => equivalent to ... * new OpenLayers.Bounds(5, 42, 10, 45); * (end) * * Parameters: * str - {String} Comma-separated bounds string. (e.g. "5,42,10,45") * reverseAxisOrder - {Boolean} Does the string use reverse axis order? * * Returns: * {} New bounds object built from the * passed-in String. */ OpenLayers.Bounds.fromString = function(str, reverseAxisOrder) { var bounds = str.split(","); return OpenLayers.Bounds.fromArray(bounds, reverseAxisOrder); }; /** * APIFunction: fromArray * Alternative constructor that builds a new OpenLayers.Bounds from an array. * * (begin code) * OpenLayers.Bounds.fromArray( [5, 42, 10, 45] ); * // => equivalent to ... * new OpenLayers.Bounds(5, 42, 10, 45); * (end) * * Parameters: * bbox - {Array(Float)} Array of bounds values (e.g. [5,42,10,45]) * reverseAxisOrder - {Boolean} Does the array use reverse axis order? * * Returns: * {} New bounds object built from the passed-in Array. */ OpenLayers.Bounds.fromArray = function(bbox, reverseAxisOrder) { return reverseAxisOrder === true ? new OpenLayers.Bounds(bbox[1], bbox[0], bbox[3], bbox[2]) : new OpenLayers.Bounds(bbox[0], bbox[1], bbox[2], bbox[3]); }; /** * APIFunction: fromSize * Alternative constructor that builds a new OpenLayers.Bounds from a size. * * (begin code) * OpenLayers.Bounds.fromSize( new OpenLayers.Size(10, 20) ); * // => equivalent to ... * new OpenLayers.Bounds(0, 20, 10, 0); * (end) * * Parameters: * size - { or Object} or an object with * both 'w' and 'h' properties. * * Returns: * {} New bounds object built from the passed-in size. */ OpenLayers.Bounds.fromSize = function(size) { return new OpenLayers.Bounds(0, size.h, size.w, 0); }; /** * Function: oppositeQuadrant * Get the opposite quadrant for a given quadrant string. * * (begin code) * OpenLayers.Bounds.oppositeQuadrant( "tl" ); * // => "br" * * OpenLayers.Bounds.oppositeQuadrant( "tr" ); * // => "bl" * (end) * * Parameters: * quadrant - {String} two character quadrant shortstring * * Returns: * {String} The opposing quadrant ("br" "tr" "tl" "bl"). For Example, if * you pass in "bl" it returns "tr", if you pass in "br" it * returns "tl", etc. */ OpenLayers.Bounds.oppositeQuadrant = function(quadrant) { var opp = ""; opp += (quadrant.charAt(0) == 't') ? 'b' : 't'; opp += (quadrant.charAt(1) == 'l') ? 'r' : 'l'; return opp; }; /** FILE: OpenLayers/BaseTypes/Element.js **/ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for * full list of contributors). Published 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 */ /** * Namespace: OpenLayers.Element */ OpenLayers.Element = { /** * APIFunction: visible * * Parameters: * element - {DOMElement} * * Returns: * {Boolean} Is the element visible? */ visible: function(element) { return OpenLayers.Util.getElement(element).style.display != 'none'; }, /** * APIFunction: toggle * Toggle the visibility of element(s) passed in * * Parameters: * element - {DOMElement} Actually user can pass any number of elements */ toggle: function() { for (var i=0, len=arguments.length; i"lon=5,lat=42") */ toString:function() { return ("lon=" + this.lon + ",lat=" + this.lat); }, /** * APIMethod: toShortString * * Returns: * {String} Shortened String representation of OpenLayers.LonLat object. * (e.g. "5, 42") */ toShortString:function() { return (this.lon + ", " + this.lat); }, /** * APIMethod: clone * * Returns: * {} New OpenLayers.LonLat object with the same lon * and lat values */ clone:function() { return new OpenLayers.LonLat(this.lon, this.lat); }, /** * APIMethod: add * * Parameters: * lon - {Float} * lat - {Float} * * Returns: * {} A new OpenLayers.LonLat object with the lon and * lat passed-in added to this's. */ add:function(lon, lat) { if ( (lon == null) || (lat == null) ) { throw new TypeError('LonLat.add cannot receive null values'); } return new OpenLayers.LonLat(this.lon + OpenLayers.Util.toFloat(lon), this.lat + OpenLayers.Util.toFloat(lat)); }, /** * APIMethod: equals * * Parameters: * ll - {} * * Returns: * {Boolean} Boolean value indicating whether the passed-in * object has the same lon and lat * components as this. * Note: if ll passed in is null, returns false */ equals:function(ll) { var equals = false; if (ll != null) { equals = ((this.lon == ll.lon && this.lat == ll.lat) || (isNaN(this.lon) && isNaN(this.lat) && isNaN(ll.lon) && isNaN(ll.lat))); } return equals; }, /** * APIMethod: transform * Transform the LonLat object from source to dest. This transformation is * *in place*: if you want a *new* lonlat, use .clone() first. * * Parameters: * source - {} Source projection. * dest - {} Destination projection. * * Returns: * {} Itself, for use in chaining operations. */ transform: function(source, dest) { var point = OpenLayers.Projection.transform( {'x': this.lon, 'y': this.lat}, source, dest); this.lon = point.x; this.lat = point.y; return this; }, /** * APIMethod: wrapDateLine * * Parameters: * maxExtent - {} * * Returns: * {} A copy of this lonlat, but wrapped around the * "dateline" (as specified by the borders of * maxExtent) */ wrapDateLine: function(maxExtent) { var newLonLat = this.clone(); if (maxExtent) { //shift right? while (newLonLat.lon < maxExtent.left) { newLonLat.lon += maxExtent.getWidth(); } //shift left? while (newLonLat.lon > maxExtent.right) { newLonLat.lon -= maxExtent.getWidth(); } } return newLonLat; }, CLASS_NAME: "OpenLayers.LonLat" }); /** * Function: fromString * Alternative constructor that builds a new from a * parameter string * * Parameters: * str - {String} Comma-separated Lon,Lat coordinate string. * (e.g. "5,40") * * Returns: * {} New object built from the * passed-in String. */ OpenLayers.LonLat.fromString = function(str) { var pair = str.split(","); return new OpenLayers.LonLat(pair[0], pair[1]); }; /** * Function: fromArray * Alternative constructor that builds a new from an * array of two numbers that represent lon- and lat-values. * * Parameters: * arr - {Array(Float)} Array of lon/lat values (e.g. [5,-42]) * * Returns: * {} New object built from the * passed-in array. */ OpenLayers.LonLat.fromArray = function(arr) { var gotArr = OpenLayers.Util.isArray(arr), lon = gotArr && arr[0], lat = gotArr && arr[1]; return new OpenLayers.LonLat(lon, lat); }; /** FILE: OpenLayers/BaseTypes/Pixel.js **/ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for * full list of contributors). Published 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.Pixel * This class represents a screen coordinate, in x and y coordinates */ OpenLayers.Pixel = OpenLayers.Class({ /** * APIProperty: x * {Number} The x coordinate */ x: 0.0, /** * APIProperty: y * {Number} The y coordinate */ y: 0.0, /** * Constructor: OpenLayers.Pixel * Create a new OpenLayers.Pixel instance * * Parameters: * x - {Number} The x coordinate * y - {Number} The y coordinate * * Returns: * An instance of OpenLayers.Pixel */ initialize: function(x, y) { this.x = parseFloat(x); this.y = parseFloat(y); }, /** * Method: toString * Cast this object into a string * * Returns: * {String} The string representation of Pixel. ex: "x=200.4,y=242.2" */ toString:function() { return ("x=" + this.x + ",y=" + this.y); }, /** * APIMethod: clone * Return a clone of this pixel object * * Returns: * {} A clone pixel */ clone:function() { return new OpenLayers.Pixel(this.x, this.y); }, /** * APIMethod: equals * Determine whether one pixel is equivalent to another * * Parameters: * px - {|Object} An OpenLayers.Pixel or an object with * a 'x' and 'y' properties. * * Returns: * {Boolean} The point passed in as parameter is equal to this. Note that * if px passed in is null, returns false. */ equals:function(px) { var equals = false; if (px != null) { equals = ((this.x == px.x && this.y == px.y) || (isNaN(this.x) && isNaN(this.y) && isNaN(px.x) && isNaN(px.y))); } return equals; }, /** * APIMethod: distanceTo * Returns the distance to the pixel point passed in as a parameter. * * Parameters: * px - {} * * Returns: * {Float} The pixel point passed in as parameter to calculate the * distance to. */ distanceTo:function(px) { return Math.sqrt( Math.pow(this.x - px.x, 2) + Math.pow(this.y - px.y, 2) ); }, /** * APIMethod: add * * Parameters: * x - {Integer} * y - {Integer} * * Returns: * {} A new Pixel with this pixel's x&y augmented by the * values passed in. */ add:function(x, y) { if ( (x == null) || (y == null) ) { throw new TypeError('Pixel.add cannot receive null values'); } return new OpenLayers.Pixel(this.x + x, this.y + y); }, /** * APIMethod: offset * * Parameters * px - {|Object} An OpenLayers.Pixel or an object with * a 'x' and 'y' properties. * * Returns: * {} A new Pixel with this pixel's x&y augmented by the * x&y values of the pixel passed in. */ offset:function(px) { var newPx = this.clone(); if (px) { newPx = this.add(px.x, px.y); } return newPx; }, CLASS_NAME: "OpenLayers.Pixel" }); /** FILE: OpenLayers/BaseTypes/Size.js **/ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for * full list of contributors). Published 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.Size * Instances of this class represent a width/height pair */ OpenLayers.Size = OpenLayers.Class({ /** * APIProperty: w * {Number} width */ w: 0.0, /** * APIProperty: h * {Number} height */ h: 0.0, /** * Constructor: OpenLayers.Size * Create an instance of OpenLayers.Size * * Parameters: * w - {Number} width * h - {Number} height */ initialize: function(w, h) { this.w = parseFloat(w); this.h = parseFloat(h); }, /** * Method: toString * Return the string representation of a size object * * Returns: * {String} The string representation of OpenLayers.Size object. * (e.g. "w=55,h=66") */ toString:function() { return ("w=" + this.w + ",h=" + this.h); }, /** * APIMethod: clone * Create a clone of this size object * * Returns: * {} A new OpenLayers.Size object with the same w and h * values */ clone:function() { return new OpenLayers.Size(this.w, this.h); }, /** * * APIMethod: equals * Determine where this size is equal to another * * Parameters: * sz - {|Object} An OpenLayers.Size or an object with * a 'w' and 'h' properties. * * Returns: * {Boolean} The passed in size has the same h and w properties as this one. * Note that if sz passed in is null, returns false. */ equals:function(sz) { var equals = false; 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" }); /** FILE: 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 method to set this value and the method to * retrieve it. */ code: null, /** * APIProperty: defaultCode * {String} Default language to use when a specific language can't be * found. Default is "en". */ defaultCode: "en", /** * APIFunction: getCode * Get the current language code. * * Returns: * {String} The current language code. */ getCode: function() { if(!OpenLayers.Lang.code) { OpenLayers.Lang.setCode(); } return OpenLayers.Lang.code; }, /** * APIFunction: setCode * Set the language code for string translation. This code is used by * the method. * * Parameters: * code - {String} These codes follow the IETF recommendations at * http://www.ietf.org/rfc/rfc3066.txt. If no value is set, the * browser's language setting will be tested. If no * dictionary exists for the code, the * will be used. */ setCode: function(code) { var lang; if(!code) { code = (OpenLayers.BROWSER_NAME == "msie") ? navigator.userLanguage : navigator.language; } var parts = code.split('-'); parts[0] = parts[0].toLowerCase(); if(typeof OpenLayers.Lang[parts[0]] == "object") { lang = parts[0]; } // check for regional extensions if(parts[1]) { var testLang = parts[0] + '-' + parts[1].toUpperCase(); if(typeof OpenLayers.Lang[testLang] == "object") { lang = testLang; } } if(!lang) { OpenLayers.Console.warn( 'Failed to find OpenLayers.Lang.' + parts.join("-") + ' dictionary, falling back to default language' ); lang = OpenLayers.Lang.defaultCode; } OpenLayers.Lang.code = lang; }, /** * APIMethod: translate * Looks up a key from a dictionary based on the current language string. * The value of will be used to determine the appropriate * dictionary. Dictionaries are stored in . * * Parameters: * key - {String} The key for an i18n string value in the dictionary. * context - {Object} Optional context to be used with * . * * Returns: * {String} A internationalized string. */ translate: function(key, context) { var dictionary = OpenLayers.Lang[OpenLayers.Lang.getCode()]; var message = dictionary && dictionary[key]; if(!message) { // Message not found, fall back to message key message = key; } if(context) { message = OpenLayers.String.format(message, context); } return message; } }; /** * APIMethod: OpenLayers.i18n * Alias for . Looks up a key from a dictionary * based on the current language string. The value of * will be used to determine the appropriate * dictionary. Dictionaries are stored in . * * Parameters: * key - {String} The key for an i18n string value in the dictionary. * context - {Object} Optional context to be used with * . * * Returns: * {String} A internationalized string. */ OpenLayers.i18n = OpenLayers.Lang.translate; /** FILE: OpenLayers/Util.js **/ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for * full list of contributors). Published 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.js * @requires OpenLayers/BaseTypes/Bounds.js * @requires OpenLayers/BaseTypes/Element.js * @requires OpenLayers/BaseTypes/LonLat.js * @requires OpenLayers/BaseTypes/Pixel.js * @requires OpenLayers/BaseTypes/Size.js * @requires OpenLayers/Lang.js */ /** * Namespace: Util */ OpenLayers.Util = OpenLayers.Util || {}; /** * Function: getElement * This is the old $() from prototype * * Parameters: * e - {String or DOMElement or Window} * * Returns: * {Array(DOMElement) or DOMElement} */ OpenLayers.Util.getElement = function() { var elements = []; for (var i=0, len=arguments.length; i= 0; i--) { if(array[i] == item) { array.splice(i,1); //break;more than once?? } } return array; }; /** * Function: indexOf * Seems to exist already in FF, but not in MOZ. * * Parameters: * array - {Array} * obj - {*} * * Returns: * {Integer} The index at which the first object was found in the array. * If not found, returns -1. */ OpenLayers.Util.indexOf = function(array, obj) { // use the build-in function if available. if (typeof array.indexOf == "function") { return array.indexOf(obj); } else { for (var i = 0, len = array.length; i < len; i++) { if (array[i] == obj) { return i; } } return -1; } }; /** * Property: dotless * {RegExp} * Compiled regular expression to match dots ("."). This is used for replacing * dots in identifiers. Because object identifiers are frequently used for * DOM element identifiers by the library, we avoid using dots to make for * more sensible CSS selectors. * * TODO: Use a module pattern to avoid bloating the API with stuff like this. */ OpenLayers.Util.dotless = /\./g; /** * Function: modifyDOMElement * * Modifies many properties of a DOM element all at once. Passing in * null to an individual parameter will avoid setting the attribute. * * Parameters: * element - {DOMElement} DOM element to modify. * id - {String} The element id attribute to set. Note that dots (".") will be * replaced with underscore ("_") in setting the element id. * px - {|Object} The element left and top position, * OpenLayers.Pixel or an object with * a 'x' and 'y' properties. * sz - {|Object} The element width and height, * OpenLayers.Size or an object with a * 'w' and 'h' properties. * position - {String} The position attribute. eg: absolute, * relative, etc. * border - {String} The style.border attribute. eg: * solid black 2px * overflow - {String} The style.overview attribute. * opacity - {Float} Fractional value (0.0 - 1.0) */ OpenLayers.Util.modifyDOMElement = function(element, id, px, sz, position, border, overflow, opacity) { if (id) { element.id = id.replace(OpenLayers.Util.dotless, "_"); } if (px) { element.style.left = px.x + "px"; element.style.top = px.y + "px"; } if (sz) { element.style.width = sz.w + "px"; element.style.height = sz.h + "px"; } if (position) { element.style.position = position; } if (border) { element.style.border = border; } if (overflow) { element.style.overflow = overflow; } if (parseFloat(opacity) >= 0.0 && parseFloat(opacity) < 1.0) { element.style.filter = 'alpha(opacity=' + (opacity * 100) + ')'; element.style.opacity = opacity; } else if (parseFloat(opacity) == 1.0) { element.style.filter = ''; element.style.opacity = ''; } }; /** * Function: createDiv * Creates a new div and optionally set some standard attributes. * Null may be passed to each parameter if you do not wish to * set a particular attribute. * Note - zIndex is NOT set on the resulting div. * * Parameters: * id - {String} An identifier for this element. If no id is * passed an identifier will be created * automatically. Note that dots (".") will be replaced with * underscore ("_") when generating ids. * px - {|Object} The element left and top position, * OpenLayers.Pixel or an object with * a 'x' and 'y' properties. * sz - {|Object} The element width and height, * OpenLayers.Size or an object with a * 'w' and 'h' properties. * imgURL - {String} A url pointing to an image to use as a * background image. * position - {String} The style.position value. eg: absolute, * relative etc. * border - {String} The the style.border value. * eg: 2px solid black * overflow - {String} The style.overflow value. Eg. hidden * opacity - {Float} Fractional value (0.0 - 1.0) * * Returns: * {DOMElement} A DOM Div created with the specified attributes. */ OpenLayers.Util.createDiv = function(id, px, sz, imgURL, position, border, overflow, opacity) { var dom = document.createElement('div'); if (imgURL) { dom.style.backgroundImage = 'url(' + imgURL + ')'; } //set generic properties if (!id) { id = OpenLayers.Util.createUniqueID("OpenLayersDiv"); } if (!position) { position = "absolute"; } OpenLayers.Util.modifyDOMElement(dom, id, px, sz, position, border, overflow, opacity); return dom; }; /** * Function: createImage * Creates an img element with specific attribute values. * * Parameters: * id - {String} The id field for the img. If none assigned one will be * automatically generated. * px - {|Object} The element left and top position, * OpenLayers.Pixel or an object with * a 'x' and 'y' properties. * sz - {|Object} The element width and height, * OpenLayers.Size or an object with a * 'w' and 'h' properties. * imgURL - {String} The url to use as the image source. * position - {String} The style.position value. * border - {String} The border to place around the image. * opacity - {Float} Fractional value (0.0 - 1.0) * delayDisplay - {Boolean} If true waits until the image has been * loaded. * * Returns: * {DOMElement} A DOM Image created with the specified attributes. */ OpenLayers.Util.createImage = function(id, px, sz, imgURL, position, border, opacity, delayDisplay) { var image = document.createElement("img"); //set generic properties if (!id) { id = OpenLayers.Util.createUniqueID("OpenLayersDiv"); } if (!position) { position = "relative"; } OpenLayers.Util.modifyDOMElement(image, id, px, sz, position, border, null, opacity); if (delayDisplay) { image.style.display = "none"; function display() { image.style.display = ""; OpenLayers.Event.stopObservingElement(image); } OpenLayers.Event.observe(image, "load", display); OpenLayers.Event.observe(image, "error", display); } //set special properties image.style.alt = id; image.galleryImg = "no"; if (imgURL) { image.src = imgURL; } return image; }; /** * Property: IMAGE_RELOAD_ATTEMPTS * {Integer} How many times should we try to reload an image before giving up? * Default is 0 */ OpenLayers.IMAGE_RELOAD_ATTEMPTS = 0; /** * Property: alphaHackNeeded * {Boolean} true if the png alpha hack is necessary and possible, false otherwise. */ OpenLayers.Util.alphaHackNeeded = null; /** * Function: alphaHack * Checks whether it's necessary (and possible) to use the png alpha * hack which allows alpha transparency for png images under Internet * Explorer. * * Returns: * {Boolean} true if the png alpha hack is necessary and possible, false otherwise. */ OpenLayers.Util.alphaHack = function() { if (OpenLayers.Util.alphaHackNeeded == null) { var arVersion = navigator.appVersion.split("MSIE"); var version = parseFloat(arVersion[1]); var filter = false; // IEs4Lin dies when trying to access document.body.filters, because // the property is there, but requires a DLL that can't be provided. This // means that we need to wrap this in a try/catch so that this can // continue. try { filter = !!(document.body.filters); } catch (e) {} OpenLayers.Util.alphaHackNeeded = (filter && (version >= 5.5) && (version < 7)); } return OpenLayers.Util.alphaHackNeeded; }; /** * Function: modifyAlphaImageDiv * * Parameters: * div - {DOMElement} Div containing Alpha-adjusted Image * id - {String} * px - {|Object} OpenLayers.Pixel or an object with * a 'x' and 'y' properties. * sz - {|Object} OpenLayers.Size or an object with * a 'w' and 'h' properties. * imgURL - {String} * position - {String} * border - {String} * sizing - {String} 'crop', 'scale', or 'image'. Default is "scale" * opacity - {Float} Fractional value (0.0 - 1.0) */ OpenLayers.Util.modifyAlphaImageDiv = function(div, id, px, sz, imgURL, position, border, sizing, opacity) { OpenLayers.Util.modifyDOMElement(div, id, px, sz, position, null, null, opacity); var img = div.childNodes[0]; if (imgURL) { img.src = imgURL; } OpenLayers.Util.modifyDOMElement(img, div.id + "_innerImage", null, sz, "relative", border); if (OpenLayers.Util.alphaHack()) { if(div.style.display != "none") { div.style.display = "inline-block"; } if (sizing == null) { sizing = "scale"; } div.style.filter = "progid:DXImageTransform.Microsoft" + ".AlphaImageLoader(src='" + img.src + "', " + "sizingMethod='" + sizing + "')"; if (parseFloat(div.style.opacity) >= 0.0 && parseFloat(div.style.opacity) < 1.0) { div.style.filter += " alpha(opacity=" + div.style.opacity * 100 + ")"; } img.style.filter = "alpha(opacity=0)"; } }; /** * Function: createAlphaImageDiv * * Parameters: * id - {String} * px - {|Object} OpenLayers.Pixel or an object with * a 'x' and 'y' properties. * sz - {|Object} OpenLayers.Size or an object with * a 'w' and 'h' properties. * imgURL - {String} * position - {String} * border - {String} * sizing - {String} 'crop', 'scale', or 'image'. Default is "scale" * opacity - {Float} Fractional value (0.0 - 1.0) * delayDisplay - {Boolean} If true waits until the image has been * loaded. * * Returns: * {DOMElement} A DOM Div created with a DOM Image inside it. If the hack is * needed for transparency in IE, it is added. */ OpenLayers.Util.createAlphaImageDiv = function(id, px, sz, imgURL, position, border, sizing, opacity, delayDisplay) { var div = OpenLayers.Util.createDiv(); var img = OpenLayers.Util.createImage(null, null, null, null, null, null, null, delayDisplay); img.className = "olAlphaImg"; div.appendChild(img); OpenLayers.Util.modifyAlphaImageDiv(div, id, px, sz, imgURL, position, border, sizing, opacity); return div; }; /** * Function: upperCaseObject * Creates a new hashtable and copies over all the keys from the * passed-in object, but storing them under an uppercased * version of the key at which they were stored. * * Parameters: * object - {Object} * * Returns: * {Object} A new Object with all the same keys but uppercased */ OpenLayers.Util.upperCaseObject = function (object) { var uObject = {}; for (var key in object) { uObject[key.toUpperCase()] = object[key]; } return uObject; }; /** * Function: applyDefaults * Takes an object and copies any properties that don't exist from * another properties, by analogy with OpenLayers.Util.extend() from * Prototype.js. * * Parameters: * to - {Object} The destination object. * from - {Object} The source object. Any properties of this object that * are undefined in the to object will be set on the to object. * * Returns: * {Object} A reference to the to object. Note that the to argument is modified * in place and returned by this function. */ OpenLayers.Util.applyDefaults = function (to, from) { to = to || {}; /* * FF/Windows < 2.0.0.13 reports "Illegal operation on WrappedNative * prototype object" when calling hawOwnProperty if the source object is an * instance of window.Event. */ var fromIsEvt = typeof window.Event == "function" && from instanceof window.Event; for (var key in from) { if (to[key] === undefined || (!fromIsEvt && from.hasOwnProperty && from.hasOwnProperty(key) && !to.hasOwnProperty(key))) { to[key] = from[key]; } } /** * IE doesn't include the toString property when iterating over an object's * properties with the for(property in object) syntax. Explicitly check if * the source has its own toString property. */ if(!fromIsEvt && from && from.hasOwnProperty && from.hasOwnProperty('toString') && !to.hasOwnProperty('toString')) { to.toString = from.toString; } return to; }; /** * Function: getParameterString * * Parameters: * params - {Object} * * Returns: * {String} A concatenation of the properties of an object in * http parameter notation. * (ex. "key1=value1&key2=value2&key3=value3") * If a parameter is actually a list, that parameter will then * be set to a comma-seperated list of values (foo,bar) instead * of being URL escaped (foo%3Abar). */ OpenLayers.Util.getParameterString = function(params) { var paramsArray = []; for (var key in params) { var value = params[key]; if ((value != null) && (typeof value != 'function')) { var encodedValue; if (typeof value == 'object' && value.constructor == Array) { /* value is an array; encode items and separate with "," */ var encodedItemArray = []; var item; for (var itemIndex=0, len=value.length; itemIndex} (or any object with both .lat, .lon properties) * p2 - {} (or any object with both .lat, .lon properties) * * Returns: * {Float} The distance (in km) between the two input points as measured on an * ellipsoid. Note that the input point objects must be in geographic * coordinates (decimal degrees) and the return distance is in kilometers. */ OpenLayers.Util.distVincenty = function(p1, p2) { var ct = OpenLayers.Util.VincentyConstants; var a = ct.a, b = ct.b, f = ct.f; var L = OpenLayers.Util.rad(p2.lon - p1.lon); var U1 = Math.atan((1-f) * Math.tan(OpenLayers.Util.rad(p1.lat))); var U2 = Math.atan((1-f) * Math.tan(OpenLayers.Util.rad(p2.lat))); var sinU1 = Math.sin(U1), cosU1 = Math.cos(U1); var sinU2 = Math.sin(U2), cosU2 = Math.cos(U2); var lambda = L, lambdaP = 2*Math.PI; var iterLimit = 20; while (Math.abs(lambda-lambdaP) > 1e-12 && --iterLimit>0) { var sinLambda = Math.sin(lambda), cosLambda = Math.cos(lambda); var sinSigma = Math.sqrt((cosU2*sinLambda) * (cosU2*sinLambda) + (cosU1*sinU2-sinU1*cosU2*cosLambda) * (cosU1*sinU2-sinU1*cosU2*cosLambda)); if (sinSigma==0) { return 0; // co-incident points } var cosSigma = sinU1*sinU2 + cosU1*cosU2*cosLambda; var sigma = Math.atan2(sinSigma, cosSigma); var alpha = Math.asin(cosU1 * cosU2 * sinLambda / sinSigma); var cosSqAlpha = Math.cos(alpha) * Math.cos(alpha); var cos2SigmaM = cosSigma - 2*sinU1*sinU2/cosSqAlpha; var C = f/16*cosSqAlpha*(4+f*(4-3*cosSqAlpha)); lambdaP = lambda; lambda = L + (1-C) * f * Math.sin(alpha) * (sigma + C*sinSigma*(cos2SigmaM+C*cosSigma*(-1+2*cos2SigmaM*cos2SigmaM))); } if (iterLimit==0) { return NaN; // formula failed to converge } var uSq = cosSqAlpha * (a*a - b*b) / (b*b); var A = 1 + uSq/16384*(4096+uSq*(-768+uSq*(320-175*uSq))); var B = uSq/1024 * (256+uSq*(-128+uSq*(74-47*uSq))); var deltaSigma = B*sinSigma*(cos2SigmaM+B/4*(cosSigma*(-1+2*cos2SigmaM*cos2SigmaM)- B/6*cos2SigmaM*(-3+4*sinSigma*sinSigma)*(-3+4*cos2SigmaM*cos2SigmaM))); var s = b*A*(sigma-deltaSigma); var d = s.toFixed(3)/1000; // round to 1mm precision return d; }; /** * APIFunction: destinationVincenty * Calculate destination point given start point lat/long (numeric degrees), * bearing (numeric degrees) & distance (in m). * Adapted from Chris Veness work, see * http://www.movable-type.co.uk/scripts/latlong-vincenty-direct.html * * Parameters: * lonlat - {} (or any object with both .lat, .lon * properties) The start point. * brng - {Float} The bearing (degrees). * dist - {Float} The ground distance (meters). * * Returns: * {} The destination point. */ OpenLayers.Util.destinationVincenty = function(lonlat, brng, dist) { var u = OpenLayers.Util; var ct = u.VincentyConstants; var a = ct.a, b = ct.b, f = ct.f; var lon1 = lonlat.lon; var lat1 = lonlat.lat; var s = dist; var alpha1 = u.rad(brng); var sinAlpha1 = Math.sin(alpha1); var cosAlpha1 = Math.cos(alpha1); var tanU1 = (1-f) * Math.tan(u.rad(lat1)); var cosU1 = 1 / Math.sqrt((1 + tanU1*tanU1)), sinU1 = tanU1*cosU1; var sigma1 = Math.atan2(tanU1, cosAlpha1); var sinAlpha = cosU1 * sinAlpha1; var cosSqAlpha = 1 - sinAlpha*sinAlpha; var uSq = cosSqAlpha * (a*a - b*b) / (b*b); var A = 1 + uSq/16384*(4096+uSq*(-768+uSq*(320-175*uSq))); var B = uSq/1024 * (256+uSq*(-128+uSq*(74-47*uSq))); var sigma = s / (b*A), sigmaP = 2*Math.PI; while (Math.abs(sigma-sigmaP) > 1e-12) { var cos2SigmaM = Math.cos(2*sigma1 + sigma); var sinSigma = Math.sin(sigma); var cosSigma = Math.cos(sigma); var deltaSigma = B*sinSigma*(cos2SigmaM+B/4*(cosSigma*(-1+2*cos2SigmaM*cos2SigmaM)- B/6*cos2SigmaM*(-3+4*sinSigma*sinSigma)*(-3+4*cos2SigmaM*cos2SigmaM))); sigmaP = sigma; sigma = s / (b*A) + deltaSigma; } var tmp = sinU1*sinSigma - cosU1*cosSigma*cosAlpha1; var lat2 = Math.atan2(sinU1*cosSigma + cosU1*sinSigma*cosAlpha1, (1-f)*Math.sqrt(sinAlpha*sinAlpha + tmp*tmp)); var lambda = Math.atan2(sinSigma*sinAlpha1, cosU1*cosSigma - sinU1*sinSigma*cosAlpha1); var C = f/16*cosSqAlpha*(4+f*(4-3*cosSqAlpha)); var L = lambda - (1-C) * f * sinAlpha * (sigma + C*sinSigma*(cos2SigmaM+C*cosSigma*(-1+2*cos2SigmaM*cos2SigmaM))); var revAz = Math.atan2(sinAlpha, -tmp); // final bearing return new OpenLayers.LonLat(lon1+u.deg(L), u.deg(lat2)); }; /** * Function: getParameters * Parse the parameters from a URL or from the current page itself into a * JavaScript Object. Note that parameter values with commas are separated * out into an Array. * * Parameters: * url - {String} Optional url used to extract the query string. * If url is null or is not supplied, query string is taken * from the page location. * options - {Object} Additional options. Optional. * * Valid options: * splitArgs - {Boolean} Split comma delimited params into arrays? Default is * true. * * Returns: * {Object} An object of key/value pairs from the query string. */ OpenLayers.Util.getParameters = function(url, options) { options = options || {}; // if no url specified, take it from the location bar url = (url === null || url === undefined) ? window.location.href : url; //parse out parameters portion of url string var paramsString = ""; if (OpenLayers.String.contains(url, '?')) { var start = url.indexOf('?') + 1; var end = OpenLayers.String.contains(url, "#") ? url.indexOf('#') : url.length; paramsString = url.substring(start, end); } var parameters = {}; var pairs = paramsString.split(/[&;]/); for(var i=0, len=pairs.length; i 1.0) ? (1.0 / scale) : scale; return normScale; }; /** * Function: getResolutionFromScale * * Parameters: * scale - {Float} * units - {String} Index into OpenLayers.INCHES_PER_UNIT hashtable. * Default is degrees * * Returns: * {Float} The corresponding resolution given passed-in scale and unit * parameters. If the given scale is falsey, the returned resolution will * be undefined. */ OpenLayers.Util.getResolutionFromScale = function (scale, units) { var resolution; if (scale) { if (units == null) { units = "degrees"; } var normScale = OpenLayers.Util.normalizeScale(scale); resolution = 1 / (normScale * OpenLayers.INCHES_PER_UNIT[units] * OpenLayers.DOTS_PER_INCH); } return resolution; }; /** * Function: getScaleFromResolution * * Parameters: * resolution - {Float} * units - {String} Index into OpenLayers.INCHES_PER_UNIT hashtable. * Default is degrees * * Returns: * {Float} The corresponding scale given passed-in resolution and unit * parameters. */ OpenLayers.Util.getScaleFromResolution = function (resolution, units) { if (units == null) { units = "degrees"; } var scale = resolution * OpenLayers.INCHES_PER_UNIT[units] * OpenLayers.DOTS_PER_INCH; return scale; }; /** * Function: pagePosition * Calculates the position of an element on the page (see * http://code.google.com/p/doctype/wiki/ArticlePageOffset) * * OpenLayers.Util.pagePosition is based on Yahoo's getXY method, which is * Copyright (c) 2006, Yahoo! Inc. * All rights reserved. * * Redistribution and use of this software in source and binary forms, with or * without modification, are permitted provided that the following conditions * are met: * * * Redistributions of source code must retain the above copyright notice, * this list of conditions and the following disclaimer. * * * Redistributions in binary form must reproduce the above copyright notice, * this list of conditions and the following disclaimer in the documentation * and/or other materials provided with the distribution. * * * Neither the name of Yahoo! Inc. nor the names of its contributors may be * used to endorse or promote products derived from this software without * specific prior written permission of Yahoo! Inc. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE * POSSIBILITY OF SUCH DAMAGE. * * Parameters: * forElement - {DOMElement} * * Returns: * {Array} two item array, Left value then Top value. */ OpenLayers.Util.pagePosition = function(forElement) { // NOTE: If element is hidden (display none or disconnected or any the // ancestors are hidden) we get (0,0) by default but we still do the // accumulation of scroll position. var pos = [0, 0]; var viewportElement = OpenLayers.Util.getViewportElement(); if (!forElement || forElement == window || forElement == viewportElement) { // viewport is always at 0,0 as that defined the coordinate system for // this function - this avoids special case checks in the code below return pos; } // Gecko browsers normally use getBoxObjectFor to calculate the position. // When invoked for an element with an implicit absolute position though it // can be off by one. Therefore the recursive implementation is used in // those (relatively rare) cases. var BUGGY_GECKO_BOX_OBJECT = OpenLayers.IS_GECKO && document.getBoxObjectFor && OpenLayers.Element.getStyle(forElement, 'position') == 'absolute' && (forElement.style.top == '' || forElement.style.left == ''); var parent = null; var box; if (forElement.getBoundingClientRect) { // IE box = forElement.getBoundingClientRect(); var scrollTop = window.pageYOffset || viewportElement.scrollTop; var scrollLeft = window.pageXOffset || viewportElement.scrollLeft; pos[0] = box.left + scrollLeft; pos[1] = box.top + scrollTop; } else if (document.getBoxObjectFor && !BUGGY_GECKO_BOX_OBJECT) { // gecko // Gecko ignores the scroll values for ancestors, up to 1.9. See: // https://bugzilla.mozilla.org/show_bug.cgi?id=328881 and // https://bugzilla.mozilla.org/show_bug.cgi?id=330619 box = document.getBoxObjectFor(forElement); var vpBox = document.getBoxObjectFor(viewportElement); pos[0] = box.screenX - vpBox.screenX; pos[1] = box.screenY - vpBox.screenY; } else { // safari/opera pos[0] = forElement.offsetLeft; pos[1] = forElement.offsetTop; parent = forElement.offsetParent; if (parent != forElement) { while (parent) { pos[0] += parent.offsetLeft; pos[1] += parent.offsetTop; parent = parent.offsetParent; } } var browser = OpenLayers.BROWSER_NAME; // opera & (safari absolute) incorrectly account for body offsetTop if (browser == "opera" || (browser == "safari" && OpenLayers.Element.getStyle(forElement, 'position') == 'absolute')) { pos[1] -= document.body.offsetTop; } // accumulate the scroll positions for everything but the body element parent = forElement.offsetParent; while (parent && parent != document.body) { pos[0] -= parent.scrollLeft; // see https://bugs.opera.com/show_bug.cgi?id=249965 if (browser != "opera" || parent.tagName != 'TR') { pos[1] -= parent.scrollTop; } parent = parent.offsetParent; } } return pos; }; /** * Function: getViewportElement * Returns die viewport element of the document. The viewport element is * usually document.documentElement, except in IE,where it is either * document.body or document.documentElement, depending on the document's * compatibility mode (see * http://code.google.com/p/doctype/wiki/ArticleClientViewportElement) * * Returns: * {DOMElement} */ OpenLayers.Util.getViewportElement = function() { var viewportElement = arguments.callee.viewportElement; if (viewportElement == undefined) { viewportElement = (OpenLayers.BROWSER_NAME == "msie" && document.compatMode != 'CSS1Compat') ? document.body : document.documentElement; arguments.callee.viewportElement = viewportElement; } return viewportElement; }; /** * Function: isEquivalentUrl * Test two URLs for equivalence. * * Setting 'ignoreCase' allows for case-independent comparison. * * Comparison is based on: * - Protocol * - Host (evaluated without the port) * - Port (set 'ignorePort80' to ignore "80" values) * - Hash ( set 'ignoreHash' to disable) * - Pathname (for relative <-> absolute comparison) * - Arguments (so they can be out of order) * * Parameters: * url1 - {String} * url2 - {String} * options - {Object} Allows for customization of comparison: * 'ignoreCase' - Default is True * 'ignorePort80' - Default is True * 'ignoreHash' - Default is True * * Returns: * {Boolean} Whether or not the two URLs are equivalent */ OpenLayers.Util.isEquivalentUrl = function(url1, url2, options) { options = options || {}; OpenLayers.Util.applyDefaults(options, { ignoreCase: true, ignorePort80: true, ignoreHash: true, splitArgs: false }); var urlObj1 = OpenLayers.Util.createUrlObject(url1, options); var urlObj2 = OpenLayers.Util.createUrlObject(url2, options); //compare all keys except for "args" (treated below) for(var key in urlObj1) { if(key !== "args") { if(urlObj1[key] != urlObj2[key]) { return false; } } } // compare search args - irrespective of order for(var key in urlObj1.args) { if(urlObj1.args[key] != urlObj2.args[key]) { return false; } delete urlObj2.args[key]; } // urlObj2 shouldn't have any args left for(var key in urlObj2.args) { return false; } return true; }; /** * Function: createUrlObject * * Parameters: * url - {String} * options - {Object} A hash of options. * * Valid options: * ignoreCase - {Boolean} lowercase url, * ignorePort80 - {Boolean} don't include explicit port if port is 80, * ignoreHash - {Boolean} Don't include part of url after the hash (#). * splitArgs - {Boolean} Split comma delimited params into arrays? Default is * true. * * Returns: * {Object} An object with separate url, a, port, host, and args parsed out * and ready for comparison */ OpenLayers.Util.createUrlObject = function(url, options) { options = options || {}; // deal with relative urls first if(!(/^\w+:\/\//).test(url)) { var loc = window.location; var port = loc.port ? ":" + loc.port : ""; var fullUrl = loc.protocol + "//" + loc.host.split(":").shift() + port; if(url.indexOf("/") === 0) { // full pathname url = fullUrl + url; } else { // relative to current path var parts = loc.pathname.split("/"); parts.pop(); url = fullUrl + parts.join("/") + "/" + url; } } if (options.ignoreCase) { url = url.toLowerCase(); } var a = document.createElement('a'); a.href = url; var urlObject = {}; //host (without port) urlObject.host = a.host.split(":").shift(); //protocol urlObject.protocol = a.protocol; //port (get uniform browser behavior with port 80 here) if(options.ignorePort80) { urlObject.port = (a.port == "80" || a.port == "0") ? "" : a.port; } else { urlObject.port = (a.port == "" || a.port == "0") ? "80" : a.port; } //hash urlObject.hash = (options.ignoreHash || a.hash === "#") ? "" : a.hash; //args var queryString = a.search; if (!queryString) { var qMark = url.indexOf("?"); queryString = (qMark != -1) ? url.substr(qMark) : ""; } urlObject.args = OpenLayers.Util.getParameters(queryString, {splitArgs: options.splitArgs}); // pathname // // This is a workaround for Internet Explorer where // window.location.pathname has a leading "/", but // a.pathname has no leading "/". urlObject.pathname = (a.pathname.charAt(0) == "/") ? a.pathname : "/" + a.pathname; return urlObject; }; /** * Function: removeTail * Takes a url and removes everything after the ? and # * * Parameters: * url - {String} The url to process * * Returns: * {String} The string with all queryString and Hash removed */ OpenLayers.Util.removeTail = function(url) { var head = null; var qMark = url.indexOf("?"); var hashMark = url.indexOf("#"); if (qMark == -1) { head = (hashMark != -1) ? url.substr(0,hashMark) : url; } else { head = (hashMark != -1) ? url.substr(0,Math.min(qMark, hashMark)) : url.substr(0, qMark); } return head; }; /** * Constant: IS_GECKO * {Boolean} True if the userAgent reports the browser to use the Gecko engine */ OpenLayers.IS_GECKO = (function() { var ua = navigator.userAgent.toLowerCase(); return ua.indexOf("webkit") == -1 && ua.indexOf("gecko") != -1; })(); /** * Constant: CANVAS_SUPPORTED * {Boolean} True if canvas 2d is supported. */ OpenLayers.CANVAS_SUPPORTED = (function() { var elem = document.createElement('canvas'); return !!(elem.getContext && elem.getContext('2d')); })(); /** * Constant: BROWSER_NAME * {String} * A substring of the navigator.userAgent property. Depending on the userAgent * property, this will be the empty string or one of the following: * * "opera" -- Opera * * "msie" -- Internet Explorer * * "safari" -- Safari * * "firefox" -- Firefox * * "mozilla" -- Mozilla */ OpenLayers.BROWSER_NAME = (function() { var name = ""; var ua = navigator.userAgent.toLowerCase(); if (ua.indexOf("opera") != -1) { name = "opera"; } else if (ua.indexOf("msie") != -1) { name = "msie"; } else if (ua.indexOf("safari") != -1) { name = "safari"; } else if (ua.indexOf("mozilla") != -1) { if (ua.indexOf("firefox") != -1) { name = "firefox"; } else { name = "mozilla"; } } return name; })(); /** * Function: getBrowserName * * Returns: * {String} A string which specifies which is the current * browser in which we are running. * * Currently-supported browser detection and codes: * * 'opera' -- Opera * * 'msie' -- Internet Explorer * * 'safari' -- Safari * * 'firefox' -- Firefox * * 'mozilla' -- Mozilla * * If we are unable to property identify the browser, we * return an empty string. */ OpenLayers.Util.getBrowserName = function() { return OpenLayers.BROWSER_NAME; }; /** * Method: getRenderedDimensions * Renders the contentHTML offscreen to determine actual dimensions for * popup sizing. As we need layout to determine dimensions the content * is rendered -9999px to the left and absolute to ensure the * scrollbars do not flicker * * Parameters: * contentHTML * size - {} If either the 'w' or 'h' properties is * specified, we fix that dimension of the div to be measured. This is * useful in the case where we have a limit in one dimension and must * therefore meaure the flow in the other dimension. * options - {Object} * * Allowed Options: * displayClass - {String} Optional parameter. A CSS class name(s) string * to provide the CSS context of the rendered content. * containerElement - {DOMElement} Optional parameter. Insert the HTML to * this node instead of the body root when calculating dimensions. * * Returns: * {} */ OpenLayers.Util.getRenderedDimensions = function(contentHTML, size, options) { var w, h; // create temp container div with restricted size var container = document.createElement("div"); container.style.visibility = "hidden"; var containerElement = (options && options.containerElement) ? options.containerElement : document.body; // Opera and IE7 can't handle a node with position:aboslute if it inherits // position:absolute from a parent. var parentHasPositionAbsolute = false; var superContainer = null; var parent = containerElement; while (parent && parent.tagName.toLowerCase()!="body") { var parentPosition = OpenLayers.Element.getStyle(parent, "position"); if(parentPosition == "absolute") { parentHasPositionAbsolute = true; break; } else if (parentPosition && parentPosition != "static") { break; } parent = parent.parentNode; } if(parentHasPositionAbsolute && (containerElement.clientHeight === 0 || containerElement.clientWidth === 0) ){ superContainer = document.createElement("div"); superContainer.style.visibility = "hidden"; superContainer.style.position = "absolute"; superContainer.style.overflow = "visible"; superContainer.style.width = document.body.clientWidth + "px"; superContainer.style.height = document.body.clientHeight + "px"; superContainer.appendChild(container); } container.style.position = "absolute"; //fix a dimension, if specified. if (size) { if (size.w) { w = size.w; container.style.width = w + "px"; } else if (size.h) { h = size.h; container.style.height = h + "px"; } } //add css classes, if specified if (options && options.displayClass) { container.className = options.displayClass; } // create temp content div and assign content var content = document.createElement("div"); content.innerHTML = contentHTML; // we need overflow visible when calculating the size content.style.overflow = "visible"; if (content.childNodes) { for (var i=0, l=content.childNodes.length; i= 60) { coordinateseconds -= 60; coordinateminutes += 1; if( coordinateminutes >= 60) { coordinateminutes -= 60; coordinatedegrees += 1; } } if( coordinatedegrees < 10 ) { coordinatedegrees = "0" + coordinatedegrees; } var str = coordinatedegrees + "\u00B0"; if (dmsOption.indexOf('dm') >= 0) { if( coordinateminutes < 10 ) { coordinateminutes = "0" + coordinateminutes; } str += coordinateminutes + "'"; if (dmsOption.indexOf('dms') >= 0) { if( coordinateseconds < 10 ) { coordinateseconds = "0" + coordinateseconds; } str += coordinateseconds + '"'; } } if (axis == "lon") { str += coordinate < 0 ? OpenLayers.i18n("W") : OpenLayers.i18n("E"); } else { str += coordinate < 0 ? OpenLayers.i18n("S") : OpenLayers.i18n("N"); } return str; }; /** * Function: getConstructor * Take an OpenLayers style CLASS_NAME and return a constructor. * * Parameters: * className - {String} The dot delimited class name (e.g. 'OpenLayers.Foo'). * * Returns: * {Function} The constructor. */ OpenLayers.Util.getConstructor = function(className) { var Constructor; var parts = className.split('.'); if (parts[0] === "OpenLayers") { Constructor = OpenLayers; } else { // someone extended our base class and used their own namespace // this will not work when the library is evaluated in a closure // but it is the best we can do (until we ourselves provide a global) Constructor = window[parts[0]]; } for (var i = 1, ii = parts.length; i < ii; ++i) { Constructor = Constructor[parts[i]]; } return Constructor; }; /** FILE: 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 1; }, /** * Method: isLeftClick * Determine whether event was caused by a left click. * * Parameters: * event - {Event} * * Returns: * {Boolean} */ isLeftClick: function(event) { return (((event.which) && (event.which == 1)) || ((event.button) && (event.button == 1))); }, /** * Method: isRightClick * Determine whether event was caused by a right mouse click. * * Parameters: * event - {Event} * * Returns: * {Boolean} */ isRightClick: function(event) { return (((event.which) && (event.which == 3)) || ((event.button) && (event.button == 2))); }, /** * Method: stop * Stops an event from propagating. * * Parameters: * event - {Event} * allowDefault - {Boolean} If true, we stop the event chain but * still allow the default browser behaviour (text selection, * radio-button clicking, etc). Default is false. */ stop: function(event, allowDefault) { if (!allowDefault) { OpenLayers.Event.preventDefault(event); } if (event.stopPropagation) { event.stopPropagation(); } else { event.cancelBubble = true; } }, /** * Method: preventDefault * Cancels the event if it is cancelable, without stopping further * propagation of the event. * * Parameters: * event - {Event} */ preventDefault: function(event) { if (event.preventDefault) { event.preventDefault(); } else { event.returnValue = false; } }, /** * Method: findElement * * Parameters: * event - {Event} * tagName - {String} * * Returns: * {DOMElement} The first node with the given tagName, starting from the * node the event was triggered on and traversing the DOM upwards */ findElement: function(event, tagName) { var element = OpenLayers.Event.element(event); while (element.parentNode && (!element.tagName || (element.tagName.toUpperCase() != tagName.toUpperCase()))){ element = element.parentNode; } return element; }, /** * Method: observe * * Parameters: * elementParam - {DOMElement || String} * name - {String} * observer - {function} * useCapture - {Boolean} */ observe: function(elementParam, name, observer, useCapture) { var element = OpenLayers.Util.getElement(elementParam); useCapture = useCapture || false; if (name == 'keypress' && (navigator.appVersion.match(/Konqueror|Safari|KHTML/) || element.attachEvent)) { name = 'keydown'; } //if observers cache has not yet been created, create it if (!this.observers) { this.observers = {}; } //if not already assigned, make a new unique cache ID if (!element._eventCacheID) { var idPrefix = "eventCacheID_"; if (element.id) { idPrefix = element.id + "_" + idPrefix; } element._eventCacheID = OpenLayers.Util.createUniqueID(idPrefix); } var cacheID = element._eventCacheID; //if there is not yet a hash entry for this element, add one if (!this.observers[cacheID]) { this.observers[cacheID] = []; } //add a new observer to this element's list this.observers[cacheID].push({ 'element': element, 'name': name, 'observer': observer, 'useCapture': useCapture }); //add the actual browser event listener if (element.addEventListener) { element.addEventListener(name, observer, useCapture); } else if (element.attachEvent) { element.attachEvent('on' + name, observer); } }, /** * Method: stopObservingElement * Given the id of an element to stop observing, cycle through the * element's cached observers, calling stopObserving on each one, * skipping those entries which can no longer be removed. * * parameters: * elementParam - {DOMElement || String} */ stopObservingElement: function(elementParam) { var element = OpenLayers.Util.getElement(elementParam); var cacheID = element._eventCacheID; this._removeElementObservers(OpenLayers.Event.observers[cacheID]); }, /** * Method: _removeElementObservers * * Parameters: * elementObservers - {Array(Object)} Array of (element, name, * observer, usecapture) objects, * taken directly from hashtable */ _removeElementObservers: function(elementObservers) { if (elementObservers) { for(var i = elementObservers.length-1; i >= 0; i--) { var entry = elementObservers[i]; OpenLayers.Event.stopObserving.apply(this, [ entry.element, entry.name, entry.observer, entry.useCapture ]); } } }, /** * Method: stopObserving * * Parameters: * elementParam - {DOMElement || String} * name - {String} * observer - {function} * useCapture - {Boolean} * * Returns: * {Boolean} Whether or not the event observer was removed */ stopObserving: function(elementParam, name, observer, useCapture) { useCapture = useCapture || false; var element = OpenLayers.Util.getElement(elementParam); var cacheID = element._eventCacheID; if (name == 'keypress') { if ( navigator.appVersion.match(/Konqueror|Safari|KHTML/) || element.detachEvent) { name = 'keydown'; } } // find element's entry in this.observers cache and remove it var foundEntry = false; var elementObservers = OpenLayers.Event.observers[cacheID]; if (elementObservers) { // find the specific event type in the element's list var i=0; while(!foundEntry && i < elementObservers.length) { var cacheEntry = elementObservers[i]; if ((cacheEntry.name == name) && (cacheEntry.observer == observer) && (cacheEntry.useCapture == useCapture)) { elementObservers.splice(i, 1); if (elementObservers.length == 0) { delete OpenLayers.Event.observers[cacheID]; } foundEntry = true; break; } i++; } } //actually remove the event listener from browser if (foundEntry) { if (element.removeEventListener) { element.removeEventListener(name, observer, useCapture); } else if (element && element.detachEvent) { element.detachEvent('on' + name, observer); } } return foundEntry; }, /** * Method: unloadCache * Cycle through all the element entries in the events cache and call * stopObservingElement on each. */ unloadCache: function() { // check for OpenLayers.Event before checking for observers, because // OpenLayers.Event may be undefined in IE if no map instance was // created if (OpenLayers.Event && OpenLayers.Event.observers) { for (var cacheID in OpenLayers.Event.observers) { var elementObservers = OpenLayers.Event.observers[cacheID]; OpenLayers.Event._removeElementObservers.apply(this, [elementObservers]); } OpenLayers.Event.observers = false; } }, CLASS_NAME: "OpenLayers.Event" }; /* prevent memory leaks in IE */ OpenLayers.Event.observe(window, 'unload', OpenLayers.Event.unloadCache, false); /** * Class: OpenLayers.Events */ OpenLayers.Events = OpenLayers.Class({ /** * Constant: BROWSER_EVENTS * {Array(String)} supported events */ BROWSER_EVENTS: [ "mouseover", "mouseout", "mousedown", "mouseup", "mousemove", "click", "dblclick", "rightclick", "dblrightclick", "resize", "focus", "blur", "touchstart", "touchmove", "touchend", "keydown" ], /** * Property: listeners * {Object} Hashtable of Array(Function): events listener functions */ listeners: null, /** * Property: object * {Object} the code object issuing application events */ object: null, /** * Property: element * {DOMElement} the DOM element receiving browser events */ element: null, /** * Property: eventHandler * {Function} bound event handler attached to elements */ eventHandler: null, /** * APIProperty: fallThrough * {Boolean} */ fallThrough: null, /** * APIProperty: includeXY * {Boolean} Should the .xy property automatically be created for browser * mouse events? In general, this should be false. If it is true, then * mouse events will automatically generate a '.xy' property on the * event object that is passed. (Prior to OpenLayers 2.7, this was true * by default.) Otherwise, you can call the getMousePosition on the * relevant events handler on the object available via the 'evt.object' * property of the evt object. So, for most events, you can call: * function named(evt) { * this.xy = this.object.events.getMousePosition(evt) * } * * This option typically defaults to false for performance reasons: * when creating an events object whose primary purpose is to manage * relatively positioned mouse events within a div, it may make * sense to set it to true. * * This option is also used to control whether the events object caches * offsets. If this is false, it will not: the reason for this is that * it is only expected to be called many times if the includeXY property * is set to true. If you set this to true, you are expected to clear * the offset cache manually (using this.clearMouseCache()) if: * the border of the element changes * the location of the element in the page changes */ includeXY: false, /** * APIProperty: extensions * {Object} Event extensions registered with this instance. Keys are * event types, values are {OpenLayers.Events.*} extension instances or * {Boolean} for events that an instantiated extension provides in * addition to the one it was created for. * * Extensions create an event in addition to browser events, which usually * fires when a sequence of browser events is completed. Extensions are * automatically instantiated when a listener is registered for an event * provided by an extension. * * Extensions are created in the namespace using * , and named after the event they provide. * The constructor receives the target instance as * argument. Extensions that need to capture browser events before they * propagate can register their listeners events using , with * {extension: true} as 4th argument. * * If an extension creates more than one event, an alias for each event * type should be created and reference the same class. The constructor * should set a reference in the target's extensions registry to itself. * * Below is a minimal extension that provides the "foostart" and "fooend" * event types, which replace the native "click" event type if clicked on * an element with the css class "foo": * * (code) * OpenLayers.Events.foostart = OpenLayers.Class({ * initialize: function(target) { * this.target = target; * this.target.register("click", this, this.doStuff, {extension: true}); * // only required if extension provides more than one event type * this.target.extensions["foostart"] = true; * this.target.extensions["fooend"] = true; * }, * destroy: function() { * var target = this.target; * target.unregister("click", this, this.doStuff); * delete this.target; * // only required if extension provides more than one event type * delete target.extensions["foostart"]; * delete target.extensions["fooend"]; * }, * doStuff: function(evt) { * var propagate = true; * if (OpenLayers.Event.element(evt).className === "foo") { * propagate = false; * var target = this.target; * target.triggerEvent("foostart"); * window.setTimeout(function() { * target.triggerEvent("fooend"); * }, 1000); * } * return propagate; * } * }); * // only required if extension provides more than one event type * OpenLayers.Events.fooend = OpenLayers.Events.foostart; * (end) * */ extensions: null, /** * Property: extensionCount * {Object} Keys are event types (like in ), values are the * number of extension listeners for each event type. */ extensionCount: null, /** * Method: clearMouseListener * A version of that is bound to this instance so that * it can be used with and * . */ clearMouseListener: null, /** * Constructor: OpenLayers.Events * Construct an OpenLayers.Events object. * * Parameters: * object - {Object} The js object to which this Events object is being added * element - {DOMElement} A dom element to respond to browser events * eventTypes - {Array(String)} Deprecated. Array of custom application * events. A listener may be registered for any named event, regardless * of the values provided here. * fallThrough - {Boolean} Allow events to fall through after these have * been handled? * options - {Object} Options for the events object. */ initialize: function (object, element, eventTypes, fallThrough, options) { OpenLayers.Util.extend(this, options); this.object = object; this.fallThrough = fallThrough; this.listeners = {}; this.extensions = {}; this.extensionCount = {}; this._msTouches = []; // if a dom element is specified, add a listeners list // for browser events on the element and register them if (element != null) { this.attachToElement(element); } }, /** * APIMethod: destroy */ destroy: function () { for (var e in this.extensions) { if (typeof this.extensions[e] !== "boolean") { this.extensions[e].destroy(); } } this.extensions = null; if (this.element) { OpenLayers.Event.stopObservingElement(this.element); if(this.element.hasScrollEvent) { OpenLayers.Event.stopObserving( window, "scroll", this.clearMouseListener ); } } this.element = null; this.listeners = null; this.object = null; this.fallThrough = null; this.eventHandler = null; }, /** * APIMethod: addEventType * Deprecated. Any event can be triggered without adding it first. * * Parameters: * eventName - {String} */ addEventType: function(eventName) { }, /** * Method: attachToElement * * Parameters: * element - {HTMLDOMElement} a DOM element to attach browser events to */ attachToElement: function (element) { if (this.element) { OpenLayers.Event.stopObservingElement(this.element); } else { // keep a bound copy of handleBrowserEvent() so that we can // pass the same function to both Event.observe() and .stopObserving() this.eventHandler = OpenLayers.Function.bindAsEventListener( this.handleBrowserEvent, this ); // to be used with observe and stopObserving this.clearMouseListener = OpenLayers.Function.bind( this.clearMouseCache, this ); } this.element = element; var msTouch = !!window.navigator.msMaxTouchPoints; var type; for (var i = 0, len = this.BROWSER_EVENTS.length; i < len; i++) { type = this.BROWSER_EVENTS[i]; // register the event cross-browser OpenLayers.Event.observe(element, type, this.eventHandler ); if (msTouch && type.indexOf('touch') === 0) { this.addMsTouchListener(element, type, this.eventHandler); } } // disable dragstart in IE so that mousedown/move/up works normally OpenLayers.Event.observe(element, "dragstart", OpenLayers.Event.stop); }, /** * APIMethod: on * Convenience method for registering listeners with a common scope. * Internally, this method calls as shown in the examples * below. * * Example use: * (code) * // register a single listener for the "loadstart" event * events.on({"loadstart": loadStartListener}); * * // this is equivalent to the following * events.register("loadstart", undefined, loadStartListener); * * // register multiple listeners to be called with the same `this` object * events.on({ * "loadstart": loadStartListener, * "loadend": loadEndListener, * scope: object * }); * * // this is equivalent to the following * events.register("loadstart", object, loadStartListener); * events.register("loadend", object, loadEndListener); * (end) * * Parameters: * object - {Object} */ on: function(object) { for(var type in object) { if(type != "scope" && object.hasOwnProperty(type)) { this.register(type, object.scope, object[type]); } } }, /** * APIMethod: register * Register an event on the events object. * * When the event is triggered, the 'func' function will be called, in the * context of 'obj'. Imagine we were to register an event, specifying an * OpenLayers.Bounds Object as 'obj'. When the event is triggered, the * context in the callback function will be our Bounds object. This means * that within our callback function, we can access the properties and * methods of the Bounds object through the "this" variable. So our * callback could execute something like: * : leftStr = "Left: " + this.left; * * or * * : centerStr = "Center: " + this.getCenterLonLat(); * * Parameters: * type - {String} Name of the event to register * obj - {Object} The object to bind the context to for the callback#. * If no object is specified, default is the Events's 'object' property. * func - {Function} The callback function. If no callback is * specified, this function does nothing. * priority - {Boolean|Object} If true, adds the new listener to the * *front* of the events queue instead of to the end. * * Valid options for priority: * extension - {Boolean} If true, then the event will be registered as * extension event. Extension events are handled before all other * events. */ register: function (type, obj, func, priority) { if (type in OpenLayers.Events && !this.extensions[type]) { this.extensions[type] = new OpenLayers.Events[type](this); } if (func != null) { if (obj == null) { obj = this.object; } var listeners = this.listeners[type]; if (!listeners) { listeners = []; this.listeners[type] = listeners; this.extensionCount[type] = 0; } var listener = {obj: obj, func: func}; if (priority) { listeners.splice(this.extensionCount[type], 0, listener); if (typeof priority === "object" && priority.extension) { this.extensionCount[type]++; } } else { listeners.push(listener); } } }, /** * APIMethod: registerPriority * Same as register() but adds the new listener to the *front* of the * events queue instead of to the end. * * TODO: get rid of this in 3.0 - Decide whether listeners should be * called in the order they were registered or in reverse order. * * * Parameters: * type - {String} Name of the event to register * obj - {Object} The object to bind the context to for the callback#. * If no object is specified, default is the Events's * 'object' property. * func - {Function} The callback function. If no callback is * specified, this function does nothing. */ registerPriority: function (type, obj, func) { this.register(type, obj, func, true); }, /** * APIMethod: un * Convenience method for unregistering listeners with a common scope. * Internally, this method calls as shown in the examples * below. * * Example use: * (code) * // unregister a single listener for the "loadstart" event * events.un({"loadstart": loadStartListener}); * * // this is equivalent to the following * events.unregister("loadstart", undefined, loadStartListener); * * // unregister multiple listeners with the same `this` object * events.un({ * "loadstart": loadStartListener, * "loadend": loadEndListener, * scope: object * }); * * // this is equivalent to the following * events.unregister("loadstart", object, loadStartListener); * events.unregister("loadend", object, loadEndListener); * (end) */ un: function(object) { for(var type in object) { if(type != "scope" && object.hasOwnProperty(type)) { this.unregister(type, object.scope, object[type]); } } }, /** * APIMethod: unregister * * Parameters: * type - {String} * obj - {Object} If none specified, defaults to this.object * func - {Function} */ unregister: function (type, obj, func) { if (obj == null) { obj = this.object; } var listeners = this.listeners[type]; if (listeners != null) { for (var i=0, len=listeners.length; i Math.floor(evt.pageY) || evt.pageX === 0 && Math.floor(x) > Math.floor(evt.pageX)) { // iOS4 include scroll offset in clientX/Y x = x - winPageX; y = y - winPageY; } else if (y < (evt.pageY - winPageY) || x < (evt.pageX - winPageX) ) { // Some Android browsers have totally bogus values for clientX/Y // when scrolling/zooming a page x = evt.pageX - winPageX; y = evt.pageY - winPageY; } evt.olClientX = x; evt.olClientY = y; return { clientX: x, clientY: y }; }, /** * APIMethod: clearMouseCache * Clear cached data about the mouse position. This should be called any * time the element that events are registered on changes position * within the page. */ clearMouseCache: function() { this.element.scrolls = null; this.element.lefttop = null; this.element.offsets = null; }, /** * Method: getMousePosition * * Parameters: * evt - {Event} * * Returns: * {} The current xy coordinate of the mouse, adjusted * for offsets */ getMousePosition: function (evt) { if (!this.includeXY) { this.clearMouseCache(); } else if (!this.element.hasScrollEvent) { OpenLayers.Event.observe(window, "scroll", this.clearMouseListener); this.element.hasScrollEvent = true; } if (!this.element.scrolls) { var viewportElement = OpenLayers.Util.getViewportElement(); this.element.scrolls = [ window.pageXOffset || viewportElement.scrollLeft, window.pageYOffset || viewportElement.scrollTop ]; } if (!this.element.lefttop) { this.element.lefttop = [ (document.documentElement.clientLeft || 0), (document.documentElement.clientTop || 0) ]; } if (!this.element.offsets) { this.element.offsets = OpenLayers.Util.pagePosition(this.element); } return new OpenLayers.Pixel( (evt.clientX + this.element.scrolls[0]) - this.element.offsets[0] - this.element.lefttop[0], (evt.clientY + this.element.scrolls[1]) - this.element.offsets[1] - this.element.lefttop[1] ); }, /** * Method: addMsTouchListener * * Parameters: * element - {DOMElement} The DOM element to register the listener on * type - {String} The event type * handler - {Function} the handler */ addMsTouchListener: function (element, type, handler) { var eventHandler = this.eventHandler; var touches = this._msTouches; function msHandler(evt) { handler(OpenLayers.Util.applyDefaults({ stopPropagation: function() { for (var i=touches.length-1; i>=0; --i) { touches[i].stopPropagation(); } }, preventDefault: function() { for (var i=touches.length-1; i>=0; --i) { touches[i].preventDefault(); } }, type: type }, evt)); } switch (type) { case 'touchstart': return this.addMsTouchListenerStart(element, type, msHandler); case 'touchend': return this.addMsTouchListenerEnd(element, type, msHandler); case 'touchmove': return this.addMsTouchListenerMove(element, type, msHandler); default: throw 'Unknown touch event type'; } }, /** * Method: addMsTouchListenerStart * * Parameters: * element - {DOMElement} The DOM element to register the listener on * type - {String} The event type * handler - {Function} the handler */ addMsTouchListenerStart: function(element, type, handler) { var touches = this._msTouches; var cb = function(e) { var alreadyInArray = false; for (var i=0, ii=touches.length; i 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); /** FILE: 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. */ /** * @requires OpenLayers/BaseTypes/Class.js * @requires OpenLayers/Animation.js */ /** * Namespace: OpenLayers.Tween */ OpenLayers.Tween = OpenLayers.Class({ /** * APIProperty: easing * {(Function)} Easing equation used for the animation * Defaultly set to OpenLayers.Easing.Expo.easeOut */ easing: null, /** * APIProperty: begin * {Object} Values to start the animation with */ begin: null, /** * APIProperty: finish * {Object} Values to finish the animation with */ finish: null, /** * APIProperty: duration * {int} duration of the tween (number of steps) */ duration: null, /** * APIProperty: callbacks * {Object} An object with start, eachStep and done properties whose values * are functions to be call during the animation. They are passed the * current computed value as argument. */ callbacks: null, /** * Property: time * {int} Step counter */ time: null, /** * APIProperty: minFrameRate * {Number} The minimum framerate for animations in frames per second. After * each step, the time spent in the animation is compared to the calculated * time at this frame rate. If the animation runs longer than the calculated * time, the next step is skipped. Default is 30. */ minFrameRate: null, /** * Property: startTime * {Number} The timestamp of the first execution step. Used for skipping * frames */ startTime: null, /** * Property: animationId * {int} Loop id returned by OpenLayers.Animation.start */ animationId: null, /** * Property: playing * {Boolean} Tells if the easing is currently playing */ playing: false, /** * Constructor: OpenLayers.Tween * Creates a Tween. * * Parameters: * easing - {(Function)} easing function method to use */ initialize: function(easing) { this.easing = (easing) ? easing : OpenLayers.Easing.Expo.easeOut; }, /** * APIMethod: start * Plays the Tween, and calls the callback method on each step * * Parameters: * begin - {Object} values to start the animation with * finish - {Object} values to finish the animation with * duration - {int} duration of the tween (number of steps) * options - {Object} hash of options (callbacks (start, eachStep, done), * minFrameRate) */ start: function(begin, finish, duration, options) { this.playing = true; this.begin = begin; this.finish = finish; this.duration = duration; this.callbacks = options.callbacks; this.minFrameRate = options.minFrameRate || 30; this.time = 0; this.startTime = new Date().getTime(); OpenLayers.Animation.stop(this.animationId); this.animationId = null; if (this.callbacks && this.callbacks.start) { this.callbacks.start.call(this, this.begin); } this.animationId = OpenLayers.Animation.start( OpenLayers.Function.bind(this.play, this) ); }, /** * APIMethod: stop * Stops the Tween, and calls the done callback * Doesn't do anything if animation is already finished */ stop: function() { if (!this.playing) { return; } if (this.callbacks && this.callbacks.done) { this.callbacks.done.call(this, this.finish); } OpenLayers.Animation.stop(this.animationId); this.animationId = null; this.playing = false; }, /** * Method: play * Calls the appropriate easing method */ play: function() { var value = {}; for (var i in this.begin) { var b = this.begin[i]; var f = this.finish[i]; if (b == null || f == null || isNaN(b) || isNaN(f)) { throw new TypeError('invalid value for Tween'); } var c = f - b; value[i] = this.easing.apply(this, [this.time, b, c, this.duration]); } this.time++; if (this.callbacks && this.callbacks.eachStep) { // skip frames if frame rate drops below threshold if ((new Date().getTime() - this.startTime) / this.time <= 1000 / this.minFrameRate) { this.callbacks.eachStep.call(this, value); } } if (this.time > this.duration) { this.stop(); } }, /** * Create empty functions for all easing methods. */ CLASS_NAME: "OpenLayers.Tween" }); /** * Namespace: OpenLayers.Easing * * Credits: * Easing Equations by Robert Penner, */ OpenLayers.Easing = { /** * Create empty functions for all easing methods. */ CLASS_NAME: "OpenLayers.Easing" }; /** * Namespace: OpenLayers.Easing.Linear */ OpenLayers.Easing.Linear = { /** * Function: easeIn * * Parameters: * t - {Float} time * b - {Float} beginning position * c - {Float} total change * d - {Float} duration of the transition * * Returns: * {Float} */ easeIn: function(t, b, c, d) { return c*t/d + b; }, /** * Function: easeOut * * Parameters: * t - {Float} time * b - {Float} beginning position * c - {Float} total change * d - {Float} duration of the transition * * Returns: * {Float} */ easeOut: function(t, b, c, d) { return c*t/d + b; }, /** * Function: easeInOut * * Parameters: * t - {Float} time * b - {Float} beginning position * c - {Float} total change * d - {Float} duration of the transition * * Returns: * {Float} */ easeInOut: function(t, b, c, d) { return c*t/d + b; }, CLASS_NAME: "OpenLayers.Easing.Linear" }; /** * Namespace: OpenLayers.Easing.Expo */ OpenLayers.Easing.Expo = { /** * Function: easeIn * * Parameters: * t - {Float} time * b - {Float} beginning position * c - {Float} total change * d - {Float} duration of the transition * * Returns: * {Float} */ easeIn: function(t, b, c, d) { return (t==0) ? b : c * Math.pow(2, 10 * (t/d - 1)) + b; }, /** * Function: easeOut * * Parameters: * t - {Float} time * b - {Float} beginning position * c - {Float} total change * d - {Float} duration of the transition * * Returns: * {Float} */ easeOut: function(t, b, c, d) { return (t==d) ? b+c : c * (-Math.pow(2, -10 * t/d) + 1) + b; }, /** * Function: easeInOut * * Parameters: * t - {Float} time * b - {Float} beginning position * c - {Float} total change * d - {Float} duration of the transition * * Returns: * {Float} */ easeInOut: function(t, b, c, d) { if (t==0) return b; if (t==d) return b+c; if ((t/=d/2) < 1) return c/2 * Math.pow(2, 10 * (t - 1)) + b; return c/2 * (-Math.pow(2, -10 * --t) + 2) + b; }, CLASS_NAME: "OpenLayers.Easing.Expo" }; /** * Namespace: OpenLayers.Easing.Quad */ OpenLayers.Easing.Quad = { /** * Function: easeIn * * Parameters: * t - {Float} time * b - {Float} beginning position * c - {Float} total change * d - {Float} duration of the transition * * Returns: * {Float} */ easeIn: function(t, b, c, d) { return c*(t/=d)*t + b; }, /** * Function: easeOut * * Parameters: * t - {Float} time * b - {Float} beginning position * c - {Float} total change * d - {Float} duration of the transition * * Returns: * {Float} */ easeOut: function(t, b, c, d) { return -c *(t/=d)*(t-2) + b; }, /** * Function: easeInOut * * Parameters: * t - {Float} time * b - {Float} beginning position * c - {Float} total change * d - {Float} duration of the transition * * Returns: * {Float} */ easeInOut: function(t, b, c, d) { if ((t/=d/2) < 1) return c/2*t*t + b; return -c/2 * ((--t)*(t-2) - 1) + b; }, CLASS_NAME: "OpenLayers.Easing.Quad" }; /** FILE: OpenLayers/Projection.js **/ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for * full list of contributors). Published 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 */ /** * Namespace: OpenLayers.Projection * Methods for coordinate transforms between coordinate systems. By default, * OpenLayers ships with the ability to transform coordinates between * geographic (EPSG:4326) and web or spherical mercator (EPSG:900913 et al.) * coordinate reference systems. See the method for details * on usage. * * Additional transforms may be added by using the * library. If the proj4js library is included, the method * will work between any two coordinate reference systems with proj4js * definitions. * * If the proj4js library is not included, or if you wish to allow transforms * between arbitrary coordinate reference systems, use the * method to register a custom transform method. */ OpenLayers.Projection = OpenLayers.Class({ /** * Property: proj * {Object} Proj4js.Proj instance. */ proj: null, /** * Property: projCode * {String} */ projCode: null, /** * Property: titleRegEx * {RegExp} regular expression to strip the title from a proj4js definition */ titleRegEx: /\+title=[^\+]*/, /** * Constructor: OpenLayers.Projection * This class offers several methods for interacting with a wrapped * pro4js projection object. * * Parameters: * projCode - {String} A string identifying the Well Known Identifier for * the projection. * options - {Object} An optional object to set additional properties * on the projection. * * Returns: * {} A projection object. */ initialize: function(projCode, options) { OpenLayers.Util.extend(this, options); this.projCode = projCode; if (typeof Proj4js == "object") { this.proj = new Proj4js.Proj(projCode); } }, /** * APIMethod: getCode * Get the string SRS code. * * Returns: * {String} The SRS code. */ getCode: function() { return this.proj ? this.proj.srsCode : this.projCode; }, /** * APIMethod: getUnits * Get the units string for the projection -- returns null if * proj4js is not available. * * Returns: * {String} The units abbreviation. */ getUnits: function() { return this.proj ? this.proj.units : null; }, /** * Method: toString * Convert projection to string (getCode wrapper). * * Returns: * {String} The projection code. */ toString: function() { return this.getCode(); }, /** * Method: equals * Test equality of two projection instances. Determines equality based * soley on the projection code. * * Returns: * {Boolean} The two projections are equivalent. */ equals: function(projection) { var p = projection, equals = false; if (p) { if (!(p instanceof OpenLayers.Projection)) { p = new OpenLayers.Projection(p); } if ((typeof Proj4js == "object") && this.proj.defData && p.proj.defData) { equals = this.proj.defData.replace(this.titleRegEx, "") == p.proj.defData.replace(this.titleRegEx, ""); } else if (p.getCode) { var source = this.getCode(), target = p.getCode(); equals = source == target || !!OpenLayers.Projection.transforms[source] && OpenLayers.Projection.transforms[source][target] === OpenLayers.Projection.nullTransform; } } return equals; }, /* Method: destroy * Destroy projection object. */ destroy: function() { delete this.proj; delete this.projCode; }, CLASS_NAME: "OpenLayers.Projection" }); /** * Property: transforms * {Object} Transforms is an object, with from properties, each of which may * have a to property. This allows you to define projections without * requiring support for proj4js to be included. * * This object has keys which correspond to a 'source' projection object. The * keys should be strings, corresponding to the projection.getCode() value. * Each source projection object should have a set of destination projection * keys included in the object. * * Each value in the destination object should be a transformation function, * where the function is expected to be passed an object with a .x and a .y * property. The function should return the object, with the .x and .y * transformed according to the transformation function. * * Note - Properties on this object should not be set directly. To add a * transform method to this object, use the method. For an * example of usage, see the OpenLayers.Layer.SphericalMercator file. */ OpenLayers.Projection.transforms = {}; /** * APIProperty: defaults * {Object} Defaults for the SRS codes known to OpenLayers (currently * EPSG:4326, CRS:84, urn:ogc:def:crs:EPSG:6.6:4326, EPSG:900913, EPSG:3857, * EPSG:102113, EPSG:102100 and OSGEO:41001). Keys are the SRS code, values are * units, maxExtent (the validity extent for the SRS) and yx (true if this SRS * is known to have a reverse axis order). */ OpenLayers.Projection.defaults = { "EPSG:4326": { units: "degrees", maxExtent: [-180, -90, 180, 90], yx: true }, "CRS:84": { units: "degrees", maxExtent: [-180, -90, 180, 90] }, "EPSG:900913": { units: "m", maxExtent: [-20037508.34, -20037508.34, 20037508.34, 20037508.34] } }; /** * APIMethod: addTransform * Set a custom transform method between two projections. Use this method in * cases where the proj4js lib is not available or where custom projections * need to be handled. * * Parameters: * from - {String} The code for the source projection * to - {String} the code for the destination projection * method - {Function} A function that takes a point as an argument and * transforms that point from the source to the destination projection * in place. The original point should be modified. */ OpenLayers.Projection.addTransform = function(from, to, method) { if (method === OpenLayers.Projection.nullTransform) { var defaults = OpenLayers.Projection.defaults[from]; if (defaults && !OpenLayers.Projection.defaults[to]) { OpenLayers.Projection.defaults[to] = defaults; } } if(!OpenLayers.Projection.transforms[from]) { OpenLayers.Projection.transforms[from] = {}; } OpenLayers.Projection.transforms[from][to] = method; }; /** * APIMethod: transform * Transform a point coordinate from one projection to another. Note that * the input point is transformed in place. * * Parameters: * point - { | Object} An object with x and y * properties representing coordinates in those dimensions. * source - {OpenLayers.Projection} Source map coordinate system * dest - {OpenLayers.Projection} Destination map coordinate system * * Returns: * point - {object} A transformed coordinate. The original point is modified. */ OpenLayers.Projection.transform = function(point, source, dest) { if (source && dest) { if (!(source instanceof OpenLayers.Projection)) { source = new OpenLayers.Projection(source); } if (!(dest instanceof OpenLayers.Projection)) { dest = new OpenLayers.Projection(dest); } if (source.proj && dest.proj) { point = Proj4js.transform(source.proj, dest.proj, point); } else { var sourceCode = source.getCode(); var destCode = dest.getCode(); var transforms = OpenLayers.Projection.transforms; if (transforms[sourceCode] && transforms[sourceCode][destCode]) { transforms[sourceCode][destCode](point); } } } return point; }; /** * APIFunction: nullTransform * A null transformation - useful for defining projection aliases when * proj4js is not available: * * (code) * OpenLayers.Projection.addTransform("EPSG:3857", "EPSG:900913", * OpenLayers.Projection.nullTransform); * OpenLayers.Projection.addTransform("EPSG:900913", "EPSG:3857", * OpenLayers.Projection.nullTransform); * (end) */ OpenLayers.Projection.nullTransform = function(point) { return point; }; /** * Note: Transforms for web mercator <-> geographic * OpenLayers recognizes EPSG:3857, EPSG:900913, EPSG:102113, EPSG:102100 and * OSGEO:41001. OpenLayers originally started referring to EPSG:900913 as web * mercator. The EPSG has declared EPSG:3857 to be web mercator. * ArcGIS 10 recognizes the EPSG:3857, EPSG:102113, and EPSG:102100 as * equivalent. See http://blogs.esri.com/Dev/blogs/arcgisserver/archive/2009/11/20/ArcGIS-Online-moving-to-Google-_2F00_-Bing-tiling-scheme_3A00_-What-does-this-mean-for-you_3F00_.aspx#12084. * For geographic, OpenLayers recognizes EPSG:4326, CRS:84 and * urn:ogc:def:crs:EPSG:6.6:4326. OpenLayers also knows about the reverse axis * order for EPSG:4326. */ (function() { var pole = 20037508.34; function inverseMercator(xy) { xy.x = 180 * xy.x / pole; xy.y = 180 / Math.PI * (2 * Math.atan(Math.exp((xy.y / pole) * Math.PI)) - Math.PI / 2); return xy; } function forwardMercator(xy) { xy.x = xy.x * pole / 180; var y = Math.log(Math.tan((90 + xy.y) * Math.PI / 360)) / Math.PI * pole; xy.y = Math.max(-20037508.34, Math.min(y, 20037508.34)); return xy; } function map(base, codes) { var add = OpenLayers.Projection.addTransform; var same = OpenLayers.Projection.nullTransform; var i, len, code, other, j; for (i=0, len=codes.length; i=0; --i) { map(mercator[i], geographic); } for (i=geographic.length-1; i>=0; --i) { map(geographic[i], mercator); } })(); /** FILE: OpenLayers/Map.js **/ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for * full list of contributors). Published 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/Util/vendorPrefix.js * @requires OpenLayers/Events.js * @requires OpenLayers/Tween.js * @requires OpenLayers/Projection.js */ /** * Class: OpenLayers.Map * Instances of OpenLayers.Map are interactive maps embedded in a web page. * Create a new map with the constructor. * * On their own maps do not provide much functionality. To extend a map * it's necessary to add controls () and * layers () to the map. */ OpenLayers.Map = OpenLayers.Class({ /** * Constant: Z_INDEX_BASE * {Object} Base z-indexes for different classes of thing */ Z_INDEX_BASE: { BaseLayer: 100, Overlay: 325, Feature: 725, Popup: 750, Control: 1000 }, /** * APIProperty: events * {} * * Register a listener for a particular event with the following syntax: * (code) * map.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 map.events.object. * element - {DOMElement} A reference to map.events.element. * * Browser events have the following additional properties: * xy - {} The pixel location of the event (relative * to the the map viewport). * * Supported map event types: * preaddlayer - triggered before a layer has been added. The event * object will include a *layer* property that references the layer * to be added. When a listener returns "false" the adding will be * aborted. * addlayer - triggered after a layer has been added. The event object * will include a *layer* property that references the added layer. * preremovelayer - triggered before a layer has been removed. The event * object will include a *layer* property that references the layer * to be removed. When a listener returns "false" the removal will be * aborted. * removelayer - triggered after a layer has been removed. The event * object will include a *layer* property that references the removed * layer. * changelayer - triggered after a layer name change, order change, * opacity change, params change, visibility change (actual visibility, * not the layer's visibility property) or attribution change (due to * extent change). Listeners will receive an event object with *layer* * and *property* properties. The *layer* property will be a reference * to the changed layer. The *property* property will be a key to the * changed property (name, order, opacity, params, visibility or * attribution). * movestart - triggered after the start of a drag, pan, or zoom. The event * object may include a *zoomChanged* property that tells whether the * zoom has changed. * move - triggered after each drag, pan, or zoom * moveend - triggered after a drag, pan, or zoom completes * zoomend - triggered after a zoom completes * mouseover - triggered after mouseover the map * mouseout - triggered after mouseout the map * mousemove - triggered after mousemove the map * changebaselayer - triggered after the base layer changes * updatesize - triggered after the method was executed */ /** * Property: id * {String} Unique identifier for the map */ id: null, /** * Property: fractionalZoom * {Boolean} For a base layer that supports it, allow the map resolution * to be set to a value between one of the values in the resolutions * array. Default is false. * * When fractionalZoom is set to true, it is possible to zoom to * an arbitrary extent. This requires a base layer from a source * that supports requests for arbitrary extents (i.e. not cached * tiles on a regular lattice). This means that fractionalZoom * will not work with commercial layers (Google, Yahoo, VE), layers * using TileCache, or any other pre-cached data sources. * * If you are using fractionalZoom, then you should also use * instead of layer.resolutions[zoom] as the * former works for non-integer zoom levels. */ fractionalZoom: false, /** * APIProperty: events * {} An events object that handles all * events on the map */ events: null, /** * APIProperty: allOverlays * {Boolean} Allow the map to function with "overlays" only. Defaults to * false. If true, the lowest layer in the draw order will act as * the base layer. In addition, if set to true, all layers will * have isBaseLayer set to false when they are added to the map. * * Note: * If you set map.allOverlays to true, then you *cannot* use * map.setBaseLayer or layer.setIsBaseLayer. With allOverlays true, * the lowest layer in the draw layer is the base layer. So, to change * the base layer, use or to set the layer * index to 0. */ allOverlays: false, /** * APIProperty: div * {DOMElement|String} The element that contains the map (or an id for * that element). If the constructor is called * with two arguments, this should be provided as the first argument. * Alternatively, the map constructor can be called with the options * object as the only argument. In this case (one argument), a * div property may or may not be provided. If the div property * is not provided, the map can be rendered to a container later * using the method. * * Note: * If you are calling after map construction, do not use * auto. Instead, divide your by your * maximum expected dimension. */ div: null, /** * Property: dragging * {Boolean} The map is currently being dragged. */ dragging: false, /** * Property: size * {} Size of the main div (this.div) */ size: null, /** * Property: viewPortDiv * {HTMLDivElement} The element that represents the map viewport */ viewPortDiv: null, /** * Property: layerContainerOrigin * {} The lonlat at which the later container was * re-initialized (on-zoom) */ layerContainerOrigin: null, /** * Property: layerContainerDiv * {HTMLDivElement} The element that contains the layers. */ layerContainerDiv: null, /** * APIProperty: layers * {Array()} Ordered list of layers in the map */ layers: null, /** * APIProperty: controls * {Array()} List of controls associated with the map. * * If not provided in the map options at construction, the map will * by default be given the following controls if present in the build: * - or * - or * - * - */ controls: null, /** * Property: popups * {Array()} List of popups associated with the map */ popups: null, /** * APIProperty: baseLayer * {} The currently selected base layer. This determines * min/max zoom level, projection, etc. */ baseLayer: null, /** * Property: center * {} The current center of the map */ center: null, /** * Property: resolution * {Float} The resolution of the map. */ resolution: null, /** * Property: zoom * {Integer} The current zoom level of the map */ zoom: 0, /** * Property: panRatio * {Float} The ratio of the current extent within * which panning will tween. */ panRatio: 1.5, /** * APIProperty: options * {Object} The options object passed to the class constructor. Read-only. */ options: null, // Options /** * APIProperty: tileSize * {} Set in the map options to override the default tile * size for this map. */ tileSize: null, /** * APIProperty: projection * {String} Set in the map options to specify the default projection * for layers added to this map. When using a projection other than EPSG:4326 * (CRS:84, Geographic) or EPSG:3857 (EPSG:900913, Web Mercator), * also set maxExtent, maxResolution or resolutions. Default is "EPSG:4326". * Note that the projection of the map is usually determined * by that of the current baseLayer (see and ). */ projection: "EPSG:4326", /** * APIProperty: units * {String} The map units. 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: resolutions * {Array(Float)} 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: maxResolution * {Float} Required if you are not displaying the whole world on a tile * with the size specified in . */ maxResolution: null, /** * APIProperty: minResolution * {Float} */ minResolution: null, /** * APIProperty: maxScale * {Float} */ maxScale: null, /** * APIProperty: minScale * {Float} */ minScale: null, /** * APIProperty: maxExtent * {|Array} If provided as an array, the array * should consist of four values (left, bottom, right, top). * The maximum extent for the map. * Default depends on projection; if this is one of those defined in OpenLayers.Projection.defaults * (EPSG:4326 or web mercator), maxExtent will be set to the value defined there; * else, defaults to null. * To restrict user panning and zooming of the map, use instead. * The value for will change calculations for tile URLs. */ 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 map. Defaults to null. */ minExtent: null, /** * APIProperty: restrictedExtent * {|Array} If provided as an array, the array * should consist of four values (left, bottom, right, top). * Limit map navigation to this extent where possible. * If a non-null restrictedExtent is set, panning will be restricted * to the given bounds. In addition, zooming to a resolution that * displays more than the restricted extent will center the map * on the restricted extent. If you wish to limit the zoom level * or resolution, use maxResolution. */ restrictedExtent: null, /** * APIProperty: numZoomLevels * {Integer} Number of zoom levels for the map. Defaults to 16. Set a * different value in the map options if needed. */ numZoomLevels: 16, /** * APIProperty: theme * {String} Relative path to a CSS file from which to load theme styles. * Specify null in the map options (e.g. {theme: null}) if you * want to get cascading style declarations - by putting links to * stylesheets or style declarations directly in your page. */ theme: null, /** * APIProperty: displayProjection * {} Requires proj4js support for projections other * than EPSG:4326 or EPSG:900913/EPSG:3857. Projection used by * several controls to display data to user. If this property is set, * it will be set on any control which has a null displayProjection * property at the time the control is added to the map. */ displayProjection: null, /** * APIProperty: tileManager * {|Object} By default, and if the build contains * TileManager.js, the map will use the TileManager to queue image requests * and to cache tile image elements. To create a map without a TileManager * configure the map with tileManager: null. To create a TileManager with * non-default options, supply the options instead or alternatively supply * an instance of {}. */ /** * APIProperty: fallThrough * {Boolean} Should OpenLayers allow events on the map to fall through to * other elements on the page, or should it swallow them? (#457) * Default is to swallow. */ fallThrough: false, /** * APIProperty: autoUpdateSize * {Boolean} Should OpenLayers automatically update the size of the map * when the resize event is fired. Default is true. */ autoUpdateSize: true, /** * 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, /** * Property: panTween * {} Animated panning tween object, see panTo() */ panTween: null, /** * APIProperty: panMethod * {Function} The Easing function to be used for tweening. Default is * OpenLayers.Easing.Expo.easeOut. Setting this to 'null' turns off * animated panning. */ panMethod: OpenLayers.Easing.Expo.easeOut, /** * Property: panDuration * {Integer} The number of steps to be passed to the * OpenLayers.Tween.start() method when the map is * panned. * Default is 50. */ panDuration: 50, /** * Property: zoomTween * {} Animated zooming tween object, see zoomTo() */ zoomTween: null, /** * APIProperty: zoomMethod * {Function} The Easing function to be used for tweening. Default is * OpenLayers.Easing.Quad.easeOut. Setting this to 'null' turns off * animated zooming. */ zoomMethod: OpenLayers.Easing.Quad.easeOut, /** * Property: zoomDuration * {Integer} The number of steps to be passed to the * OpenLayers.Tween.start() method when the map is zoomed. * Default is 20. */ zoomDuration: 20, /** * Property: paddingForPopups * {} Outside margin of the popup. Used to prevent * the popup from getting too close to the map border. */ paddingForPopups : null, /** * Property: layerContainerOriginPx * {Object} Cached object representing the layer container origin (in pixels). */ layerContainerOriginPx: null, /** * Property: minPx * {Object} An object with a 'x' and 'y' values that is the lower * left of maxExtent in viewport pixel space. * Used to verify in moveByPx that the new location we're moving to * is valid. It is also used in the getLonLatFromViewPortPx function * of Layer. */ minPx: null, /** * Property: maxPx * {Object} An object with a 'x' and 'y' values that is the top * right of maxExtent in viewport pixel space. * Used to verify in moveByPx that the new location we're moving to * is valid. */ maxPx: null, /** * Constructor: OpenLayers.Map * Constructor for a new OpenLayers.Map instance. There are two possible * ways to call the map constructor. See the examples below. * * Parameters: * div - {DOMElement|String} The element or id of an element in your page * that will contain the map. May be omitted if the
option is * provided or if you intend to call the method later. * options - {Object} Optional object with properties to tag onto the map. * * Valid options (in addition to the listed API properties): * center - {|Array} The default initial center of the map. * If provided as array, the first value is the x coordinate, * and the 2nd value is the y coordinate. * Only specify if is provided. * Note that if an ArgParser/Permalink control is present, * and the querystring contains coordinates, center will be set * by that, and this option will be ignored. * zoom - {Number} The initial zoom level for the map. Only specify if * is provided. * Note that if an ArgParser/Permalink control is present, * and the querystring contains a zoom level, zoom will be set * by that, and this option will be ignored. * extent - {|Array} The initial extent of the map. * If provided as an array, the array should consist of * four values (left, bottom, right, top). * Only specify if
and are not provided. * * Examples: * (code) * // create a map with default options in an element with the id "map1" * var map = new OpenLayers.Map("map1"); * * // create a map with non-default options in an element with id "map2" * var options = { * projection: "EPSG:3857", * maxExtent: new OpenLayers.Bounds(-200000, -200000, 200000, 200000), * center: new OpenLayers.LonLat(-12356463.476333, 5621521.4854095) * }; * var map = new OpenLayers.Map("map2", options); * * // map with non-default options - same as above but with a single argument, * // a restricted extent, and using arrays for bounds and center * var map = new OpenLayers.Map({ * div: "map_id", * projection: "EPSG:3857", * maxExtent: [-18924313.432222, -15538711.094146, 18924313.432222, 15538711.094146], * restrictedExtent: [-13358338.893333, -9608371.5085962, 13358338.893333, 9608371.5085962], * center: [-12356463.476333, 5621521.4854095] * }); * * // create a map without a reference to a container - call render later * var map = new OpenLayers.Map({ * projection: "EPSG:3857", * maxExtent: new OpenLayers.Bounds(-200000, -200000, 200000, 200000) * }); * (end) */ initialize: function (div, options) { // If only one argument is provided, check if it is an object. if(arguments.length === 1 && typeof div === "object") { options = div; div = options && options.div; } // Simple-type defaults are set in class definition. // Now set complex-type defaults this.tileSize = new OpenLayers.Size(OpenLayers.Map.TILE_WIDTH, OpenLayers.Map.TILE_HEIGHT); this.paddingForPopups = new OpenLayers.Bounds(15, 15, 15, 15); this.theme = OpenLayers._getScriptLocation() + 'theme/default/style.css'; // backup original options this.options = OpenLayers.Util.extend({}, options); // now override default options OpenLayers.Util.extend(this, options); var projCode = this.projection instanceof OpenLayers.Projection ? this.projection.projCode : this.projection; OpenLayers.Util.applyDefaults(this, OpenLayers.Projection.defaults[projCode]); // allow extents and center to be arrays if (this.maxExtent && !(this.maxExtent instanceof OpenLayers.Bounds)) { this.maxExtent = new OpenLayers.Bounds(this.maxExtent); } if (this.minExtent && !(this.minExtent instanceof OpenLayers.Bounds)) { this.minExtent = new OpenLayers.Bounds(this.minExtent); } if (this.restrictedExtent && !(this.restrictedExtent instanceof OpenLayers.Bounds)) { this.restrictedExtent = new OpenLayers.Bounds(this.restrictedExtent); } if (this.center && !(this.center instanceof OpenLayers.LonLat)) { this.center = new OpenLayers.LonLat(this.center); } // initialize layers array this.layers = []; this.id = OpenLayers.Util.createUniqueID("OpenLayers.Map_"); this.div = OpenLayers.Util.getElement(div); if(!this.div) { this.div = document.createElement("div"); this.div.style.height = "1px"; this.div.style.width = "1px"; } OpenLayers.Element.addClass(this.div, 'olMap'); // the viewPortDiv is the outermost div we modify var id = this.id + "_OpenLayers_ViewPort"; this.viewPortDiv = OpenLayers.Util.createDiv(id, null, null, null, "relative", null, "hidden"); this.viewPortDiv.style.width = "100%"; this.viewPortDiv.style.height = "100%"; this.viewPortDiv.className = "olMapViewport"; this.div.appendChild(this.viewPortDiv); this.events = new OpenLayers.Events( this, this.viewPortDiv, null, this.fallThrough, {includeXY: true} ); if (OpenLayers.TileManager && this.tileManager !== null) { if (!(this.tileManager instanceof OpenLayers.TileManager)) { this.tileManager = new OpenLayers.TileManager(this.tileManager); } this.tileManager.addMap(this); } // the layerContainerDiv is the one that holds all the layers id = this.id + "_OpenLayers_Container"; this.layerContainerDiv = OpenLayers.Util.createDiv(id); this.layerContainerDiv.style.zIndex=this.Z_INDEX_BASE['Popup']-1; this.layerContainerOriginPx = {x: 0, y: 0}; this.applyTransform(); this.viewPortDiv.appendChild(this.layerContainerDiv); this.updateSize(); if(this.eventListeners instanceof Object) { this.events.on(this.eventListeners); } if (this.autoUpdateSize === true) { // updateSize on catching the window's resize // Note that this is ok, as updateSize() does nothing if the // map's size has not actually changed. this.updateSizeDestroy = OpenLayers.Function.bind(this.updateSize, this); OpenLayers.Event.observe(window, 'resize', this.updateSizeDestroy); } // only append link stylesheet if the theme property is set if(this.theme) { // check existing links for equivalent url var addNode = true; var nodes = document.getElementsByTagName('link'); for(var i=0, len=nodes.length; i=0; --i) { this.controls[i].destroy(); } this.controls = null; } if (this.layers != null) { for (var i = this.layers.length - 1; i>=0; --i) { //pass 'false' to destroy so that map wont try to set a new // baselayer after each baselayer is removed this.layers[i].destroy(false); } this.layers = null; } if (this.viewPortDiv && this.viewPortDiv.parentNode) { this.viewPortDiv.parentNode.removeChild(this.viewPortDiv); } this.viewPortDiv = null; if (this.tileManager) { this.tileManager.removeMap(this); this.tileManager = null; } if(this.eventListeners) { this.events.un(this.eventListeners); this.eventListeners = null; } this.events.destroy(); this.events = null; this.options = null; }, /** * APIMethod: setOptions * Change the map options * * Parameters: * options - {Object} Hashtable of options to tag to the map */ setOptions: function(options) { var updatePxExtent = this.minPx && options.restrictedExtent != this.restrictedExtent; OpenLayers.Util.extend(this, options); // force recalculation of minPx and maxPx updatePxExtent && this.moveTo(this.getCachedCenter(), this.zoom, { forceZoomChange: true }); }, /** * APIMethod: getTileSize * Get the tile size for the map * * Returns: * {} */ getTileSize: function() { return this.tileSize; }, /** * APIMethod: getBy * Get a list of objects given a property and a match item. * * Parameters: * array - {String} A property on the map whose value is an array. * property - {String} A property on each item of the given array. * 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(map[array][i][property]) evaluates to true, the item will * be included in the array returned. If no items are found, an empty * array is returned. * * Returns: * {Array} An array of items where the given property matches the given * criteria. */ getBy: function(array, property, match) { var test = (typeof match.test == "function"); var found = OpenLayers.Array.filter(this[array], function(item) { return item[property] == match || (test && match.test(item[property])); }); return found; }, /** * APIMethod: getLayersBy * Get a list of layers with properties matching the given criteria. * * Parameters: * property - {String} A layer 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(layer[property]) evaluates to true, the layer will be * included in the array returned. If no layers are found, an empty * array is returned. * * Returns: * {Array()} A list of layers matching the given criteria. * An empty array is returned if no matches are found. */ getLayersBy: function(property, match) { return this.getBy("layers", property, match); }, /** * APIMethod: getLayersByName * Get a list of layers with names matching the given name. * * Parameters: * match - {String | Object} A layer 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(layer.name) evaluates to true, the layer will be included * in the list of layers returned. If no layers are found, an empty * array is returned. * * Returns: * {Array()} A list of layers matching the given name. * An empty array is returned if no matches are found. */ getLayersByName: function(match) { return this.getLayersBy("name", match); }, /** * APIMethod: getLayersByClass * Get a list of layers of a given class (CLASS_NAME). * * Parameters: * match - {String | Object} A layer class name. The 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 type.test(layer.CLASS_NAME) evaluates to true, the layer will * be included in the list of layers returned. If no layers are * found, an empty array is returned. * * Returns: * {Array()} A list of layers matching the given class. * An empty array is returned if no matches are found. */ getLayersByClass: function(match) { return this.getLayersBy("CLASS_NAME", match); }, /** * 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(layer[property]) evaluates to true, the layer will be * included in the array returned. If no layers are found, an empty * array is returned. * * Returns: * {Array()} A list of controls matching the given * criteria. An empty array is returned if no matches are found. */ getControlsBy: function(property, match) { return this.getBy("controls", property, match); }, /** * APIMethod: getControlsByClass * Get a list of controls of a given class (CLASS_NAME). * * Parameters: * match - {String | Object} A control class name. The 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 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()} A list of controls matching the given class. * An empty array is returned if no matches are found. */ getControlsByClass: function(match) { return this.getControlsBy("CLASS_NAME", match); }, /********************************************************/ /* */ /* Layer Functions */ /* */ /* The following functions deal with adding and */ /* removing Layers to and from the Map */ /* */ /********************************************************/ /** * APIMethod: getLayer * Get a layer based on its id * * Parameters: * id - {String} A layer id * * Returns: * {} The Layer with the corresponding id from the map's * layer collection, or null if not found. */ getLayer: function(id) { var foundLayer = null; for (var i=0, len=this.layers.length; i} * zIdx - {int} */ setLayerZIndex: function (layer, zIdx) { layer.setZIndex( this.Z_INDEX_BASE[layer.isBaseLayer ? 'BaseLayer' : 'Overlay'] + zIdx * 5 ); }, /** * Method: resetLayersZIndex * Reset each layer's z-index based on layer's array index */ resetLayersZIndex: function() { for (var i=0, len=this.layers.length; i} * * Returns: * {Boolean} True if the layer has been added to the map. */ addLayer: function (layer) { for(var i = 0, len = this.layers.length; i < len; i++) { if (this.layers[i] == layer) { return false; } } if (this.events.triggerEvent("preaddlayer", {layer: layer}) === false) { return false; } if(this.allOverlays) { layer.isBaseLayer = false; } layer.div.className = "olLayerDiv"; layer.div.style.overflow = ""; this.setLayerZIndex(layer, this.layers.length); if (layer.isFixed) { this.viewPortDiv.appendChild(layer.div); } else { this.layerContainerDiv.appendChild(layer.div); } this.layers.push(layer); layer.setMap(this); if (layer.isBaseLayer || (this.allOverlays && !this.baseLayer)) { if (this.baseLayer == null) { // set the first baselaye we add as the baselayer this.setBaseLayer(layer); } else { layer.setVisibility(false); } } else { layer.redraw(); } this.events.triggerEvent("addlayer", {layer: layer}); layer.events.triggerEvent("added", {map: this, layer: layer}); layer.afterAdd(); return true; }, /** * APIMethod: addLayers * * Parameters: * layers - {Array()} */ addLayers: function (layers) { for (var i=0, len=layers.length; i} * setNewBaseLayer - {Boolean} Default is true */ removeLayer: function(layer, setNewBaseLayer) { if (this.events.triggerEvent("preremovelayer", {layer: layer}) === false) { return; } if (setNewBaseLayer == null) { setNewBaseLayer = true; } if (layer.isFixed) { this.viewPortDiv.removeChild(layer.div); } else { this.layerContainerDiv.removeChild(layer.div); } OpenLayers.Util.removeItem(this.layers, layer); layer.removeMap(this); layer.map = null; // if we removed the base layer, need to set a new one if(this.baseLayer == layer) { this.baseLayer = null; if(setNewBaseLayer) { for(var i=0, len=this.layers.length; i} * * Returns: * {Integer} The current (zero-based) index of the given layer in the map's * layer stack. Returns -1 if the layer isn't on the map. */ getLayerIndex: function (layer) { return OpenLayers.Util.indexOf(this.layers, layer); }, /** * APIMethod: setLayerIndex * Move the given layer to the specified (zero-based) index in the layer * list, changing its z-index in the map display. Use * map.getLayerIndex() to find out the current index of a layer. Note * that this cannot (or at least should not) be effectively used to * raise base layers above overlays. * * Parameters: * layer - {} * idx - {int} */ setLayerIndex: function (layer, idx) { var base = this.getLayerIndex(layer); if (idx < 0) { idx = 0; } else if (idx > this.layers.length) { idx = this.layers.length; } if (base != idx) { this.layers.splice(base, 1); this.layers.splice(idx, 0, layer); for (var i=0, len=this.layers.length; i} * delta - {int} */ raiseLayer: function (layer, delta) { var idx = this.getLayerIndex(layer) + delta; this.setLayerIndex(layer, idx); }, /** * APIMethod: setBaseLayer * Allows user to specify one of the currently-loaded layers as the Map's * new base layer. * * Parameters: * newBaseLayer - {} */ setBaseLayer: function(newBaseLayer) { if (newBaseLayer != this.baseLayer) { // ensure newBaseLayer is already loaded if (OpenLayers.Util.indexOf(this.layers, newBaseLayer) != -1) { // preserve center and scale when changing base layers var center = this.getCachedCenter(); var newResolution = OpenLayers.Util.getResolutionFromScale( this.getScale(), newBaseLayer.units ); // make the old base layer invisible if (this.baseLayer != null && !this.allOverlays) { this.baseLayer.setVisibility(false); } // set new baselayer this.baseLayer = newBaseLayer; if(!this.allOverlays || this.baseLayer.visibility) { this.baseLayer.setVisibility(true); // Layer may previously have been visible but not in range. // In this case we need to redraw it to make it visible. if (this.baseLayer.inRange === false) { this.baseLayer.redraw(); } } // recenter the map if (center != null) { // new zoom level derived from old scale var newZoom = this.getZoomForResolution( newResolution || this.resolution, true ); // zoom and force zoom change this.setCenter(center, newZoom, false, true); } this.events.triggerEvent("changebaselayer", { layer: this.baseLayer }); } } }, /********************************************************/ /* */ /* Control Functions */ /* */ /* The following functions deal with adding and */ /* removing Controls to and from the Map */ /* */ /********************************************************/ /** * APIMethod: addControl * Add the passed over control to the map. Optionally * position the control at the given pixel. * * Parameters: * control - {} * px - {} */ addControl: function (control, px) { this.controls.push(control); this.addControlToMap(control, px); }, /** * APIMethod: addControls * Add all of the passed over controls to the map. * You can pass over an optional second array * with pixel-objects to position the controls. * The indices of the two arrays should match and * you can add null as pixel for those controls * you want to be autopositioned. * * Parameters: * controls - {Array()} * pixels - {Array()} */ addControls: function (controls, pixels) { var pxs = (arguments.length === 1) ? [] : pixels; for (var i=0, len=controls.length; i} * px - {} */ addControlToMap: function (control, px) { // If a control doesn't have a div at this point, it belongs in the // viewport. control.outsideViewport = (control.div != null); // If the map has a displayProjection, and the control doesn't, set // the display projection. if (this.displayProjection && !control.displayProjection) { control.displayProjection = this.displayProjection; } control.setMap(this); var div = control.draw(px); if (div) { if(!control.outsideViewport) { div.style.zIndex = this.Z_INDEX_BASE['Control'] + this.controls.length; this.viewPortDiv.appendChild( div ); } } if(control.autoActivate) { control.activate(); } }, /** * APIMethod: getControl * * Parameters: * id - {String} ID of the control to return. * * Returns: * {} The control from the map's list of controls * which has a matching 'id'. If none found, * returns null. */ getControl: function (id) { var returnControl = null; for(var i=0, len=this.controls.length; i} The control to remove. */ removeControl: function (control) { //make sure control is non-null and actually part of our map if ( (control) && (control == this.getControl(control.id)) ) { if (control.div && (control.div.parentNode == this.viewPortDiv)) { this.viewPortDiv.removeChild(control.div); } OpenLayers.Util.removeItem(this.controls, control); } }, /********************************************************/ /* */ /* Popup Functions */ /* */ /* The following functions deal with adding and */ /* removing Popups to and from the Map */ /* */ /********************************************************/ /** * APIMethod: addPopup * * Parameters: * popup - {} * exclusive - {Boolean} If true, closes all other popups first */ addPopup: function(popup, exclusive) { if (exclusive) { //remove all other popups from screen for (var i = this.popups.length - 1; i >= 0; --i) { this.removePopup(this.popups[i]); } } popup.map = this; this.popups.push(popup); var popupDiv = popup.draw(); if (popupDiv) { popupDiv.style.zIndex = this.Z_INDEX_BASE['Popup'] + this.popups.length; this.layerContainerDiv.appendChild(popupDiv); } }, /** * APIMethod: removePopup * * Parameters: * popup - {} */ removePopup: function(popup) { OpenLayers.Util.removeItem(this.popups, popup); if (popup.div) { try { this.layerContainerDiv.removeChild(popup.div); } catch (e) { } // Popups sometimes apparently get disconnected // from the layerContainerDiv, and cause complaints. } popup.map = null; }, /********************************************************/ /* */ /* Container Div Functions */ /* */ /* The following functions deal with the access to */ /* and maintenance of the size of the container div */ /* */ /********************************************************/ /** * APIMethod: getSize * * Returns: * {} An object that represents the * size, in pixels, of the div into which OpenLayers * has been loaded. * Note - A clone() of this locally cached variable is * returned, so as not to allow users to modify it. */ getSize: function () { var size = null; if (this.size != null) { size = this.size.clone(); } return size; }, /** * APIMethod: updateSize * This function should be called by any external code which dynamically * changes the size of the map div (because mozilla wont let us catch * the "onresize" for an element) */ updateSize: function() { // the div might have moved on the page, also var newSize = this.getCurrentSize(); if (newSize && !isNaN(newSize.h) && !isNaN(newSize.w)) { this.events.clearMouseCache(); var oldSize = this.getSize(); if (oldSize == null) { this.size = oldSize = newSize; } if (!newSize.equals(oldSize)) { // store the new size this.size = newSize; //notify layers of mapresize for(var i=0, len=this.layers.length; i} A new object with the dimensions * of the map div */ getCurrentSize: function() { var size = new OpenLayers.Size(this.div.clientWidth, this.div.clientHeight); if (size.w == 0 && size.h == 0 || isNaN(size.w) && isNaN(size.h)) { size.w = this.div.offsetWidth; size.h = this.div.offsetHeight; } if (size.w == 0 && size.h == 0 || isNaN(size.w) && isNaN(size.h)) { size.w = parseInt(this.div.style.width); size.h = parseInt(this.div.style.height); } return size; }, /** * Method: calculateBounds * * Parameters: * center - {} Default is this.getCenter() * resolution - {float} Default is this.getResolution() * * Returns: * {} A bounds based on resolution, center, and * current mapsize. */ calculateBounds: function(center, resolution) { var extent = null; if (center == null) { center = this.getCachedCenter(); } if (resolution == null) { resolution = this.getResolution(); } if ((center != null) && (resolution != null)) { var halfWDeg = (this.size.w * resolution) / 2; var halfHDeg = (this.size.h * resolution) / 2; extent = new OpenLayers.Bounds(center.lon - halfWDeg, center.lat - halfHDeg, center.lon + halfWDeg, center.lat + halfHDeg); } return extent; }, /********************************************************/ /* */ /* Zoom, Center, Pan Functions */ /* */ /* The following functions handle the validation, */ /* getting and setting of the Zoom Level and Center */ /* as well as the panning of the Map */ /* */ /********************************************************/ /** * APIMethod: getCenter * * Returns: * {} */ getCenter: function () { var center = null; var cachedCenter = this.getCachedCenter(); if (cachedCenter) { center = cachedCenter.clone(); } return center; }, /** * Method: getCachedCenter * * Returns: * {} */ getCachedCenter: function() { if (!this.center && this.size) { this.center = this.getLonLatFromViewPortPx({ x: this.size.w / 2, y: this.size.h / 2 }); } return this.center; }, /** * APIMethod: getZoom * * Returns: * {Integer} */ getZoom: function () { return this.zoom; }, /** * APIMethod: pan * Allows user to pan by a value of screen pixels * * Parameters: * dx - {Integer} * dy - {Integer} * options - {Object} Options to configure panning: * - *animate* {Boolean} Use panTo instead of setCenter. Default is true. * - *dragging* {Boolean} Call setCenter with dragging true. Default is * false. */ pan: function(dx, dy, options) { options = OpenLayers.Util.applyDefaults(options, { animate: true, dragging: false }); if (options.dragging) { if (dx != 0 || dy != 0) { this.moveByPx(dx, dy); } } else { // getCenter var centerPx = this.getViewPortPxFromLonLat(this.getCachedCenter()); // adjust var newCenterPx = centerPx.add(dx, dy); if (this.dragging || !newCenterPx.equals(centerPx)) { var newCenterLonLat = this.getLonLatFromViewPortPx(newCenterPx); if (options.animate) { this.panTo(newCenterLonLat); } else { this.moveTo(newCenterLonLat); if(this.dragging) { this.dragging = false; this.events.triggerEvent("moveend"); } } } } }, /** * APIMethod: panTo * Allows user to pan to a new lonlat * If the new lonlat is in the current extent the map will slide smoothly * * Parameters: * lonlat - {} */ panTo: function(lonlat) { if (this.panTween && this.getExtent().scale(this.panRatio).containsLonLat(lonlat)) { var center = this.getCachedCenter(); // center will not change, don't do nothing if (lonlat.equals(center)) { return; } var from = this.getPixelFromLonLat(center); var to = this.getPixelFromLonLat(lonlat); var vector = { x: to.x - from.x, y: to.y - from.y }; var last = { x: 0, y: 0 }; this.panTween.start( { x: 0, y: 0 }, vector, this.panDuration, { callbacks: { eachStep: OpenLayers.Function.bind(function(px) { var x = px.x - last.x, y = px.y - last.y; this.moveByPx(x, y); last.x = Math.round(px.x); last.y = Math.round(px.y); }, this), done: OpenLayers.Function.bind(function(px) { this.moveTo(lonlat); this.dragging = false; this.events.triggerEvent("moveend"); }, this) } }); } else { this.setCenter(lonlat); } }, /** * APIMethod: setCenter * Set the map center (and optionally, the zoom level). * * Parameters: * lonlat - {|Array} The new center location. * If provided as array, the first value is the x coordinate, * and the 2nd value is the y coordinate. * zoom - {Integer} Optional zoom level. * dragging - {Boolean} Specifies whether or not to trigger * movestart/end events * forceZoomChange - {Boolean} Specifies whether or not to trigger zoom * change events (needed on baseLayer change) * * TBD: reconsider forceZoomChange in 3.0 */ setCenter: function(lonlat, zoom, dragging, forceZoomChange) { if (this.panTween) { this.panTween.stop(); } if (this.zoomTween) { this.zoomTween.stop(); } this.moveTo(lonlat, zoom, { 'dragging': dragging, 'forceZoomChange': forceZoomChange }); }, /** * Method: moveByPx * Drag the map by pixels. * * Parameters: * dx - {Number} * dy - {Number} */ moveByPx: function(dx, dy) { var hw = this.size.w / 2; var hh = this.size.h / 2; var x = hw + dx; var y = hh + dy; var wrapDateLine = this.baseLayer.wrapDateLine; var xRestriction = 0; var yRestriction = 0; if (this.restrictedExtent) { xRestriction = hw; yRestriction = hh; // wrapping the date line makes no sense for restricted extents wrapDateLine = false; } dx = wrapDateLine || x <= this.maxPx.x - xRestriction && x >= this.minPx.x + xRestriction ? Math.round(dx) : 0; dy = y <= this.maxPx.y - yRestriction && y >= this.minPx.y + yRestriction ? Math.round(dy) : 0; if (dx || dy) { if (!this.dragging) { this.dragging = true; this.events.triggerEvent("movestart"); } this.center = null; if (dx) { this.layerContainerOriginPx.x -= dx; this.minPx.x -= dx; this.maxPx.x -= dx; } if (dy) { this.layerContainerOriginPx.y -= dy; this.minPx.y -= dy; this.maxPx.y -= dy; } this.applyTransform(); var layer, i, len; for (i=0, len=this.layers.length; i's maxExtent. */ adjustZoom: function(zoom) { if (this.baseLayer && this.baseLayer.wrapDateLine) { var resolution, resolutions = this.baseLayer.resolutions, maxResolution = this.getMaxExtent().getWidth() / this.size.w; if (this.getResolutionForZoom(zoom) > maxResolution) { if (this.fractionalZoom) { zoom = this.getZoomForResolution(maxResolution); } else { for (var i=zoom|0, ii=resolutions.length; i set to true, this will be the * first zoom level that shows no more than one world width in the current * map viewport. Components that rely on this value (e.g. zoom sliders) * should also listen to the map's "updatesize" event and call this method * in the "updatesize" listener. * * Returns: * {Number} Minimum zoom level that shows a map not wider than its * 's maxExtent. This is an Integer value, unless the map is * configured with set to true. */ getMinZoom: function() { return this.adjustZoom(0); }, /** * Method: moveTo * * Parameters: * lonlat - {} * zoom - {Integer} * options - {Object} */ moveTo: function(lonlat, zoom, options) { if (lonlat != null && !(lonlat instanceof OpenLayers.LonLat)) { lonlat = new OpenLayers.LonLat(lonlat); } if (!options) { options = {}; } if (zoom != null) { zoom = parseFloat(zoom); if (!this.fractionalZoom) { zoom = Math.round(zoom); } } var requestedZoom = zoom; zoom = this.adjustZoom(zoom); if (zoom !== requestedZoom) { // zoom was adjusted, so keep old lonlat to avoid panning lonlat = this.getCenter(); } // dragging is false by default var dragging = options.dragging || this.dragging; // forceZoomChange is false by default var forceZoomChange = options.forceZoomChange; if (!this.getCachedCenter() && !this.isValidLonLat(lonlat)) { lonlat = this.maxExtent.getCenterLonLat(); this.center = lonlat.clone(); } if(this.restrictedExtent != null) { // In 3.0, decide if we want to change interpretation of maxExtent. if(lonlat == null) { lonlat = this.center; } if(zoom == null) { zoom = this.getZoom(); } var resolution = this.getResolutionForZoom(zoom); var extent = this.calculateBounds(lonlat, resolution); if(!this.restrictedExtent.containsBounds(extent)) { var maxCenter = this.restrictedExtent.getCenterLonLat(); if(extent.getWidth() > this.restrictedExtent.getWidth()) { lonlat = new OpenLayers.LonLat(maxCenter.lon, lonlat.lat); } else if(extent.left < this.restrictedExtent.left) { lonlat = lonlat.add(this.restrictedExtent.left - extent.left, 0); } else if(extent.right > this.restrictedExtent.right) { lonlat = lonlat.add(this.restrictedExtent.right - extent.right, 0); } if(extent.getHeight() > this.restrictedExtent.getHeight()) { lonlat = new OpenLayers.LonLat(lonlat.lon, maxCenter.lat); } else if(extent.bottom < this.restrictedExtent.bottom) { lonlat = lonlat.add(0, this.restrictedExtent.bottom - extent.bottom); } else if(extent.top > this.restrictedExtent.top) { lonlat = lonlat.add(0, this.restrictedExtent.top - extent.top); } } } var zoomChanged = forceZoomChange || ( (this.isValidZoomLevel(zoom)) && (zoom != this.getZoom()) ); var centerChanged = (this.isValidLonLat(lonlat)) && (!lonlat.equals(this.center)); // if neither center nor zoom will change, no need to do anything if (zoomChanged || centerChanged || dragging) { dragging || this.events.triggerEvent("movestart", { zoomChanged: zoomChanged }); if (centerChanged) { if (!zoomChanged && this.center) { // if zoom hasnt changed, just slide layerContainer // (must be done before setting this.center to new value) this.centerLayerContainer(lonlat); } this.center = lonlat.clone(); } var res = zoomChanged ? this.getResolutionForZoom(zoom) : this.getResolution(); // (re)set the layerContainerDiv's location if (zoomChanged || this.layerContainerOrigin == null) { this.layerContainerOrigin = this.getCachedCenter(); this.layerContainerOriginPx.x = 0; this.layerContainerOriginPx.y = 0; this.applyTransform(); var maxExtent = this.getMaxExtent({restricted: true}); var maxExtentCenter = maxExtent.getCenterLonLat(); var lonDelta = this.center.lon - maxExtentCenter.lon; var latDelta = maxExtentCenter.lat - this.center.lat; var extentWidth = Math.round(maxExtent.getWidth() / res); var extentHeight = Math.round(maxExtent.getHeight() / res); this.minPx = { x: (this.size.w - extentWidth) / 2 - lonDelta / res, y: (this.size.h - extentHeight) / 2 - latDelta / res }; this.maxPx = { x: this.minPx.x + Math.round(maxExtent.getWidth() / res), y: this.minPx.y + Math.round(maxExtent.getHeight() / res) }; } if (zoomChanged) { this.zoom = zoom; this.resolution = res; } var bounds = this.getExtent(); //send the move call to the baselayer and all the overlays if(this.baseLayer.visibility) { this.baseLayer.moveTo(bounds, zoomChanged, options.dragging); options.dragging || this.baseLayer.events.triggerEvent( "moveend", {zoomChanged: zoomChanged} ); } bounds = this.baseLayer.getExtent(); for (var i=this.layers.length-1; i>=0; --i) { var layer = this.layers[i]; if (layer !== this.baseLayer && !layer.isBaseLayer) { var inRange = layer.calculateInRange(); if (layer.inRange != inRange) { // the inRange property has changed. If the layer is // no longer in range, we turn it off right away. If // the layer is no longer out of range, the moveTo // call below will turn on the layer. layer.inRange = inRange; if (!inRange) { layer.display(false); } this.events.triggerEvent("changelayer", { layer: layer, property: "visibility" }); } if (inRange && layer.visibility) { layer.moveTo(bounds, zoomChanged, options.dragging); options.dragging || layer.events.triggerEvent( "moveend", {zoomChanged: zoomChanged} ); } } } this.events.triggerEvent("move"); dragging || this.events.triggerEvent("moveend"); if (zoomChanged) { //redraw popups for (var i=0, len=this.popups.length; i} */ centerLayerContainer: function (lonlat) { var originPx = this.getViewPortPxFromLonLat(this.layerContainerOrigin); var newPx = this.getViewPortPxFromLonLat(lonlat); if ((originPx != null) && (newPx != null)) { var oldLeft = this.layerContainerOriginPx.x; var oldTop = this.layerContainerOriginPx.y; var newLeft = Math.round(originPx.x - newPx.x); var newTop = Math.round(originPx.y - newPx.y); this.applyTransform( (this.layerContainerOriginPx.x = newLeft), (this.layerContainerOriginPx.y = newTop)); var dx = oldLeft - newLeft; var dy = oldTop - newTop; this.minPx.x -= dx; this.maxPx.x -= dx; this.minPx.y -= dy; this.maxPx.y -= dy; } }, /** * Method: isValidZoomLevel * * Parameters: * zoomLevel - {Integer} * * Returns: * {Boolean} Whether or not the zoom level passed in is non-null and * within the min/max range of zoom levels. */ isValidZoomLevel: function(zoomLevel) { return ( (zoomLevel != null) && (zoomLevel >= 0) && (zoomLevel < this.getNumZoomLevels()) ); }, /** * Method: isValidLonLat * * Parameters: * lonlat - {} * * Returns: * {Boolean} Whether or not the lonlat passed in is non-null and within * the maxExtent bounds */ isValidLonLat: function(lonlat) { var valid = false; if (lonlat != null) { var maxExtent = this.getMaxExtent(); var worldBounds = this.baseLayer.wrapDateLine && maxExtent; valid = maxExtent.containsLonLat(lonlat, {worldBounds: worldBounds}); } return valid; }, /********************************************************/ /* */ /* Layer Options */ /* */ /* Accessor functions to Layer Options parameters */ /* */ /********************************************************/ /** * APIMethod: getProjection * This method returns a string representing the projection. In * the case of projection support, this will be the srsCode which * is loaded -- otherwise it will simply be the string value that * was passed to the projection at startup. * * FIXME: In 3.0, we will remove getProjectionObject, and instead * return a Projection object from this function. * * Returns: * {String} The Projection string from the base layer or null. */ getProjection: function() { var projection = this.getProjectionObject(); return projection ? projection.getCode() : null; }, /** * APIMethod: getProjectionObject * Returns the projection obect from the baselayer. * * Returns: * {} The Projection of the base layer. */ getProjectionObject: function() { var projection = null; if (this.baseLayer != null) { projection = this.baseLayer.projection; } return projection; }, /** * APIMethod: getMaxResolution * * Returns: * {String} The Map's Maximum Resolution */ getMaxResolution: function() { var maxResolution = null; if (this.baseLayer != null) { maxResolution = this.baseLayer.maxResolution; } return maxResolution; }, /** * APIMethod: getMaxExtent * * Parameters: * options - {Object} * * Allowed Options: * restricted - {Boolean} If true, returns restricted extent (if it is * available.) * * Returns: * {} The maxExtent property as set on the current * baselayer, unless the 'restricted' option is set, in which case * the 'restrictedExtent' option from the map is returned (if it * is set). */ getMaxExtent: function (options) { var maxExtent = null; if(options && options.restricted && this.restrictedExtent){ maxExtent = this.restrictedExtent; } else if (this.baseLayer != null) { maxExtent = this.baseLayer.maxExtent; } return maxExtent; }, /** * APIMethod: getNumZoomLevels * * Returns: * {Integer} The total number of zoom levels that can be displayed by the * current baseLayer. */ getNumZoomLevels: function() { var numZoomLevels = null; if (this.baseLayer != null) { numZoomLevels = this.baseLayer.numZoomLevels; } return numZoomLevels; }, /********************************************************/ /* */ /* Baselayer Functions */ /* */ /* The following functions, all publicly exposed */ /* in the API?, are all merely wrappers to the */ /* the same calls on whatever layer is set as */ /* the current base layer */ /* */ /********************************************************/ /** * APIMethod: getExtent * * Returns: * {} A Bounds object which represents the lon/lat * bounds of the current viewPort. * If no baselayer is set, returns null. */ getExtent: function () { var extent = null; if (this.baseLayer != null) { extent = this.baseLayer.getExtent(); } return extent; }, /** * APIMethod: getResolution * * Returns: * {Float} The current resolution of the map. * If no baselayer is set, returns null. */ getResolution: function () { var resolution = null; if (this.baseLayer != null) { resolution = this.baseLayer.getResolution(); } else if(this.allOverlays === true && this.layers.length > 0) { // while adding the 1st layer to the map in allOverlays mode, // this.baseLayer is not set yet when we need the resolution // for calculateInRange. resolution = this.layers[0].getResolution(); } return resolution; }, /** * APIMethod: getUnits * * Returns: * {Float} The current units of the map. * If no baselayer is set, returns null. */ getUnits: function () { var units = null; if (this.baseLayer != null) { units = this.baseLayer.units; } return units; }, /** * APIMethod: getScale * * Returns: * {Float} The current scale denominator of the map. * If no baselayer is set, returns null. */ getScale: function () { var scale = null; if (this.baseLayer != null) { var res = this.getResolution(); var units = this.baseLayer.units; scale = OpenLayers.Util.getScaleFromResolution(res, units); } return scale; }, /** * APIMethod: getZoomForExtent * * Parameters: * 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} A suitable zoom level for the specified bounds. * If no baselayer is set, returns null. */ getZoomForExtent: function (bounds, closest) { var zoom = null; if (this.baseLayer != null) { zoom = this.baseLayer.getZoomForExtent(bounds, closest); } return zoom; }, /** * APIMethod: getResolutionForZoom * * Parameters: * zoom - {Float} * * Returns: * {Float} A suitable resolution for the specified zoom. If no baselayer * is set, returns null. */ getResolutionForZoom: function(zoom) { var resolution = null; if(this.baseLayer) { resolution = this.baseLayer.getResolutionForZoom(zoom); } return resolution; }, /** * 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} A suitable zoom level for the specified resolution. * If no baselayer is set, returns null. */ getZoomForResolution: function(resolution, closest) { var zoom = null; if (this.baseLayer != null) { zoom = this.baseLayer.getZoomForResolution(resolution, closest); } return zoom; }, /********************************************************/ /* */ /* Zooming Functions */ /* */ /* The following functions, all publicly exposed */ /* in the API, are all merely wrappers to the */ /* the setCenter() function */ /* */ /********************************************************/ /** * APIMethod: zoomTo * Zoom to a specific zoom level. Zooming will be animated unless the map * is configured with {zoomMethod: null}. To zoom without animation, use * without a lonlat argument. * * Parameters: * zoom - {Integer} */ zoomTo: function(zoom, xy) { // non-API arguments: // xy - {} optional zoom origin var map = this; if (map.isValidZoomLevel(zoom)) { if (map.baseLayer.wrapDateLine) { zoom = map.adjustZoom(zoom); } if (map.zoomTween) { var currentRes = map.getResolution(), targetRes = map.getResolutionForZoom(zoom), start = {scale: 1}, end = {scale: currentRes / targetRes}; if (map.zoomTween.playing && map.zoomTween.duration < 3 * map.zoomDuration) { // update the end scale, and reuse the running zoomTween map.zoomTween.finish = { scale: map.zoomTween.finish.scale * end.scale }; } else { if (!xy) { var size = map.getSize(); xy = {x: size.w / 2, y: size.h / 2}; } map.zoomTween.start(start, end, map.zoomDuration, { minFrameRate: 50, // don't spend much time zooming callbacks: { eachStep: function(data) { var containerOrigin = map.layerContainerOriginPx, scale = data.scale, dx = ((scale - 1) * (containerOrigin.x - xy.x)) | 0, dy = ((scale - 1) * (containerOrigin.y - xy.y)) | 0; map.applyTransform(containerOrigin.x + dx, containerOrigin.y + dy, scale); }, done: function(data) { map.applyTransform(); var resolution = map.getResolution() / data.scale, zoom = map.getZoomForResolution(resolution, true) map.moveTo(map.getZoomTargetCenter(xy, resolution), zoom, true); } } }); } } else { var center = xy ? map.getZoomTargetCenter(xy, map.getResolutionForZoom(zoom)) : null; map.setCenter(center, zoom); } } }, /** * APIMethod: zoomIn * */ zoomIn: function() { this.zoomTo(this.getZoom() + 1); }, /** * APIMethod: zoomOut * */ zoomOut: function() { this.zoomTo(this.getZoom() - 1); }, /** * APIMethod: zoomToExtent * Zoom to the passed in bounds, recenter * * Parameters: * bounds - {|Array} If provided as an array, the array * should consist of four values (left, bottom, right, top). * 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. * */ zoomToExtent: function(bounds, closest) { if (!(bounds instanceof OpenLayers.Bounds)) { bounds = new OpenLayers.Bounds(bounds); } var center = bounds.getCenterLonLat(); if (this.baseLayer.wrapDateLine) { var maxExtent = this.getMaxExtent(); //fix straddling bounds (in the case of a bbox that straddles the // dateline, it's left and right boundaries will appear backwards. // we fix this by allowing a right value that is greater than the // max value at the dateline -- this allows us to pass a valid // bounds to calculate zoom) // bounds = bounds.clone(); while (bounds.right < bounds.left) { bounds.right += maxExtent.getWidth(); } //if the bounds was straddling (see above), then the center point // we got from it was wrong. So we take our new bounds and ask it // for the center. // center = bounds.getCenterLonLat().wrapDateLine(maxExtent); } this.setCenter(center, this.getZoomForExtent(bounds, closest)); }, /** * APIMethod: zoomToMaxExtent * Zoom to the full extent and recenter. * * Parameters: * options - {Object} * * Allowed Options: * restricted - {Boolean} True to zoom to restricted extent if it is * set. Defaults to true. */ zoomToMaxExtent: function(options) { //restricted is true by default var restricted = (options) ? options.restricted : true; var maxExtent = this.getMaxExtent({ 'restricted': restricted }); this.zoomToExtent(maxExtent); }, /** * APIMethod: zoomToScale * Zoom to a specified scale * * Parameters: * scale - {float} * closest - {Boolean} Find the zoom level that most closely fits the * specified scale. Note that this may result in a zoom that does * not exactly contain the entire extent. * Default is false. * */ zoomToScale: function(scale, closest) { var res = OpenLayers.Util.getResolutionFromScale(scale, this.baseLayer.units); var halfWDeg = (this.size.w * res) / 2; var halfHDeg = (this.size.h * res) / 2; var center = this.getCachedCenter(); var extent = new OpenLayers.Bounds(center.lon - halfWDeg, center.lat - halfHDeg, center.lon + halfWDeg, center.lat + halfHDeg); this.zoomToExtent(extent, closest); }, /********************************************************/ /* */ /* Translation Functions */ /* */ /* The following functions translate between */ /* LonLat, LayerPx, and ViewPortPx */ /* */ /********************************************************/ // // TRANSLATION: LonLat <-> ViewPortPx // /** * Method: getLonLatFromViewPortPx * * Parameters: * viewPortPx - {|Object} An OpenLayers.Pixel or * an object with a 'x' * and 'y' properties. * * Returns: * {} An OpenLayers.LonLat which is the passed-in view * port , translated into lon/lat * by the current base layer. */ getLonLatFromViewPortPx: function (viewPortPx) { var lonlat = null; if (this.baseLayer != null) { lonlat = this.baseLayer.getLonLatFromViewPortPx(viewPortPx); } return lonlat; }, /** * APIMethod: getViewPortPxFromLonLat * * Parameters: * lonlat - {} * * Returns: * {} An OpenLayers.Pixel which is the passed-in * , translated into view port * pixels by the current base layer. */ getViewPortPxFromLonLat: function (lonlat) { var px = null; if (this.baseLayer != null) { px = this.baseLayer.getViewPortPxFromLonLat(lonlat); } return px; }, /** * Method: getZoomTargetCenter * * Parameters: * xy - {} The zoom origin pixel location on the screen * resolution - {Float} The resolution we want to get the center for * * Returns: * {} The location of the map center after the * transformation described by the origin xy and the target resolution. */ getZoomTargetCenter: function (xy, resolution) { var lonlat = null, size = this.getSize(), deltaX = size.w/2 - xy.x, deltaY = xy.y - size.h/2, zoomPoint = this.getLonLatFromPixel(xy); if (zoomPoint) { lonlat = new OpenLayers.LonLat( zoomPoint.lon + deltaX * resolution, zoomPoint.lat + deltaY * resolution ); } return lonlat; }, // // CONVENIENCE TRANSLATION FUNCTIONS FOR API // /** * APIMethod: getLonLatFromPixel * * Parameters: * px - {|Object} An OpenLayers.Pixel or an object with * a 'x' and 'y' properties. * * Returns: * {} An OpenLayers.LonLat corresponding to the given * OpenLayers.Pixel, translated into lon/lat by the * current base layer */ getLonLatFromPixel: function (px) { return this.getLonLatFromViewPortPx(px); }, /** * APIMethod: getPixelFromLonLat * Returns a pixel location given a map location. The map location is * translated to an integer pixel location (in viewport pixel * coordinates) by the current base layer. * * Parameters: * lonlat - {} A map location. * * Returns: * {} An OpenLayers.Pixel corresponding to the * translated into view port pixels by the current * base layer. */ getPixelFromLonLat: function (lonlat) { var px = this.getViewPortPxFromLonLat(lonlat); px.x = Math.round(px.x); px.y = Math.round(px.y); return px; }, /** * Method: getGeodesicPixelSize * * Parameters: * px - {} The pixel to get the geodesic length for. If * not provided, the center pixel of the map viewport will be used. * * Returns: * {} The geodesic size of the pixel in kilometers. */ getGeodesicPixelSize: function(px) { var lonlat = px ? this.getLonLatFromPixel(px) : ( this.getCachedCenter() || new OpenLayers.LonLat(0, 0)); var res = this.getResolution(); var left = lonlat.add(-res / 2, 0); var right = lonlat.add(res / 2, 0); var bottom = lonlat.add(0, -res / 2); var top = lonlat.add(0, res / 2); var dest = new OpenLayers.Projection("EPSG:4326"); var source = this.getProjectionObject() || dest; if(!source.equals(dest)) { left.transform(source, dest); right.transform(source, dest); bottom.transform(source, dest); top.transform(source, dest); } return new OpenLayers.Size( OpenLayers.Util.distVincenty(left, right), OpenLayers.Util.distVincenty(bottom, top) ); }, // // TRANSLATION: ViewPortPx <-> LayerPx // /** * APIMethod: getViewPortPxFromLayerPx * * Parameters: * layerPx - {} * * Returns: * {} Layer Pixel translated into ViewPort Pixel * coordinates */ getViewPortPxFromLayerPx:function(layerPx) { var viewPortPx = null; if (layerPx != null) { var dX = this.layerContainerOriginPx.x; var dY = this.layerContainerOriginPx.y; viewPortPx = layerPx.add(dX, dY); } return viewPortPx; }, /** * APIMethod: getLayerPxFromViewPortPx * * Parameters: * viewPortPx - {} * * Returns: * {} ViewPort Pixel translated into Layer Pixel * coordinates */ getLayerPxFromViewPortPx:function(viewPortPx) { var layerPx = null; if (viewPortPx != null) { var dX = -this.layerContainerOriginPx.x; var dY = -this.layerContainerOriginPx.y; layerPx = viewPortPx.add(dX, dY); if (isNaN(layerPx.x) || isNaN(layerPx.y)) { layerPx = null; } } return layerPx; }, // // TRANSLATION: LonLat <-> LayerPx // /** * Method: getLonLatFromLayerPx * * Parameters: * px - {} * * Returns: * {} */ getLonLatFromLayerPx: function (px) { //adjust for displacement of layerContainerDiv px = this.getViewPortPxFromLayerPx(px); return this.getLonLatFromViewPortPx(px); }, /** * APIMethod: getLayerPxFromLonLat * * Parameters: * lonlat - {} lonlat * * Returns: * {} An OpenLayers.Pixel which is the passed-in * , translated into layer pixels * by the current base layer */ getLayerPxFromLonLat: function (lonlat) { //adjust for displacement of layerContainerDiv var px = this.getPixelFromLonLat(lonlat); return this.getLayerPxFromViewPortPx(px); }, /** * Method: applyTransform * Applies the given transform to the . This method has * a 2-stage fallback from translate3d/scale3d via translate/scale to plain * style.left/style.top, in which case no scaling is supported. * * Parameters: * x - {Number} x parameter for the translation. Defaults to the x value of * the map's * y - {Number} y parameter for the translation. Defaults to the y value of * the map's * scale - {Number} scale. Defaults to 1 if not provided. */ applyTransform: function(x, y, scale) { scale = scale || 1; var origin = this.layerContainerOriginPx, needTransform = scale !== 1; x = x || origin.x; y = y || origin.y; var style = this.layerContainerDiv.style, transform = this.applyTransform.transform, template = this.applyTransform.template; if (transform === undefined) { transform = OpenLayers.Util.vendorPrefix.style('transform'); this.applyTransform.transform = transform; if (transform) { // Try translate3d, but only if the viewPortDiv has a transform // defined in a stylesheet var computedStyle = OpenLayers.Element.getStyle(this.viewPortDiv, OpenLayers.Util.vendorPrefix.css('transform')); if (!computedStyle || computedStyle !== 'none') { template = ['translate3d(', ',0) ', 'scale3d(', ',1)']; style[transform] = [template[0], '0,0', template[1]].join(''); } // If no transform is defined in the stylesheet or translate3d // does not stick, use translate and scale if (!template || !~style[transform].indexOf(template[0])) { template = ['translate(', ') ', 'scale(', ')']; } this.applyTransform.template = template; } } // If we do 3d transforms, we always want to use them. If we do 2d // transforms, we only use them when we need to. if (transform !== null && (template[0] === 'translate3d(' || needTransform === true)) { // Our 2d transforms are combined with style.left and style.top, so // adjust x and y values and set the origin as left and top if (needTransform === true && template[0] === 'translate(') { x -= origin.x; y -= origin.y; style.left = origin.x + 'px'; style.top = origin.y + 'px'; } style[transform] = [ template[0], x, 'px,', y, 'px', template[1], template[2], scale, ',', scale, template[3] ].join(''); } else { style.left = x + 'px'; style.top = y + 'px'; // We previously might have had needTransform, so remove transform if (transform !== null) { style[transform] = ''; } } }, CLASS_NAME: "OpenLayers.Map" }); /** * Constant: TILE_WIDTH * {Integer} 256 Default tile width (unless otherwise specified) */ OpenLayers.Map.TILE_WIDTH = 256; /** * Constant: TILE_HEIGHT * {Integer} 256 Default tile height (unless otherwise specified) */ OpenLayers.Map.TILE_HEIGHT = 256; /** FILE: 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" }); /** FILE: 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} */ 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" } }; /** FILE: 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 * {} 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()} */ 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 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 will extend the symbolizer * of every rule. Properties of the 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()} List of rules to be added to the * style. * * Returns: * {} */ 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} 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 0) { appliedRules = true; for(var i=0, len=elseRules.length; i 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 - {} * style - {Object} * feature - {} * * 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 * . * * 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)} */ 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 - {} * * Returns: * {String} key of the according symbolizer */ getSymbolizerPrefix: function(geometry) { var prefixes = OpenLayers.Style.SYMBOLIZER_PREFIXES; for (var i=0, len=prefixes.length; i} 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} optional feature to pass to * 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']; /** FILE: 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 * {} 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: * {} */ 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 - {} 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 - {} 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: * {} 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 and ). * * No default set here. Use OpenLayers.Renderer.defaultRenderer for defaults. */ /** * 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. */ /** * 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); }, CLASS_NAME: "OpenLayers.Symbolizer.Point" }); /** FILE: OpenLayers/Symbolizer/Line.js **/ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for * full list of contributors). Published 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.Line * A symbolizer used to render line features. */ OpenLayers.Symbolizer.Line = OpenLayers.Class(OpenLayers.Symbolizer, { /** * 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. */ /** * APIProperty: strokeOpacity * {Number} Stroke opacity (0-1). * * No default set here. Use OpenLayers.Renderer.defaultRenderer for defaults. */ /** * 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. */ /** * Constructor: OpenLayers.Symbolizer.Line * Create a symbolizer for rendering lines. * * 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 line symbolizer. */ initialize: function(config) { OpenLayers.Symbolizer.prototype.initialize.apply(this, arguments); }, CLASS_NAME: "OpenLayers.Symbolizer.Line" }); /** FILE: 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, { /** * 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. */ /** * APIProperty: strokeOpacity * {Number} Stroke opacity (0-1). * * No default set here. Use OpenLayers.Renderer.defaultRenderer for defaults. */ /** * 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: * 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. */ initialize: function(config) { OpenLayers.Symbolizer.prototype.initialize.apply(this, arguments); }, CLASS_NAME: "OpenLayers.Symbolizer.Polygon" }); /** FILE: 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, { /** * APIProperty: label * {String} The text for the label. * * No default set here. Use OpenLayers.Renderer.defaultRenderer for defaults. */ /** * APIProperty: fontFamily * {String} The font family for the label. * * No default set here. Use OpenLayers.Renderer.defaultRenderer for defaults. */ /** * APIProperty: fontSize * {String} The font size for the label. * * No default set here. Use OpenLayers.Renderer.defaultRenderer for defaults. */ /** * APIProperty: fontWeight * {String} The font weight for the label. * * No default set here. Use OpenLayers.Renderer.defaultRenderer for defaults. */ /** * 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: * A new text symbolizer. */ initialize: function(config) { OpenLayers.Symbolizer.prototype.initialize.apply(this, arguments); }, CLASS_NAME: "OpenLayers.Symbolizer.Text" }); /** FILE: 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. * * 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.Symbolizer.Raster" }); /** FILE: 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({ /** * Property: id * {String} A unique id for this session. */ id: null, /** * APIProperty: name * {String} Style identifier. */ name: null, /** * APIProperty: title * {String} Title of this style. */ title: null, /** * APIProperty: description * {String} Description of this style. */ description: null, /** * APIProperty: layerName * {} 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, /** * APIProperty: rules * {Array()} Collection of rendering rules. */ rules: null, /** * Constructor: OpenLayers.Style2 * Creates a style representing a collection of rendering rules. * * 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. */ 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} 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} 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, /** * 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 */ 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" }); /** FILE: 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 document. * The DOM creation and traversing methods exposed here all mimic the * W3C XML DOM methods. Create a new parser with the * constructor. * * Inherits from: * - */ 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 * 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 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 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 document * 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 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), 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 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 on the instance. On other browsers, this will * either return an existing or create a new shared document (see * ). * * 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; /** FILE: 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 constructor. This is a base class, * typical geometry types are described by subclasses of this class. * * Note that if you use the 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 * {}This is set when a Geometry is added as component * of another geometry */ parent: null, /** * Property: 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: * {} An exact clone of this geometry. */ clone: function() { return new OpenLayers.Geometry(); }, /** * Method: setBounds * Set the bounds for this Geometry. * * Parameters: * 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 - {} */ 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: * {} */ 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 - {} 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 - {|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: * {} 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: * {} 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= 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 | } 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 }; }; /** FILE: 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.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: * {} 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 - {} 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 - {} 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. "5, 42") */ 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 - {} 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: * {} 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 - {} Point of origin for resizing * ratio - {Float} Optional x:y ratio for resizing. Default ratio is 1. * * Returns: * {} - 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 - {} 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 - {} * dest - {} * * Returns: * {} */ 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" }); /** FILE: 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 (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 and functions here merely iterate through * the components, summing their respective areas and lengths. * * Create a new instance with the constructor. * * Inherits from: * - */ OpenLayers.Geometry.Collection = OpenLayers.Class(OpenLayers.Geometry, { /** * APIProperty: components * {Array()} 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()} 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: * {} An exact clone of this collection */ clone: function() { var Constructor = OpenLayers.Util.getConstructor(this.CLASS_NAME); var geometry = new Constructor(); for(var i=0, len=this.components.length; i)} An array of geometries to add */ addComponents: function(components){ if(!(OpenLayers.Util.isArray(components))) { components = [components]; } for(var i=0, len=components.length; i} 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()} 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 - {} * * 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. * * 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} 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} 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 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} 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} Center point for the rotation */ rotate: function(angle, origin) { for(var i=0, len=this.components.length; i} Point of origin for resizing * ratio - {Float} Optional x:y ratio for resizing. Default ratio is 1. * * Returns: * {} - The current geometry. */ resize: function(scale, origin, ratio) { for(var i=0; i} 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} 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} * dest - {} * * Returns: * {} */ transform: function(source, dest) { if (source && dest) { for (var i=0, len=this.components.length; i} 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 constructor. * * Inherits from: * - * - */ 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()} * * Returns: * {} */ /** * APIMethod: addPoint * Wrapper for * * Parameters: * point - {} Point to be added * index - {Integer} Optional index */ addPoint: function(point, index) { this.addComponent(point, index); }, /** * APIMethod: removePoint * Wrapper for * * Parameters: * point - {} Point to be removed */ removePoint: function(point){ this.removeComponent(point); }, CLASS_NAME: "OpenLayers.Geometry.MultiPoint" }); /** FILE: 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.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()} */ /** * 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} 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 */ OpenLayers.Geometry.LineString = OpenLayers.Class(OpenLayers.Geometry.Curve, { /** * Constructor: OpenLayers.Geometry.LineString * Create a new LineString geometry * * Parameters: * points - {Array()} 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 - {} 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 - {} * * 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 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 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 - {} 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 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 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 - {} 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 - {} 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 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 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" }); /** FILE: 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 * components. * * Inherits from: * - * - */ 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()} * */ /** * Method: split * Use this geometry (the source) to attempt to split a target geometry. * * Parameters: * 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 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 - {} 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 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" }); /** FILE: 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.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()} 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 - {} * 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 - {} * * 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} Center point for the rotation */ rotate: function(angle, origin) { for(var i=0, len=this.components.length; i} Point of origin for resizing * ratio - {Float} Optional x:y ratio for resizing. Default ratio is 1. * * Returns: * {} - The current geometry. */ resize: function(scale, origin, ratio) { for(var i=0, len=this.components.length; i} * dest - {} * * Returns: * {} */ transform: function(source, dest) { if (source && dest) { for (var i=0, len=this.components.length; i} 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} 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} * * 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= 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 - {} 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 * - */ 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()} */ /** * 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} 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} * * 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} 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} 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 - {} 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 * components. Create a new instance with the * constructor. * * Inherits from: * - */ 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()} An array of polygons * used to generate the MultiPolygon * */ CLASS_NAME: "OpenLayers.Geometry.MultiPolygon" }); /** FILE: 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 * constructor. Supports the GML simple features profile. * * Inherits from: * - */ 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()} 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 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} A point geometry. */ point: function(node) { /** * Three coordinate variations to consider: * 1) x y z * 2) x, y, z * 3) xy */ var nodeList, coordString; var coords = []; // look for 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 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 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: * {} 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} A linestring geometry. */ linestring: function(node, ring) { /** * Two coordinate variations to consider: * 1) x0 y0 z0 x1 y1 z1 * 2) x0, y0, z0 x1, y1, z1 */ var nodeList, coordString; var coords = []; var points = []; // look for 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 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} 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} 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} 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 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: * {} 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 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()} 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} 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 - {} 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 - {} 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} 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 - {} 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} 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 - {} 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} 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} 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) * ... * (end) * * Parameters: * 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 */ 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 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 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 , * but auto-configured and during read. * Subclasses making use of auto-configuration should make * the first call to the 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 * or 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()} 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 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) | 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 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" }); /** FILE: 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.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} * 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 */ OpenLayers.Format.GML.v3 = OpenLayers.Class(OpenLayers.Format.GML.Base, { /** * Property: schemaLocation * {String} Schema location for a particular minor version. The writers * conform with the Simple Features Profile for GML. */ schemaLocation: "http://www.opengis.net/gml http://schemas.opengis.net/gml/3.1.1/profiles/gmlsfProfile/1.0.0/gmlsf.xsd", /** * Property: curve * {Boolean} Write gml:Curve instead of gml:LineString elements. This also * affects the elements in multi-part geometries. Default is false. * To write gml:Curve elements instead of gml:LineString, set curve * to true in the options to the contstructor (cannot be changed after * instantiation). */ curve: false, /** * Property: multiCurve * {Boolean} Write gml:MultiCurve instead of gml:MultiLineString. Since * the latter is deprecated in GML 3, the default is true. To write * gml:MultiLineString instead of gml:MultiCurve, set multiCurve to * false in the options to the constructor (cannot be changed after * instantiation). */ multiCurve: true, /** * Property: surface * {Boolean} Write gml:Surface instead of gml:Polygon elements. This also * affects the elements in multi-part geometries. Default is false. * To write gml:Surface elements instead of gml:Polygon, set surface * to true in the options to the contstructor (cannot be changed after * instantiation). */ surface: false, /** * Property: multiSurface * {Boolean} Write gml:multiSurface instead of gml:MultiPolygon. Since * the latter is deprecated in GML 3, the default is true. To write * gml:MultiPolygon instead of gml:multiSurface, set multiSurface to * false in the options to the constructor (cannot be changed after * instantiation). */ multiSurface: true, /** * Constructor: OpenLayers.Format.GML.v3 * Create a parser for GML v3. * * 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({ "_inherit": function(node, obj, container) { // SRSReferenceGroup attributes var dim = parseInt(node.getAttribute("srsDimension"), 10) || (container && container.srsDimension); if (dim) { obj.srsDimension = dim; } }, "featureMembers": function(node, obj) { this.readChildNodes(node, obj); }, "Curve": function(node, container) { var obj = {points: []}; 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) ); }, "segments": function(node, obj) { this.readChildNodes(node, obj); }, "LineStringSegment": function(node, container) { var obj = {}; this.readChildNodes(node, obj); if(obj.points) { Array.prototype.push.apply(container.points, obj.points); } }, "pos": function(node, obj) { var str = this.getChildValue(node).replace( this.regExes.trimSpace, "" ); var coords = str.split(this.regExes.splitSpace); var point; if(this.xy) { point = new OpenLayers.Geometry.Point( coords[0], coords[1], coords[2] ); } else { point = new OpenLayers.Geometry.Point( coords[1], coords[0], coords[2] ); } obj.points = [point]; }, "posList": function(node, obj) { var str = this.getChildValue(node).replace( this.regExes.trimSpace, "" ); var coords = str.split(this.regExes.splitSpace); // The "dimension" attribute is from the GML 3.0.1 spec. var dim = obj.srsDimension || parseInt(node.getAttribute("srsDimension") || node.getAttribute("dimension"), 10) || 2; var j, x, y, z; var numPoints = coords.length / dim; var points = new Array(numPoints); for(var i=0, len=coords.length; i 0) { container.components = [ new OpenLayers.Geometry.MultiLineString(obj.components) ]; } }, "curveMember": function(node, obj) { this.readChildNodes(node, obj); }, "MultiSurface": function(node, container) { var obj = {components: []}; this.readers.gml._inherit.apply(this, [node, obj, container]); this.readChildNodes(node, obj); if(obj.components.length > 0) { container.components = [ new OpenLayers.Geometry.MultiPolygon(obj.components) ]; } }, "surfaceMember": function(node, obj) { this.readChildNodes(node, obj); }, "surfaceMembers": function(node, obj) { this.readChildNodes(node, obj); }, "pointMembers": function(node, obj) { this.readChildNodes(node, obj); }, "lineStringMembers": function(node, obj) { this.readChildNodes(node, obj); }, "polygonMembers": function(node, obj) { this.readChildNodes(node, obj); }, "geometryMembers": function(node, obj) { this.readChildNodes(node, obj); }, "Envelope": function(node, container) { var obj = {points: new Array(2)}; 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) ); }, "lowerCorner": function(node, container) { var obj = {}; this.readers.gml.pos.apply(this, [node, obj]); container.points[0] = obj.points[0]; }, "upperCorner": function(node, container) { var obj = {}; this.readers.gml.pos.apply(this, [node, obj]); container.points[1] = obj.points[0]; } }, 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} * 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": OpenLayers.Util.applyDefaults({ "featureMembers": function(features) { var node = this.createElementNSPlus("gml:featureMembers"); for(var i=0, len=features.length; i mapping. */ setGeometryTypes: function() { this.geometryTypes = { "OpenLayers.Geometry.Point": "Point", "OpenLayers.Geometry.MultiPoint": "MultiPoint", "OpenLayers.Geometry.LineString": (this.curve === true) ? "Curve": "LineString", "OpenLayers.Geometry.MultiLineString": (this.multiCurve === false) ? "MultiLineString" : "MultiCurve", "OpenLayers.Geometry.Polygon": (this.surface === true) ? "Surface" : "Polygon", "OpenLayers.Geometry.MultiPolygon": (this.multiSurface === false) ? "MultiPolygon" : "MultiSurface", "OpenLayers.Geometry.Collection": "GeometryCollection" }; }, CLASS_NAME: "OpenLayers.Format.GML.v3" }); /** FILE: 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.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" }); /** FILE: 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.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: * {} */ 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" }); /** FILE: 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/VersionedOGC.js */ /** * Class: OpenLayers.Format.WMSDescribeLayer * Read SLD WMS DescribeLayer response * DescribeLayer is meant to couple WMS to WFS and WCS * * Inherits from: * - */ 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 {} 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" }); /** FILE: 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.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 {} 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 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" }); // Version alias - workaround for http://trac.osgeo.org/mapserver/ticket/2257 OpenLayers.Format.WMSDescribeLayer.v1_1_0 = OpenLayers.Format.WMSDescribeLayer.v1_1_1; /** FILE: 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.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} */ 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" }); /** FILE: 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.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: * {} */ initialize: function(options) { this.fids = []; OpenLayers.Filter.prototype.initialize.apply(this, [options]); }, /** * APIMethod: evaluate * evaluates this rule for a specific feature * * Parameters: * 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} 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" }); /** FILE: 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.Logical = OpenLayers.Class(OpenLayers.Filter, { /** * APIProperty: filters * {Array()} 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: * {} */ 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} Clone of this filter. */ clone: function() { var filters = []; for(var i=0, len=this.filters.length; i */ 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: * {} */ 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 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: * {} 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"; /** FILE: 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.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 * { || } 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: * {} */ /** * Method: evaluate * Evaluates this filter for a specific feature. * * Parameters: * feature - {} 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: * {} 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"; /** FILE: 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 * constructor. * * Inherits from: * - */ 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" }); /** FILE: 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 * constructor. * * Inherits from: * - */ 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 - {} 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: * {} A filter object. */ CLASS_NAME: "OpenLayers.Format.Filter" }); /** FILE: 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.Function = OpenLayers.Class(OpenLayers.Filter, { /** * APIProperty: name * {String} Name of the function. */ name: null, /** * APIProperty: params * {Array( || 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: * {} */ CLASS_NAME: "OpenLayers.Filter.Function" }); /** FILE: 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; } }; /** FILE: 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.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 * 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: * {} 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 attribute"} 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 {} 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: * {} 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 * 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 * ( || String || Number) * * Parameters: * value - ( || 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 - {} 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": "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" }); /** FILE: 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.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 * 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 {} filter and converts it into XML. * * Parameters: * filter - {} 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" }); /** FILE: 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.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 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" }, /** * Constructor: OpenLayers.Format.SLD.v1 * Instances of this class are not created directly. Use the * constructor instead. * * Parameters: * options - {Object} An optional object whose properties will be set on * this instance. */ /** * Method: read * * 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. */ read: function(data, options) { options = OpenLayers.Util.applyDefaults(options, this.options); var sld = { namedLayers: options.namedLayersAsArray === true ? [] : {} }; this.readChildNodes(data, sld); return sld; }, /** * 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": { "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 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. * * Parameters: * sym - {String} A symbolizer property name. * * Returns: * {String} A CSS property name or null if none found. */ getCssProperty: function(sym) { var css = null; for(var prop in this.cssMap) { if(this.cssMap[prop] == sym) { css = prop; break; } } 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 * if one of the known is not * the file extension of the provided href. * * Parameters: * href - {String} * * Returns: * {String} The graphic format. */ 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; }, /** * Property: defaultGraphicFormat * {String} If none other can be determined from , this * default will be returned. */ defaultGraphicFormat: "image/png", /** * Property: graphicFormats * {Object} Mapping of image mime-types to regular extensions matching * well-known file extensions. */ graphicFormats: { "image/jpeg": /\.jpe?g$/i, "image/gif": /\.gif$/i, "image/png": /\.png$/i }, /** * Method: write * * Parameters: * sld - {Object} An object representing the SLD. * * Returns: * {DOMElement} The root of an SLD document. */ 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. */ 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 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 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 */ OpenLayers.Format.SLD.v1_0_0 = OpenLayers.Class( OpenLayers.Format.SLD.v1, { /** * Constant: VERSION * {String} 1.0.0 */ VERSION: "1.0.0", /** * 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", /** * Constructor: OpenLayers.Format.SLD.v1_0_0 * Instances of this class are not created directly. Use the * constructor instead. * * Parameters: * options - {Object} An optional object whose properties will be set on * this instance. */ CLASS_NAME: "OpenLayers.Format.SLD.v1_0_0" }); /** FILE: 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 - {} */ 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: * {} 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 * * 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; /** FILE: 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); } }); /** FILE: 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; })(); /** FILE: 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 * constructor. * * Inherits from: * - */ 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 , , * , or 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 - {} * 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 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 * class. * * Parameters: * 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 - {} Filter to get serialized as a * query string. * readWithPOST - {Boolean} If the request should be done with POST. * * Returns: * {} 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 - {} 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({})} or * {} * options - {Object} Optional object for configuring the request. * This object is modified and should not be reused. * * Returns: * {} An * 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 is complete. May be overridden * by subclasses. * * Parameters: * resp - {} 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 - {} * options - {Object} Optional object for configuring the request. * This object is modified and should not be reused. * * Returns: * {} An * 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 is complete. May be overridden * by subclasses. * * Parameters: * resp - {} 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 - {} * options - {Object} Optional object for configuring the request. * This object is modified and should not be reused. * * Returns: * {} An * 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 is complete. May be overridden * by subclasses. * * Parameters: * resp - {} 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 - {} 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({})} or * {} 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({})} * options - {Object} Optional object for setting up intermediate commit * callbacks. * * Valid options: * create - {Object} Optional object to be passed to the method. * update - {Object} Optional object to be passed to the method. * delete - {Object} Optional object to be passed to the 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()} 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 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= 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 - {} */ 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 - {} * 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" }); /** FILE: 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: * {} 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 , , * and for WFS 1.1.0. Note that * srsName matching with the WMS layer will not work with WFS 1.0.0. * * Parameters: * layer - {} 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.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" }; /** FILE: 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.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 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 * {} 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 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 - {} 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({})} or * {} * 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()} * options - {Object} * * Valid options properties: * nativeElements - {Array({Object})} Array of objects with information for writing * out elements, these objects have vendorId, safeToIgnore and * value properties. The element is intended to allow access to * vendor specific capabilities of any particular web feature server or * datastore. * * Returns: * {} 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 - {} 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 - {} 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 - {} */ abort: function(response) { if (response) { response.priv.abort(); } }, CLASS_NAME: "OpenLayers.Protocol.WFS.v1" }); /** FILE: 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: * {} 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" }; /** FILE: 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.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", xmlns: "http://www.w3.org/2000/xmlns/" }, /** * 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 * or * 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()} 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} */ setFilterProperty: function(filter) { if(filter.filters) { for(var i=0, len=filter.filters.length; i * - */ OpenLayers.Format.Filter.v1_1_0 = OpenLayers.Class( OpenLayers.Format.GML.v3, OpenLayers.Format.Filter.v1, { /** * Constant: VERSION * {String} 1.1.0 */ VERSION: "1.1.0", /** * Property: schemaLocation * {String} http://www.opengis.net/ogc/filter/1.1.0/filter.xsd */ schemaLocation: "http://www.opengis.net/ogc/filter/1.1.0/filter.xsd", /** * Constructor: OpenLayers.Format.Filter.v1_1_0 * Instances of this class are not created directly. Use the * constructor instead. * * Parameters: * options - {Object} An optional object whose properties will be set on * this instance. */ initialize: function(options) { OpenLayers.Format.GML.v3.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 matchCase = node.getAttribute("matchCase"); var filter = new OpenLayers.Filter.Comparison({ type: OpenLayers.Filter.Comparison.EQUAL_TO, matchCase: !(matchCase === "false" || matchCase === "0") }); this.readChildNodes(node, filter); obj.filters.push(filter); }, "PropertyIsNotEqualTo": function(node, obj) { var matchCase = node.getAttribute("matchCase"); var filter = new OpenLayers.Filter.Comparison({ type: OpenLayers.Filter.Comparison.NOT_EQUAL_TO, matchCase: !(matchCase === "false" || matchCase === "0") }); 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("escapeChar"); filter.value2regex(wildCard, singleChar, esc); obj.filters.push(filter); } }, OpenLayers.Format.Filter.v1.prototype.readers["ogc"]), "gml": OpenLayers.Format.GML.v3.prototype.readers["gml"], "feature": OpenLayers.Format.GML.v3.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", { attributes: {matchCase: filter.matchCase} }); // 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", { attributes: {matchCase: filter.matchCase} }); // 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: { matchCase: filter.matchCase, wildCard: "*", singleChar: ".", escapeChar: "!" } }); // 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 optional in 1.1.0 filter.property && this.writeNode("PropertyName", filter, node); var box = this.writeNode("gml:Envelope", filter.value); if(filter.projection) { box.setAttribute("srsName", filter.projection); } node.appendChild(box); return node; }, "SortBy": function(sortProperties) { var node = this.createElementNSPlus("ogc:SortBy"); for (var i=0,l=sortProperties.length;i} filter and converts it into XML. * * Parameters: * filter - {} 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:Envelope", filter.value); } if(filter.projection) { child.setAttribute("srsName", filter.projection); } node.appendChild(child); } return node; }, CLASS_NAME: "OpenLayers.Format.Filter.v1_1_0" }); /** FILE: OpenLayers/Format/OWSCommon.js **/ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for * full list of contributors). Published 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.OWSCommon * Read OWSCommon. Create a new instance with the * constructor. * * Inherits from: * - */ OpenLayers.Format.OWSCommon = 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.OWSCommon * Create a new parser for OWSCommon. * * 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 = this.version; if(!version) { // remember version does not correspond to the OWS version // it corresponds to the WMS/WFS/WCS etc. request version var uri = root.getAttribute("xmlns:ows"); // the above will fail if the namespace prefix is different than // ows and if the namespace is declared on a different element if (uri && uri.substring(uri.lastIndexOf("/")+1) === "1.1") { version ="1.1.0"; } if(!version) { version = this.defaultVersion; } } return version; }, /** * APIMethod: read * Read an OWSCommon document and return an object. * * Parameters: * data - {String | DOMElement} Data to read. * options - {Object} Options for the reader. * * Returns: * {Object} An object representing the structure of the document. */ CLASS_NAME: "OpenLayers.Format.OWSCommon" }); /** FILE: OpenLayers/Format/OWSCommon/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/OWSCommon.js */ /** * Class: OpenLayers.Format.OWSCommon.v1 * Common readers and writers for OWSCommon v1.X formats * * Inherits from: * - */ OpenLayers.Format.OWSCommon.v1 = OpenLayers.Class(OpenLayers.Format.XML, { /** * Property: regExes * Compiled regular expressions for manipulating strings. */ regExes: { trimSpace: (/^\s*|\s*$/g), removeSpace: (/\s*/g), splitSpace: (/\s+/), trimComma: (/\s*,\s*/g) }, /** * Method: read * * Parameters: * data - {DOMElement} An OWSCommon document element. * options - {Object} Options for the reader. * * Returns: * {Object} An object representing the OWSCommon document. */ read: function(data, options) { options = OpenLayers.Util.applyDefaults(options, this.options); var ows = {}; this.readChildNodes(data, ows); return ows; }, /** * 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: { "ows": { "Exception": function(node, exceptionReport) { var exception = { code: node.getAttribute('exceptionCode'), locator: node.getAttribute('locator'), texts: [] }; exceptionReport.exceptions.push(exception); this.readChildNodes(node, exception); }, "ExceptionText": function(node, exception) { var text = this.getChildValue(node); exception.texts.push(text); }, "ServiceIdentification": function(node, obj) { obj.serviceIdentification = {}; this.readChildNodes(node, obj.serviceIdentification); }, "Title": function(node, obj) { obj.title = this.getChildValue(node); }, "Abstract": function(node, serviceIdentification) { serviceIdentification["abstract"] = this.getChildValue(node); }, "Keywords": function(node, serviceIdentification) { serviceIdentification.keywords = {}; this.readChildNodes(node, serviceIdentification.keywords); }, "Keyword": function(node, keywords) { keywords[this.getChildValue(node)] = true; }, "ServiceType": function(node, serviceIdentification) { serviceIdentification.serviceType = { codeSpace: node.getAttribute('codeSpace'), value: this.getChildValue(node)}; }, "ServiceTypeVersion": function(node, serviceIdentification) { serviceIdentification.serviceTypeVersion = this.getChildValue(node); }, "Fees": function(node, serviceIdentification) { serviceIdentification.fees = this.getChildValue(node); }, "AccessConstraints": function(node, serviceIdentification) { serviceIdentification.accessConstraints = this.getChildValue(node); }, "ServiceProvider": function(node, obj) { obj.serviceProvider = {}; this.readChildNodes(node, obj.serviceProvider); }, "ProviderName": function(node, serviceProvider) { serviceProvider.providerName = this.getChildValue(node); }, "ProviderSite": function(node, serviceProvider) { serviceProvider.providerSite = this.getAttributeNS(node, this.namespaces.xlink, "href"); }, "ServiceContact": function(node, serviceProvider) { serviceProvider.serviceContact = {}; this.readChildNodes(node, serviceProvider.serviceContact); }, "IndividualName": function(node, serviceContact) { serviceContact.individualName = this.getChildValue(node); }, "PositionName": function(node, serviceContact) { serviceContact.positionName = this.getChildValue(node); }, "ContactInfo": function(node, serviceContact) { serviceContact.contactInfo = {}; this.readChildNodes(node, serviceContact.contactInfo); }, "Phone": function(node, contactInfo) { contactInfo.phone = {}; this.readChildNodes(node, contactInfo.phone); }, "Voice": function(node, phone) { phone.voice = 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); }, "AdministrativeArea": function(node, address) { address.administrativeArea = 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); }, "Role": function(node, serviceContact) { serviceContact.role = this.getChildValue(node); }, "OperationsMetadata": function(node, obj) { obj.operationsMetadata = {}; this.readChildNodes(node, obj.operationsMetadata); }, "Operation": function(node, operationsMetadata) { var name = node.getAttribute("name"); operationsMetadata[name] = {}; this.readChildNodes(node, operationsMetadata[name]); }, "DCP": function(node, operation) { operation.dcp = {}; this.readChildNodes(node, operation.dcp); }, "HTTP": function(node, dcp) { dcp.http = {}; this.readChildNodes(node, dcp.http); }, "Get": function(node, http) { if (!http.get) { http.get = []; } var obj = { url: this.getAttributeNS(node, this.namespaces.xlink, "href") }; this.readChildNodes(node, obj); http.get.push(obj); }, "Post": function(node, http) { if (!http.post) { http.post = []; } var obj = { url: this.getAttributeNS(node, this.namespaces.xlink, "href") }; this.readChildNodes(node, obj); http.post.push(obj); }, "Parameter": function(node, operation) { if (!operation.parameters) { operation.parameters = {}; } var name = node.getAttribute("name"); operation.parameters[name] = {}; this.readChildNodes(node, operation.parameters[name]); }, "Constraint": function(node, obj) { if (!obj.constraints) { obj.constraints = {}; } var name = node.getAttribute("name"); obj.constraints[name] = {}; this.readChildNodes(node, obj.constraints[name]); }, "Value": function(node, allowedValues) { allowedValues[this.getChildValue(node)] = true; }, "OutputFormat": function(node, obj) { obj.formats.push({value: this.getChildValue(node)}); this.readChildNodes(node, obj); }, "WGS84BoundingBox": function(node, obj) { var boundingBox = {}; boundingBox.crs = node.getAttribute("crs"); if (obj.BoundingBox) { obj.BoundingBox.push(boundingBox); } else { obj.projection = boundingBox.crs; boundingBox = obj; } this.readChildNodes(node, boundingBox); }, "BoundingBox": function(node, obj) { // FIXME: We consider that BoundingBox is the same as WGS84BoundingBox // LowerCorner = "min_x min_y" // UpperCorner = "max_x max_y" // It should normally depend on the projection this.readers['ows']['WGS84BoundingBox'].apply(this, [node, obj]); }, "LowerCorner": 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); obj.left = pointList[0]; obj.bottom = pointList[1]; }, "UpperCorner": 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); obj.right = pointList[0]; obj.top = pointList[1]; obj.bounds = new OpenLayers.Bounds(obj.left, obj.bottom, obj.right, obj.top); delete obj.left; delete obj.bottom; delete obj.right; delete obj.top; }, "Language": function(node, obj) { obj.language = this.getChildValue(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: { "ows": { "BoundingBox": function(options, nodeName) { var node = this.createElementNSPlus(nodeName || "ows:BoundingBox", { attributes: { crs: options.projection } }); this.writeNode("ows:LowerCorner", options, node); this.writeNode("ows:UpperCorner", options, node); return node; }, "LowerCorner": function(options) { var node = this.createElementNSPlus("ows:LowerCorner", { value: options.bounds.left + " " + options.bounds.bottom }); return node; }, "UpperCorner": function(options) { var node = this.createElementNSPlus("ows:UpperCorner", { value: options.bounds.right + " " + options.bounds.top }); return node; }, "Identifier": function(identifier) { var node = this.createElementNSPlus("ows:Identifier", { value: identifier }); return node; }, "Title": function(title) { var node = this.createElementNSPlus("ows:Title", { value: title }); return node; }, "Abstract": function(abstractValue) { var node = this.createElementNSPlus("ows:Abstract", { value: abstractValue }); return node; }, "OutputFormat": function(format) { var node = this.createElementNSPlus("ows:OutputFormat", { value: format }); return node; } } }, CLASS_NAME: "OpenLayers.Format.OWSCommon.v1" }); /** FILE: OpenLayers/Format/OWSCommon/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/OWSCommon/v1.js */ /** * Class: OpenLayers.Format.OWSCommon.v1_0_0 * Parser for OWS Common version 1.0.0. * * Inherits from: * - */ OpenLayers.Format.OWSCommon.v1_0_0 = OpenLayers.Class(OpenLayers.Format.OWSCommon.v1, { /** * Property: namespaces * {Object} Mapping of namespace aliases to namespace URIs. */ namespaces: { ows: "http://www.opengis.net/ows", xlink: "http://www.w3.org/1999/xlink" }, /** * 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: { "ows": OpenLayers.Util.applyDefaults({ "ExceptionReport": function(node, obj) { obj.success = false; obj.exceptionReport = { version: node.getAttribute('version'), language: node.getAttribute('language'), exceptions: [] }; this.readChildNodes(node, obj.exceptionReport); } }, OpenLayers.Format.OWSCommon.v1.prototype.readers.ows) }, /** * 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: { "ows": OpenLayers.Format.OWSCommon.v1.prototype.writers.ows }, CLASS_NAME: "OpenLayers.Format.OWSCommon.v1_0_0" }); /** FILE: OpenLayers/Format/WFST/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_1_0.js * @requires OpenLayers/Format/OWSCommon/v1_0_0.js */ /** * Class: OpenLayers.Format.WFST.v1_1_0 * A format for creating WFS v1.1.0 transactions. Create a new instance with the * constructor. * * Inherits from: * - * - */ OpenLayers.Format.WFST.v1_1_0 = OpenLayers.Class( OpenLayers.Format.Filter.v1_1_0, OpenLayers.Format.WFST.v1, { /** * Property: version * {String} WFS version number. */ version: "1.1.0", /** * Property: schemaLocations * {Object} Properties are namespace aliases, values are schema locations. */ schemaLocations: { "wfs": "http://schemas.opengis.net/wfs/1.1.0/wfs.xsd" }, /** * Constructor: OpenLayers.Format.WFST.v1_1_0 * A class for parsing and generating WFS v1.1.0 transactions. * * To read additional information like hit count (numberOfFeatures) from * the FeatureCollection, call the method * with {output: "object"} as 2nd argument. Note that it is possible to * just request the hit count from a WFS 1.1.0 server with the * resultType="hits" request parameter. * * 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_1_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.v3. 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.v3.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({ "FeatureCollection": function(node, obj) { obj.numberOfFeatures = parseInt(node.getAttribute( "numberOfFeatures")); OpenLayers.Format.WFST.v1.prototype.readers["wfs"]["FeatureCollection"].apply( this, arguments); }, "TransactionResponse": function(node, obj) { obj.insertIds = []; obj.success = false; this.readChildNodes(node, obj); }, "TransactionSummary": function(node, obj) { // this is a limited test of success obj.success = true; }, "InsertResults": function(node, obj) { this.readChildNodes(node, obj); }, "Feature": function(node, container) { var obj = {fids: []}; this.readChildNodes(node, obj); container.insertIds.push(obj.fids[0]); } }, OpenLayers.Format.WFST.v1.prototype.readers["wfs"]), "gml": OpenLayers.Format.GML.v3.prototype.readers["gml"], "feature": OpenLayers.Format.GML.v3.prototype.readers["feature"], "ogc": OpenLayers.Format.Filter.v1_1_0.prototype.readers["ogc"], "ows": OpenLayers.Format.OWSCommon.v1_0_0.prototype.readers["ows"] }, /** * 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({ "GetFeature": function(options) { var node = OpenLayers.Format.WFST.v1.prototype.writers["wfs"]["GetFeature"].apply(this, arguments); options && this.setAttributes(node, { resultType: options.resultType, startIndex: options.startIndex, count: options.count }); return node; }, "Query": function(options) { options = OpenLayers.Util.extend({ featureNS: this.featureNS, featurePrefix: this.featurePrefix, featureType: this.featureType, srsName: this.srsName }, options); var prefix = options.featurePrefix; var node = this.createElementNSPlus("wfs:Query", { attributes: { typeName: (prefix ? prefix + ":" : "") + options.featureType, srsName: options.srsName } }); if(options.featureNS) { this.setAttributeNS(node, this.namespaces.xmlns, "xmlns:" + prefix, options.featureNS); } if(options.propertyNames) { for(var i=0,len = options.propertyNames.length; i 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_1_0 = OpenLayers.Class(OpenLayers.Protocol.WFS.v1, { /** * Property: version * {String} WFS version number. */ version: "1.1.0", /** * 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 - {} 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.Protocol.WFS.v1_1_0" }); /** FILE: GeoExt/data/LayerRecord.js **/ /** * Copyright (c) 2008-2012 The Open Source Geospatial Foundation * * Published under the BSD license. * See http://svn.geoext.org/core/trunk/geoext/license.txt for the full text * of the license. */ /** api: (define) * module = GeoExt.data * class = LayerRecord * base_link = `Ext.data.Record `_ */ Ext.namespace("GeoExt.data"); /** api: constructor * .. class:: LayerRecord * * A record that represents an ``OpenLayers.Layer``. This record * will always have at least the following fields: * * * title ``String`` */ GeoExt.data.LayerRecord = Ext.data.Record.create([ {name: "layer"}, {name: "title", type: "string", mapping: "name"} ]); /** api: method[getLayer] * :return: ``OpenLayers.Layer`` * * Gets the layer for this record. */ GeoExt.data.LayerRecord.prototype.getLayer = function() { return this.get("layer"); }; /** api: method[setLayer] * :param layer: ``OpenLayers.Layer`` * * Sets the layer for this record. */ GeoExt.data.LayerRecord.prototype.setLayer = function(layer) { if (layer !== this.data.layer) { this.dirty = true; if(!this.modified) { this.modified = {}; } if(this.modified.layer === undefined) { this.modified.layer = this.data.layer; } this.data.layer = layer; if(!this.editing) { this.afterEdit(); } } }; /** api: method[clone] * :param id: ``String`` (optional) A new Record id. * :return: class:`GeoExt.data.LayerRecord` A new layer record. * * Creates a clone of this LayerRecord. */ GeoExt.data.LayerRecord.prototype.clone = function(id) { var layer = this.getLayer() && this.getLayer().clone(); return new this.constructor( Ext.applyIf({layer: layer}, this.data), id || layer.id ); }; /** api: classmethod[create] * :param o: ``Array`` Field definition as in ``Ext.data.Record.create``. Can * be omitted if no additional fields are required. * :return: ``Function`` A specialized :class:`GeoExt.data.LayerRecord` * constructor. * * Creates a constructor for a :class:`GeoExt.data.LayerRecord`, optionally * with additional fields. */ GeoExt.data.LayerRecord.create = function(o) { var f = Ext.extend(GeoExt.data.LayerRecord, {}); var p = f.prototype; p.fields = new Ext.util.MixedCollection(false, function(field) { return field.name; }); GeoExt.data.LayerRecord.prototype.fields.each(function(f) { p.fields.add(f); }); if(o) { for(var i = 0, len = o.length; i < len; i++){ p.fields.add(new Ext.data.Field(o[i])); } } f.getField = function(name) { return p.fields.get(name); }; return f; }; /** FILE: GeoExt/data/AttributeStore.js **/ /** * Copyright (c) 2008-2012 The Open Source Geospatial Foundation * * Published under the BSD license. * See http://svn.geoext.org/core/trunk/geoext/license.txt for the full text * of the license. */ /** * @include GeoExt/data/AttributeReader.js */ /** api: (define) * module = GeoExt.data * class = AttributeStore * base_link = `Ext.data.Store `_ */ Ext.namespace("GeoExt.data"); /** * Function: GeoExt.data.AttributeStoreMixin * * This function generates a mixin object to be used when extending an Ext.data.Store * to create an attribute store. * * (start code) * var AttrStore = Ext.extend(Ext.data.Store, GeoExt.data.AttributeStoreMixin); * var store = new AttrStore(); * (end) * * For convenience, a GeoExt.data.AttributeStore class is available as a * shortcut to the Ext.extend sequence in the above code snippet. The above * is equivalent to: * (start code) * var store = new GeoExt.data.AttributeStore(); * (end) */ GeoExt.data.AttributeStoreMixin = function() { return { /** private */ constructor: function(c) { c = c || {}; arguments.callee.superclass.constructor.call( this, Ext.apply(c, { proxy: c.proxy || (!c.data ? new Ext.data.HttpProxy({url: c.url, disableCaching: false, method: "GET"}) : undefined ), reader: new GeoExt.data.AttributeReader( c, c.fields || ["name", "type", "restriction", { name: "nillable", type: "boolean" }, "annotation"] ) }) ); if(this.feature) { this.bind(); } }, /** private: method[bind] */ bind: function() { this.on({ "update": this.onUpdate, "load": this.onLoad, "add": this.onAdd, scope: this }); var records = []; this.each(function(record) { records.push(record); }); this.updateFeature(records); }, /** private: method[onUpdate] * :param store: ``Ext.data.Store`` * :param record: ``Ext.data.Record`` * :param operation: ``String`` * * Handler for store update event. */ onUpdate: function(store, record, operation) { this.updateFeature([record]); }, /** private: method[onLoad] * :param store: ``Ext.data.Store`` * :param records: ``Array(Ext.data.Record)`` * :param options: ``Object`` * * Handler for store load event */ onLoad: function(store, records, options) { // if options.add is true an "add" event was already // triggered, and onAdd already did the work of // adding the features to the layer. if(!options || options.add !== true) { this.updateFeature(records); } }, /** private: method[onAdd] * :param store: ``Ext.data.Store`` * :param records: ``Array(Ext.data.Record)`` * :param index: ``Number`` * * Handler for store add event */ onAdd: function(store, records, index) { this.updateFeature(records); }, /** private: method[updateFeature] * :param records: ``Array(Ext.data.Record)`` * * Update feature from records. */ updateFeature: function(records) { var feature = this.feature, layer = feature.layer; var i, len, record, name, value, oldValue, dirty; for(i=0,len=records.length; i`_ */ Ext.namespace("GeoExt.data"); /** api: constructor * .. class:: AttributeReader(meta, recordType) * * :arg meta: ``Object`` Reader configuration. * :arg recordType: ``Array or Ext.data.Record`` An array of field * configuration objects or a record object. * * Create a new attributes reader object. * * Valid meta properties: * * * format - ``OpenLayers.Format`` A parser for transforming the XHR response * into an array of objects representing attributes. Defaults to * an ``OpenLayers.Format.WFSDescribeFeatureType`` parser. * * ignore - ``Object`` Properties of the ignore object should be field names. * Values are either arrays or regular expressions. * * feature - ``OpenLayers.Feature.Vector`` A vector feature. If provided * records created by the reader will include a field named "value" * referencing the attribute value as set in the feature. */ GeoExt.data.AttributeReader = function(meta, recordType) { meta = meta || {}; if(!meta.format) { meta.format = new OpenLayers.Format.WFSDescribeFeatureType(); } GeoExt.data.AttributeReader.superclass.constructor.call( this, meta, recordType || meta.fields ); if(meta.feature) { this.recordType.prototype.fields.add(new Ext.data.Field("value")); } }; Ext.extend(GeoExt.data.AttributeReader, Ext.data.DataReader, { /** private: method[read] * :arg request: ``Object`` The XHR object that contains the parsed doc. * :return: ``Object`` A data block which is used by an ``Ext.data.Store`` * as a cache of ``Ext.data.Records``. * * This method is only used by a DataProxy which has retrieved data from a * remote server. */ read: function(request) { var data = request.responseXML; if(!data || !data.documentElement) { data = request.responseText; } return this.readRecords(data); }, /** private: method[readRecords] * :arg data: ``DOMElement or String or Array`` A document element or XHR * response string. As an alternative to fetching attributes data from * a remote source, an array of attribute objects can be provided given * that the properties of each attribute object map to a provided field * name. * :return: ``Object`` A data block which is used by an ``Ext.data.Store`` * as a cache of ``Ext.data.Records``. * * Create a data block containing Ext.data.Records from an XML document. */ readRecords: function(data) { var attributes; if(data instanceof Array) { attributes = data; } else { // only works with one featureType in the doc var output = this.meta.format.read(data); if (!!output.error) { throw new Ext.data.DataReader.Error("invalid-response", output.error); } attributes = output.featureTypes[0].properties; } var feature = this.meta.feature; var recordType = this.recordType; var fields = recordType.prototype.fields; var numFields = fields.length; var attr, values, name, record, ignore, value, field, records = []; for(var i=0, len=attributes.length; i -1); } else if(matches instanceof RegExp) { ignore = (matches.test(value)); } } return ignore; } }); /** FILE: GeoExt/data/ProtocolProxy.js **/ /** * Copyright (c) 2008-2012 The Open Source Geospatial Foundation * * Published under the BSD license. * See http://svn.geoext.org/core/trunk/geoext/license.txt for the full text * of the license. */ /** * @require OpenLayers/BaseTypes.js */ /** api: (define) * module = GeoExt.data * class = ProtocolProxy * base_link = `Ext.data.DataProxy `_ */ Ext.namespace('GeoExt', 'GeoExt.data'); GeoExt.data.ProtocolProxy = function(config) { Ext.apply(this, config); GeoExt.data.ProtocolProxy.superclass.constructor.apply(this, arguments); }; /** api: constructor * .. class:: ProtocolProxy * * A data proxy for use with ``OpenLayers.Protocol`` objects. */ Ext.extend(GeoExt.data.ProtocolProxy, Ext.data.DataProxy, { /** api: config[protocol] * ``OpenLayers.Protocol`` * The protocol used to fetch features. */ protocol: null, /** api: config[abortPrevious] * ``Boolean`` * Abort any previous request before issuing another. Default is ``true``. */ abortPrevious: true, /** api: config[setParamsAsOptions] * ``Boolean`` * Should options.params be set directly on options before passing it into * the protocol's read method? Default is ``false``. */ setParamsAsOptions: false, /** private: property[response] * ``OpenLayers.Protocol.Response`` * The response returned by the read call on the protocol. */ response: null, /** private: method[load] * :param params: ``Object`` An object containing properties which are to * be used as HTTP parameters for the request to the remote server. * :param reader: ``Ext.data.DataReader`` The Reader object which converts * the data object into a block of ``Ext.data.Records``. * :param callback: ``Function`` The function into which to pass the block * of ``Ext.data.Records``. The function is passed the Record block * object, the ``args`` argument passed to the load function, and a * boolean success indicator. * :param scope: ``Object`` The scope in which to call the callback. * :param arg: ``Object`` An optional argument which is passed to the * callback as its second parameter. * * Calls ``read`` on the protocol. */ load: function(params, reader, callback, scope, arg) { if (this.fireEvent("beforeload", this, params) !== false) { var o = { params: params || {}, request: { callback: callback, scope: scope, arg: arg }, reader: reader }; var cb = OpenLayers.Function.bind(this.loadResponse, this, o); if (this.abortPrevious) { this.abortRequest(); } var options = { params: params, callback: cb, scope: this }; Ext.applyIf(options, arg); if (this.setParamsAsOptions === true) { Ext.applyIf(options, options.params); delete options.params; } this.response = this.protocol.read(options); } else { callback.call(scope || this, null, arg, false); } }, /** private: method[abortRequest] * Called to abort any ongoing request. */ abortRequest: function() { if (this.response) { this.protocol.abort(this.response); this.response = null; } }, /** private: method[loadResponse] * :param o: ``Object`` * :param response: ``OpenLayers.Protocol.Response`` * * Handle response from the protocol */ loadResponse: function(o, response) { if (response.success()) { var result; try { result = o.reader.read(response); } catch(e) { // @deprecated: fire old loadexception for backwards-compat. // TODO remove this.fireEvent('loadexception', this, o, response, e); this.fireEvent('exception', this, 'response', null, o, response, e); o.request.callback.call(o.request.scope, null, o.request.arg, false); return; } this.fireEvent("load", this, o, o.request.arg); o.request.callback.call( o.request.scope, result, o.request.arg, true); } else { this.fireEvent("loadexception", this, o, response); o.request.callback.call( o.request.scope, null, o.request.arg, false); } } }); /** FILE: GeoExt/data/FeatureStore.js **/ /** * Copyright (c) 2008-2012 The Open Source Geospatial Foundation * * Published under the BSD license. * See http://svn.geoext.org/core/trunk/geoext/license.txt for the full text * of the license. */ /** * @include GeoExt/data/FeatureReader.js * @require OpenLayers/Feature/Vector.js */ /** api: (define) * module = GeoExt.data * class = FeatureStore * base_link = `Ext.data.Store `_ */ Ext.namespace("GeoExt.data"); /** api: constructor * .. class:: FeatureStore * * A store containing :class:`GeoExt.data.FeatureRecord` entries that * optionally synchronizes with an ``OpenLayers.Layer.Vector``. */ /** api: example * Sample code to create a store with features from a vector layer: * * .. code-block:: javascript * * var store = new GeoExt.data.FeatureStore({ * layer: myLayer, * features: myFeatures * }); */ /** * Class: GeoExt.data.FeatureStoreMixin * A store that synchronizes a features array of an {OpenLayers.Layer.Vector} with a * feature store holding {} entries. * * This class can not be instantiated directly. Instead, it is meant to extend * {Ext.data.Store} or a subclass of it: * (start code) * var store = new (Ext.extend(Ext.data.Store, new GeoExt.data.FeatureStoreMixin))({ * layer: myLayer, * features: myFeatures * }); * (end) * * For convenience, a {} class is available as a * shortcut to the Ext.extend sequence in the above code snippet. The above * is equivalent to: * (start code) * var store = new GeoExt.data.FeatureStore({ * layer: myLayer, * features: myFeatures * }); * (end) */ GeoExt.data.FeatureStoreMixin = function() { return { /** api: config[layer] * ``OpenLayers.Layer.Vector`` Layer to synchronize the store with. */ layer: null, /** api: config[features] * ``Array(OpenLayers.Feature.Vector)`` Features that will be added to the * store (and the layer if provided). */ /** api: config[reader] * ``Ext.data.DataReader`` The reader used to produce records from objects * features. Default is :class:`GeoExt.data.FeatureReader`. */ reader: null, /** api: config[featureFilter] * ``OpenLayers.Filter`` This filter is evaluated before a feature * record is added to the store. */ featureFilter: null, /** api: config[initDir] * ``Number`` Bitfields specifying the direction to use for the * initial sync between the layer and the store, if set to 0 then no * initial sync is done. Default is * ``GeoExt.data.FeatureStore.LAYER_TO_STORE|GeoExt.data.FeatureStore.STORE_TO_LAYER``. */ /** private */ constructor: function(config) { config = config || {}; config.reader = config.reader || new GeoExt.data.FeatureReader({}, config.fields); var layer = config.layer; delete config.layer; // 'features' option - is an alias 'data' option if (config.features) { config.data = config.features; } delete config.features; // "initDir" option var options = {initDir: config.initDir}; delete config.initDir; arguments.callee.superclass.constructor.call(this, config); if(layer) { this.bind(layer, options); } }, /** api: method[bind] * :param layer: ``OpenLayers.Layer`` Layer that the store should be * synchronized with. * * Bind this store to a layer instance, once bound the store * is synchronized with the layer and vice-versa. */ bind: function(layer, options) { if(this.layer) { // already bound return; } this.layer = layer; options = options || {}; var initDir = options.initDir; if(options.initDir == undefined) { initDir = GeoExt.data.FeatureStore.LAYER_TO_STORE | GeoExt.data.FeatureStore.STORE_TO_LAYER; } // create a snapshot of the layer's features var features = layer.features.slice(0); if(initDir & GeoExt.data.FeatureStore.STORE_TO_LAYER) { var records = this.getRange(); for(var i=records.length - 1; i>=0; i--) { this.layer.addFeatures([records[i].getFeature()]); } } if(initDir & GeoExt.data.FeatureStore.LAYER_TO_STORE) { this.loadData(features, true /* append */); } layer.events.on({ "featuresadded": this.onFeaturesAdded, "featuresremoved": this.onFeaturesRemoved, "featuremodified": this.onFeatureModified, scope: this }); this.on({ "load": this.onLoad, "clear": this.onClear, "add": this.onAdd, "remove": this.onRemove, "update": this.onUpdate, scope: this }); }, /** api: method[unbind] * Unbind this store from the layer it is currently bound. */ unbind: function() { if(this.layer) { this.layer.events.un({ "featuresadded": this.onFeaturesAdded, "featuresremoved": this.onFeaturesRemoved, "featuremodified": this.onFeatureModified, scope: this }); this.un("load", this.onLoad, this); this.un("clear", this.onClear, this); this.un("add", this.onAdd, this); this.un("remove", this.onRemove, this); this.un("update", this.onUpdate, this); this.layer = null; } }, /** api: method[getRecordFromFeature] * :arg feature: ``OpenLayers.Vector.Feature`` * :returns: :class:`GeoExt.data.FeatureRecord` The record corresponding * to the given feature. Returns null if no record matches. * * *Deprecated* Use getByFeature instead. * * Get the record corresponding to a feature. */ getRecordFromFeature: function(feature) { return this.getByFeature(feature) || null; }, /** api: method[getByFeature] * :arg feature: ``OpenLayers.Vector.Feature`` * :returns: :class:`GeoExt.data.FeatureRecord` The record corresponding * to the given feature. Returns undefined if no record matches. * * Get the record corresponding to a feature. */ getByFeature: function(feature) { var record; if(feature.state !== OpenLayers.State.INSERT) { record = this.getById(feature.id); } else { var index = this.findBy(function(r) { return r.getFeature() === feature; }); if(index > -1) { record = this.getAt(index); } } return record; }, /** private: method[onFeaturesAdded] * Handler for layer featuresadded event */ onFeaturesAdded: function(evt) { if(!this._adding) { var features = evt.features, toAdd = features; if(this.featureFilter) { toAdd = []; var i, len, feature; for(var i=0, len=features.length; i=0; i--) { feature = features[i]; record = this.getByFeature(feature); if(record !== undefined) { this._removing = true; this.remove(record); delete this._removing; } } } }, /** private: method[onFeatureModified] * Handler for layer featuremodified event */ onFeatureModified: function(evt) { if(!this._updating) { var feature = evt.feature; var record = this.getByFeature(feature); if(record !== undefined) { record.beginEdit(); var attributes = feature.attributes; if(attributes) { var fields = this.recordType.prototype.fields; for(var i=0, len=fields.length; i 0) { this._adding = true; this.layer.addFeatures(features); delete this._adding; } }, /** private: method[onLoad] * :param store: ``Ext.data.Store`` * :param records: ``Array(Ext.data.Record)`` * :param options: ``Object`` * * Handler for store load event */ onLoad: function(store, records, options) { // if options.add is true an "add" event was already // triggered, and onAdd already did the work of // adding the features to the layer. if(!options || options.add !== true) { this._removing = true; this.layer.removeFeatures(this.layer.features); delete this._removing; this.addFeaturesToLayer(records); } }, /** private: method[onClear] * :param store: ``Ext.data.Store`` * * Handler for store clear event */ onClear: function(store) { this._removing = true; this.layer.removeFeatures(this.layer.features); delete this._removing; }, /** private: method[onAdd] * :param store: ``Ext.data.Store`` * :param records: ``Array(Ext.data.Record)`` * :param index: ``Number`` * * Handler for store add event */ onAdd: function(store, records, index) { if(!this._adding) { // addFeaturesToLayer takes care of setting // this._adding to true and deleting it this.addFeaturesToLayer(records); } }, /** private: method[onRemove] * :param store: ``Ext.data.Store`` * :param records: ``Array(Ext.data.Record)`` * :param index: ``Number`` * * Handler for store remove event */ onRemove: function(store, record, index){ if(!this._removing) { var feature = record.getFeature(); if (this.layer.getFeatureById(feature.id) != null) { this._removing = true; this.layer.removeFeatures([record.getFeature()]); delete this._removing; } } }, /** private: method[onUpdate] * :param store: ``Ext.data.Store`` * :param record: ``Ext.data.Record`` * :param operation: ``String`` * * Handler for update. */ onUpdate: function(store, record, operation) { if(!this._updating) { /** * TODO: remove this if the FeatureReader adds attributes * for all fields that map to feature.attributes. * In that case, it would be sufficient to check (key in feature.attributes). */ var defaultFields = new GeoExt.data.FeatureRecord().fields; var feature = record.getFeature(); if (feature.state !== OpenLayers.State.INSERT) { feature.state = OpenLayers.State.UPDATE; } if(record.fields) { var cont = this.layer.events.triggerEvent( "beforefeaturemodified", {feature: feature} ); if(cont !== false) { var attributes = feature.attributes; record.fields.each( function(field) { var key = field.mapping || field.name; if (!defaultFields.containsKey(key)) { attributes[key] = record.get(field.name); } } ); this._updating = true; this.layer.events.triggerEvent( "featuremodified", {feature: feature} ); delete this._updating; if (this.layer.getFeatureById(feature.id) != null) { this.layer.drawFeature(feature); } } } } }, /** private: method[destroy] */ destroy: function() { this.unbind(); GeoExt.data.FeatureStore.superclass.destroy.call(this); } }; }; GeoExt.data.FeatureStore = Ext.extend( Ext.data.Store, new GeoExt.data.FeatureStoreMixin ); /** * Constant: GeoExt.data.FeatureStore.LAYER_TO_STORE * {Integer} Constant used to make the store be automatically updated * when changes occur in the layer. */ GeoExt.data.FeatureStore.LAYER_TO_STORE = 1; /** * Constant: GeoExt.data.FeatureStore.STORE_TO_LAYER * {Integer} Constant used to make the layer be automatically updated * when changes occur in the store. */ GeoExt.data.FeatureStore.STORE_TO_LAYER = 2; /** FILE: GeoExt/data/FeatureReader.js **/ /** * Copyright (c) 2008-2012 The Open Source Geospatial Foundation * * Published under the BSD license. * See http://svn.geoext.org/core/trunk/geoext/license.txt for the full text * of the license. */ /** * @include GeoExt/data/FeatureRecord.js * @require OpenLayers/Feature/Vector.js */ /** api: (define) * module = GeoExt.data * class = FeatureReader * base_link = `Ext.data.DataReader `_ */ Ext.namespace('GeoExt', 'GeoExt.data'); /** api: example * Typical usage in a store: * * .. code-block:: javascript * * var store = new Ext.data.Store({ * reader: new GeoExt.data.FeatureReader({}, [ * {name: 'name', type: 'string'}, * {name: 'elevation', type: 'float'} * ]) * }); * */ /** api: constructor * .. class:: FeatureReader(meta, recordType) * * Data reader class to create an array of * :class:`GeoExt.data.FeatureRecord` objects from an * ``OpenLayers.Protocol.Response`` object for use in a * :class:`GeoExt.data.FeatureStore` object. */ GeoExt.data.FeatureReader = function(meta, recordType) { meta = meta || {}; if(!(recordType instanceof Function)) { recordType = GeoExt.data.FeatureRecord.create( recordType || meta.fields || {}); } GeoExt.data.FeatureReader.superclass.constructor.call( this, meta, recordType); }; Ext.extend(GeoExt.data.FeatureReader, Ext.data.DataReader, { /** * APIProperty: totalRecords * {Integer} */ totalRecords: null, /** private: method[read] * :param response: ``OpenLayers.Protocol.Response`` * :return: ``Object`` An object with two properties. The value of the * ``records`` property is the array of records corresponding to * the features. The value of the ``totalRecords" property is the * number of records in the array. * * This method is only used by a DataProxy which has retrieved data. */ read: function(response) { return this.readRecords(response.features); }, /** api: method[readRecords] * :param features: ``Array(OpenLayers.Feature.Vector)`` List of * features for creating records * :return: ``Object`` An object with ``records`` and ``totalRecords`` * properties. * * Create a data block containing :class:`GeoExt.data.FeatureRecord` * objects from an array of features. */ readRecords : function(features) { var records = []; if (features) { var recordType = this.recordType, fields = recordType.prototype.fields; var i, lenI, j, lenJ, feature, values, field, v; for (i = 0, lenI = features.length; i < lenI; i++) { feature = features[i]; values = {}; if (feature.attributes) { for (j = 0, lenJ = fields.length; j < lenJ; j++){ field = fields.items[j]; if (/[\[\.]/.test(field.mapping)) { try { v = new Function("obj", "return obj." + field.mapping)(feature.attributes); } catch(e){ v = field.defaultValue; } } else { v = feature.attributes[field.mapping || field.name] || field.defaultValue; } if (field.convert) { v = field.convert(v, feature); } values[field.name] = v; } } values.feature = feature; values.state = feature.state; values.fid = feature.fid; // newly inserted features need to be made into phantom records var id = (feature.state === OpenLayers.State.INSERT) ? undefined : feature.id; records[records.length] = new recordType(values, id); } } return { records: records, totalRecords: this.totalRecords != null ? this.totalRecords : records.length }; } }); /** FILE: GeoExt/data/FeatureRecord.js **/ /** * Copyright (c) 2008-2012 The Open Source Geospatial Foundation * * Published under the BSD license. * See http://svn.geoext.org/core/trunk/geoext/license.txt for the full text * of the license. */ /** api: (define) * module = GeoExt.data * class = FeatureRecord * base_link = `Ext.data.Record `_ */ Ext.namespace("GeoExt.data"); /** api: constructor * .. class:: FeatureRecord * * A record that represents an ``OpenLayers.Feature.Vector``. This record * will always have at least the following fields: * * * state ``String`` * * fid ``String`` * */ GeoExt.data.FeatureRecord = Ext.data.Record.create([ {name: "feature"}, {name: "state"}, {name: "fid"} ]); /** api: method[getFeature] * :return: ``OpenLayers.Feature.Vector`` * * Gets the feature for this record. */ GeoExt.data.FeatureRecord.prototype.getFeature = function() { return this.get("feature"); }; /** api: method[setFeature] * :param feature: ``OpenLayers.Feature.Vector`` * * Sets the feature for this record. */ GeoExt.data.FeatureRecord.prototype.setFeature = function(feature) { if (feature !== this.data.feature) { this.dirty = true; if (!this.modified) { this.modified = {}; } if (this.modified.feature === undefined) { this.modified.feature = this.data.feature; } this.data.feature = feature; if (!this.editing){ this.afterEdit(); } } }; /** api: classmethod[create] * :param o: ``Array`` Field definition as in ``Ext.data.Record.create``. Can * be omitted if no additional fields are required. * :return: ``Function`` A specialized :class:`GeoExt.data.FeatureRecord` * constructor. * * Creates a constructor for a :class:`GeoExt.data.FeatureRecord`, optionally * with additional fields. */ GeoExt.data.FeatureRecord.create = function(o) { var f = Ext.extend(GeoExt.data.FeatureRecord, {}); var p = f.prototype; p.fields = new Ext.util.MixedCollection(false, function(field) { return field.name; }); GeoExt.data.FeatureRecord.prototype.fields.each(function(f) { p.fields.add(f); }); if(o) { for(var i = 0, len = o.length; i < len; i++){ p.fields.add(new Ext.data.Field(o[i])); } } f.getField = function(name) { return p.fields.get(name); }; return f; }; /** FILE: GeoExt/plugins/PrintPageField.js **/ /** * Copyright (c) 2008-2012 The Open Source Geospatial Foundation * * Published under the BSD license. * See http://svn.geoext.org/core/trunk/geoext/license.txt for the full text * of the license. */ Ext.namespace("GeoExt.plugins"); /** api: (define) * module = GeoExt.plugins * class = PrintPageField * base_link = `Ext.util.Observable `_ */ /** api: example * A form with a combo box for the scale and text fields for rotation and a * page title. The page title is a custom parameter of the print module's * page configuration: * * .. code-block:: javascript * * var printPage = new GeoExt.data.PrintPage({ * printProvider: new GeoExt.data.PrintProvider({ * capabilities: printCapabilities * }) * }); * new Ext.form.FormPanel({ * renderTo: "form", * width: 200, * height: 300, * items: [{ * xtype: "combo", * displayField: "name", * store: printPage.scales, // printPage.scale * name: "scale", * fieldLabel: "Scale", * typeAhead: true, * mode: "local", * forceSelection: true, * triggerAction: "all", * selectOnFocus: true, * plugins: new GeoExt.plugins.PrintPageField({ * printPage: printPage * }) * }, { * xtype: "textfield", * name: "rotation", // printPage.rotation * fieldLabel: "Rotation", * plugins: new GeoExt.plugins.PrintPageField({ * printPage: printPage * }) * }, { * xtype: "textfield", * name: "mapTitle", // printPage.customParams["mapTitle"] * fieldLabel: "Map Title", * plugins: new GeoExt.plugins.PrintPageField({ * printPage: printPage * }) * }] * }); */ /** api: constructor * .. class:: PrintPageField * * A plugin for ``Ext.form.Field`` components which provides synchronization * with a :class:`GeoExt.data.PrintPage`. The field name has to match the * respective property of the printPage (e.g. ``scale``, ``rotation``). */ GeoExt.plugins.PrintPageField = Ext.extend(Ext.util.Observable, { /** api: config[printPage] * ``GeoExt.data.PrintPage`` The print page to synchronize with. */ /** private: property[printPage] * ``GeoExt.data.PrintPage`` The print page to synchronize with. * Read-only. */ printPage: null, /** private: property[target] * ``Ext.form.Field`` This plugin's target field. */ target: null, /** private: method[constructor] */ constructor: function(config) { this.initialConfig = config; Ext.apply(this, config); GeoExt.plugins.PrintPageField.superclass.constructor.apply(this, arguments); }, /** private: method[init] * :param target: ``Ext.form.Field`` The component that this plugin * extends. * @param {Object} target */ init: function(target) { this.target = target; var onCfg = { "beforedestroy": this.onBeforeDestroy, scope: this }; var eventName = target instanceof Ext.form.ComboBox ? "select" : target instanceof Ext.form.Checkbox ? "check" : "valid"; onCfg[eventName] = this.onFieldChange; target.on(onCfg); this.printPage.on({ "change": this.onPageChange, scope: this }); this.printPage.printProvider.on({ "layoutchange": this.onLayoutChange, scope: this }); this.setValue(this.printPage); }, /** private: method[onFieldChange] * :param field: ``Ext.form.Field`` * :param record: ``Ext.data.Record`` Optional. * * Handler for the target field's "valid" or "select" event. */ onFieldChange: function(field, record) { var printProvider = this.printPage.printProvider; var value = field.getValue(); this._updating = true; if(field.store === printProvider.scales || field.name === "scale") { this.printPage.setScale(record); } else if(field.name == "rotation") { !isNaN(value) && this.printPage.setRotation(value); } else { this.printPage.customParams[field.name] = value; } delete this._updating; }, /** private: method[onPageChange] * :param printPage: :class:`GeoExt.data.PrintPage` * * Handler for the "change" event for the page this plugin is configured * with. */ onPageChange: function(printPage) { if(!this._updating) { this.setValue(printPage); } }, /** private: method[onPageChange] * :param printProvider: :class:`GeoExt.data.PrintProvider` * :param layout: ``Ext.Record`` * * Handler for the "layoutchange" event of the printProvider. */ onLayoutChange: function(printProvider, layout) { var t = this.target; t.name == "rotation" && t.setDisabled(!layout.get("rotation")); }, /** private: method[setValue] * :param printPage: :class:`GeoExt.data.PrintPage` * * Sets the value in the target field. */ setValue: function(printPage) { var t = this.target; t.suspendEvents(); if(t.store === printPage.printProvider.scales || t.name === "scale") { if(printPage.scale) { t.setValue(printPage.scale.get(t.displayField)); } } else if(t.name == "rotation") { t.setValue(printPage.rotation); } t.resumeEvents(); }, /** private: method[onBeforeDestroy] */ onBeforeDestroy: function() { this.target.un("beforedestroy", this.onBeforeDestroy, this); this.target.un("select", this.onFieldChange, this); this.target.un("valid", this.onFieldChange, this); this.printPage.un("change", this.onPageChange, this); this.printPage.printProvider.un("layoutchange", this.onLayoutChange, this); } }); /** api: ptype = gx_printpagefield */ Ext.preg("gx_printpagefield", GeoExt.plugins.PrintPageField); /** FILE: GeoExt/plugins/PrintProviderField.js **/ /** * Copyright (c) 2008-2012 The Open Source Geospatial Foundation * * Published under the BSD license. * See http://svn.geoext.org/core/trunk/geoext/license.txt for the full text * of the license. */ Ext.namespace("GeoExt.plugins"); /** api: (define) * module = GeoExt.plugins * class = PrintProviderField * base_link = `Ext.util.Observable `_ */ /** api: example * A form with combo boxes for layout and resolution, and a text field for a * map title. The latter is a custom parameter to the print module, which is * a default for all print pages. For setting custom parameters on the page * level, use :class:`GeoExt.plugins.PrintPageField`): * * .. code-block:: javascript * * var printProvider = new GeoExt.data.PrintProvider({ * capabilities: printCapabilities * }); * new Ext.form.FormPanel({ * renderTo: "form", * width: 200, * height: 300, * items: [{ * xtype: "combo", * displayField: "name", * store: printProvider.layouts, // printProvider.layout * fieldLabel: "Layout", * typeAhead: true, * mode: "local", * forceSelection: true, * triggerAction: "all", * selectOnFocus: true, * plugins: new GeoExt.plugins.PrintProviderField({ * printProvider: printProvider * }) * }, { * xtype: "combo", * displayField: "name", * store: printProvider.dpis, // printProvider.dpi * fieldLabel: "Resolution", * typeAhead: true, * mode: "local", * forceSelection: true, * triggerAction: "all", * selectOnFocus: true, * plugins: new GeoExt.plugins.PrintProviderField({ * printProvider: printProvider * }) * }, { * xtype: "textfield", * name: "mapTitle", // printProvider.customParams.mapTitle * fieldLabel: "Map Title", * plugins: new GeoExt.plugins.PrintProviderField({ * printProvider: printProvider * }) * }] * }): */ /** api: constructor * .. class:: PrintProviderField * * A plugin for ``Ext.form.Field`` components which provides synchronization * with a :class:`GeoExt.data.PrintProvider`. */ GeoExt.plugins.PrintProviderField = Ext.extend(Ext.util.Observable, { /** api: config[printProvider] * ``GeoExt.data.PrintProvider`` The print provider to use with this * plugin's field. Not required if set on the owner container of the * field. */ /** private: property[target] * ``Ext.form.Field`` This plugin's target field. */ target: null, /** private: method[constructor] */ constructor: function(config) { this.initialConfig = config; Ext.apply(this, config); GeoExt.plugins.PrintProviderField.superclass.constructor.apply(this, arguments); }, /** private: method[init] * :param target: ``Ext.form.Field`` The component that this plugin * extends. */ init: function(target) { this.target = target; var onCfg = { scope: this, "render": this.onRender, "beforedestroy": this.onBeforeDestroy }; onCfg[target instanceof Ext.form.ComboBox ? "select" : "valid"] = this.onFieldChange; target.on(onCfg); }, /** private: method[onRender] * :param field: ``Ext.Form.Field`` * * Handler for the target field's "render" event. */ onRender: function(field) { var printProvider = this.printProvider || field.ownerCt.printProvider; if(field.store === printProvider.layouts) { field.setValue(printProvider.layout.get(field.displayField)); printProvider.on({ "layoutchange": this.onProviderChange, scope: this }); } else if(field.store === printProvider.dpis) { field.setValue(printProvider.dpi.get(field.displayField)); printProvider.on({ "dpichange": this.onProviderChange, scope: this }); } else if(field.initialConfig.value === undefined) { field.setValue(printProvider.customParams[field.name]); } }, /** private: method[onFieldChange] * :param field: ``Ext.form.Field`` * :param record: ``Ext.data.Record`` Optional. * * Handler for the target field's "valid" or "select" event. */ onFieldChange: function(field, record) { var printProvider = this.printProvider || field.ownerCt.printProvider; var value = field.getValue(); this._updating = true; if(record) { switch(field.store) { case printProvider.layouts: printProvider.setLayout(record); break; case printProvider.dpis: printProvider.setDpi(record); } } else { printProvider.customParams[field.name] = value; } delete this._updating; }, /** private: method[onProviderChange] * :param printProvider: :class:`GeoExt.data.PrintProvider` * :param rec: ``Ext.data.Record`` * * Handler for the printProvider's dpichange and layoutchange event */ onProviderChange: function(printProvider, rec) { if(!this._updating) { this.target.setValue(rec.get(this.target.displayField)); } }, /** private: method[onBeforeDestroy] */ onBeforeDestroy: function() { var target = this.target; target.un("beforedestroy", this.onBeforeDestroy, this); target.un("render", this.onRender, this); target.un("select", this.onFieldChange, this); target.un("valid", this.onFieldChange, this); var printProvider = this.printProvider || target.ownerCt.printProvider; printProvider.un("layoutchange", this.onProviderChange, this); printProvider.un("dpichange", this.onProviderChange, this); } }); /** api: ptype = gx_printproviderfield */ Ext.preg("gx_printproviderfield", GeoExt.plugins.PrintProviderField); /** FILE: 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, /** * 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, /** * 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 and * to change control state. */ active: null, /** * Property: handlerOptions * {Object} Used to set non-default properties on the control's handler */ handlerOptions: null, /** * Property: handler * {} null */ handler: null, /** * 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. * * 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 - {} */ 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. * * Parameters: * px - {} */ 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; /** FILE: GeoExt/widgets/Action.js **/ /** * Copyright (c) 2008-2012 The Open Source Geospatial Foundation * * Published under the BSD license. * See http://svn.geoext.org/core/trunk/geoext/license.txt for the full text * of the license. */ /** * @require OpenLayers/Control.js */ /** api: (define) * module = GeoExt * class = Action * base_link = `Ext.Action `_ */ Ext.namespace("GeoExt"); /** api: example * Sample code to create a toolbar with an OpenLayers control into it. * * .. code-block:: javascript * * var action = new GeoExt.Action({ * text: "max extent", * control: new OpenLayers.Control.ZoomToMaxExtent(), * map: map * }); * var toolbar = new Ext.Toolbar([action]); */ /** api: constructor * .. class:: Action(config) * * Create a GeoExt.Action instance. A GeoExt.Action is created * to insert an OpenLayers control in a toolbar as a button or * in a menu as a menu item. A GeoExt.Action instance can be * used like a regular Ext.Action, look at the Ext.Action API * doc for more detail. */ GeoExt.Action = Ext.extend(Ext.Action, { /** api: config[control] * ``OpenLayers.Control`` The OpenLayers control wrapped in this action. */ control: null, /** api: config[activateOnEnable] * ``Boolean`` Activate the action's control when the action is enabled. * Default is ``false``. */ /** api: property[activateOnEnable] * ``Boolean`` Activate the action's control when the action is enabled. */ activateOnEnable: false, /** api: config[deactivateOnDisable] * ``Boolean`` Deactivate the action's control when the action is disabled. * Default is ``false``. */ /** api: property[deactivateOnDisable] * ``Boolean`` Deactivate the action's control when the action is disabled. */ deactivateOnDisable: false, /** api: config[map] * ``OpenLayers.Map`` The OpenLayers map that the control should be added * to. For controls that don't need to be added to a map or have already * been added to one, this config property may be omitted. */ map: null, /** private: property[uScope] * ``Object`` The user-provided scope, used when calling uHandler, * uToggleHandler, and uCheckHandler. */ uScope: null, /** private: property[uHandler] * ``Function`` References the function the user passes through * the "handler" property. */ uHandler: null, /** private: property[uToggleHandler] * ``Function`` References the function the user passes through * the "toggleHandler" property. */ uToggleHandler: null, /** private: property[uCheckHandler] * ``Function`` References the function the user passes through * the "checkHandler" property. */ uCheckHandler: null, /** private */ constructor: function(config) { // store the user scope and handlers this.uScope = config.scope; this.uHandler = config.handler; this.uToggleHandler = config.toggleHandler; this.uCheckHandler = config.checkHandler; config.scope = this; config.handler = this.pHandler; config.toggleHandler = this.pToggleHandler; config.checkHandler = this.pCheckHandler; // set control in the instance, the Ext.Action // constructor won't do it for us var ctrl = this.control = config.control; delete config.control; this.activateOnEnable = !!config.activateOnEnable; delete config.activateOnEnable; this.deactivateOnDisable = !!config.deactivateOnDisable; delete config.deactivateOnDisable; // register "activate" and "deactivate" listeners // on the control if(ctrl) { // If map is provided in config, add control to map. if(config.map) { config.map.addControl(ctrl); delete config.map; } if((config.pressed || config.checked) && ctrl.map) { ctrl.activate(); } if (ctrl.active) { config.pressed = true; config.checked = true; } ctrl.events.on({ activate: this.onCtrlActivate, deactivate: this.onCtrlDeactivate, scope: this }); } arguments.callee.superclass.constructor.call(this, config); }, /** private: method[pHandler] * :param cmp: ``Ext.Component`` The component that triggers the handler. * * The private handler. */ pHandler: function(cmp) { var ctrl = this.control; if(ctrl && ctrl.type == OpenLayers.Control.TYPE_BUTTON) { ctrl.trigger(); } if(this.uHandler) { this.uHandler.apply(this.uScope, arguments); } }, /** private: method[pTogleHandler] * :param cmp: ``Ext.Component`` The component that triggers the toggle handler. * :param state: ``Boolean`` The state of the toggle. * * The private toggle handler. */ pToggleHandler: function(cmp, state) { this.changeControlState(state); if(this.uToggleHandler) { this.uToggleHandler.apply(this.uScope, arguments); } }, /** private: method[pCheckHandler] * :param cmp: ``Ext.Component`` The component that triggers the check handler. * :param state: ``Boolean`` The state of the toggle. * * The private check handler. */ pCheckHandler: function(cmp, state) { this.changeControlState(state); if(this.uCheckHandler) { this.uCheckHandler.apply(this.uScope, arguments); } }, /** private: method[changeControlState] * :param state: ``Boolean`` The state of the toggle. * * Change the control state depending on the state boolean. */ changeControlState: function(state) { if(state) { if(!this._activating) { this._activating = true; this.control.activate(); // update initialConfig for next component created from this action this.initialConfig.pressed = true; this.initialConfig.checked = true; this._activating = false; } } else { if(!this._deactivating) { this._deactivating = true; this.control.deactivate(); // update initialConfig for next component created from this action this.initialConfig.pressed = false; this.initialConfig.checked = false; this._deactivating = false; } } }, /** private: method[onCtrlActivate] * * Called when this action's control is activated. */ onCtrlActivate: function() { var ctrl = this.control; if(ctrl.type == OpenLayers.Control.TYPE_BUTTON) { this.enable(); } else { // deal with buttons this.safeCallEach("toggle", [true]); // deal with check items this.safeCallEach("setChecked", [true]); } }, /** private: method[onCtrlDeactivate] * * Called when this action's control is deactivated. */ onCtrlDeactivate: function() { var ctrl = this.control; if(ctrl.type == OpenLayers.Control.TYPE_BUTTON) { this.disable(); } else { // deal with buttons this.safeCallEach("toggle", [false]); // deal with check items this.safeCallEach("setChecked", [false]); } }, /** private: method[safeCallEach] * */ safeCallEach: function(fnName, args) { var cs = this.items; for(var i = 0, len = cs.length; i < len; i++){ if(cs[i][fnName]) { cs[i].rendered ? cs[i][fnName].apply(cs[i], args) : cs[i].on({ "render": cs[i][fnName].createDelegate(cs[i], args), single: true }); } } }, /** private: method[setDisabled] * :param v: ``Boolean`` Disable the action's components. * * Override method on super to optionally deactivate controls on disable. */ setDisabled : function(v) { if (!v && this.activateOnEnable && this.control && !this.control.active) { this.control.activate(); } if (v && this.deactivateOnDisable && this.control && this.control.active) { this.control.deactivate(); } return GeoExt.Action.superclass.setDisabled.apply(this, arguments); } }); /** FILE: GeoExt/widgets/tree/LayerContainer.js **/ /** * Copyright (c) 2008-2012 The Open Source Geospatial Foundation * * Published under the BSD license. * See http://svn.geoext.org/core/trunk/geoext/license.txt for the full text * of the license. */ /** * @include GeoExt/widgets/tree/LayerLoader.js */ Ext.namespace("GeoExt.tree"); /** api: (define) * module = GeoExt.tree * class = LayerContainer * base_link = `Ext.tree.AsyncTreeNode `_ */ /** api: constructor * .. class:: LayerContainer * * A subclass of ``Ext.tree.AsyncTreeNode`` that will collect all layers of an * OpenLayers map. Only layers that have displayInLayerSwitcher set to true * will be included. The childrens' iconCls defaults to * "gx-tree-layer-icon" and this node' text defaults to "Layers". * * Note: if this container is loaded by an ``Ext.tree.TreeLoader``, the * ``applyLoader`` config option of that loader needs to be set to * "false". Also note that the list of available uiProviders will be * taken from the ownerTree if this container's loader is configured * without one. * * To use this node type in ``TreePanel`` config, set nodeType to * "gx_layercontainer". */ GeoExt.tree.LayerContainer = Ext.extend(Ext.tree.AsyncTreeNode, { /** api: config[loader] * :class:`GeoExt.tree.LayerLoader` or ``Object`` The loader to use with * this container. If an ``Object`` is provided, a * :class:`GeoExt.tree.LayerLoader`, configured with the the properties * from the provided object, will be created. */ /** api: config[layerStore] * :class:`GeoExt.data.LayerStore` The layer store containing layers to be * displayed in the container. If loader is not provided or provided as * ``Object``, this property will be set as the store option of the * loader. Otherwise it will be ignored. */ /** private: property[text] * ``String`` The text for this node. */ text: 'Layers', /** private: method[constructor] * Private constructor override. */ constructor: function(config) { config = Ext.applyIf(config || {}, { text: this.text }); this.loader = config.loader instanceof GeoExt.tree.LayerLoader ? config.loader : new GeoExt.tree.LayerLoader(Ext.applyIf(config.loader || {}, { store: config.layerStore })); GeoExt.tree.LayerContainer.superclass.constructor.call(this, config); }, /** private: method[recordIndexToNodeIndex] * :param index: ``Number`` The record index in the layer store. * :return: ``Number`` The appropriate child node index for the record. */ recordIndexToNodeIndex: function(index) { var store = this.loader.store; var count = store.getCount(); var nodeCount = this.childNodes.length; var nodeIndex = -1; for(var i=count-1; i>=0; --i) { if(this.loader.filter(store.getAt(i)) === true) { ++nodeIndex; if(index === i || nodeIndex > nodeCount-1) { break; } } } return nodeIndex; }, /** private: method[destroy] */ destroy: function() { delete this.layerStore; GeoExt.tree.LayerContainer.superclass.destroy.apply(this, arguments); } }); /** * NodeType: gx_layercontainer */ Ext.tree.TreePanel.nodeTypes.gx_layercontainer = GeoExt.tree.LayerContainer; /** FILE: GeoExt/widgets/tree/LayerLoader.js **/ /** * Copyright (c) 2008-2012 The Open Source Geospatial Foundation * * Published under the BSD license. * See http://svn.geoext.org/core/trunk/geoext/license.txt for the full text * of the license. */ /** * @include GeoExt/widgets/MapPanel.js * @include GeoExt/widgets/tree/LayerNode.js * @include GeoExt/widgets/tree/LayerContainer.js */ Ext.namespace("GeoExt.tree"); /** api: (define) * module = GeoExt.tree * class = LayerLoader * base_link = `Ext.util.Observable `_ */ /** api: constructor * .. class:: LayerLoader * * A loader that will load all layers of a :class:`GeoExt.data.LayerStore` * By default, only layers that have displayInLayerSwitcher set to true * will be included. The childrens' iconCls defaults to * "gx-tree-layer-icon". */ GeoExt.tree.LayerLoader = function(config) { Ext.apply(this, config); this.addEvents( /** api: event[beforeload] * Triggered before loading children. Return false to avoid * loading children. * * Listener arguments: * * * loader - :class:`GeoExt.tree.LayerLoader` this loader * * node - ``Ex.tree.TreeNode`` the node that this loader is * configured with */ "beforeload", /** api: event[load] * Triggered after children wer loaded. * * Listener arguments: * * * loader - :class:`GeoExt.tree.LayerLoader` this loader * * node - ``Ex.tree.TreeNode`` the node that this loader is * configured with */ "load" ); this.iconCls = {}; GeoExt.tree.LayerLoader.superclass.constructor.call(this); }; Ext.extend(GeoExt.tree.LayerLoader, Ext.util.Observable, { /** api: config[store] * :class:`GeoExt.data.LayerStore` * The layer store containing layers to be added by this loader. */ store: null, /** api: config[filter] * ``Function`` * A function, called in the scope of this loader, with a layer record * as argument. Is expected to return true for layers to be loaded, false * otherwise. By default, the filter checks for displayInLayerSwitcher: * * .. code-block:: javascript * * filter: function(record) { * return record.getLayer().displayInLayerSwitcher == true * } */ filter: function(record) { return record.getLayer().displayInLayerSwitcher == true; }, /** api: config[baseAttrs] * An object containing attributes to be added to all nodes created by * this loader. */ baseAttrs: null, /** api: config[uiProviders] * ``Object`` * An optional object containing properties which specify custom * GeoExt.tree.LayerNodeUI implementations. If the optional uiProvider * attribute for child nodes is a string rather than a reference to a * TreeNodeUI implementation, then that string value is used as a * property name in the uiProviders object. If not provided, the * uiProviders object will be taken from the ownerTree's loader. */ uiProviders: null, /** private: property[iconCls] * ``Object`` An object where the keys are the record ids and the * values are the iconCls values of the corresponding nodes. This is used * to restore the iconCls of a node after move whenever possible. It is * needed since moving a layer node involves removing it and re-adding it. */ /** private: method[load] * :param node: ``Ext.tree.TreeNode`` The node to add children to. * :param callback: ``Function`` */ load: function(node, callback) { if(this.fireEvent("beforeload", this, node)) { this.removeStoreHandlers(); while (node.firstChild) { node.removeChild(node.firstChild); } if(!this.uiProviders) { this.uiProviders = node.getOwnerTree().getLoader().uiProviders; } if(!this.store) { this.store = GeoExt.MapPanel.guess().layers; } this.store.each(function(record) { this.addLayerNode(node, record); }, this); this.addStoreHandlers(node); if(typeof callback == "function"){ callback(); } this.fireEvent("load", this, node); } }, /** private: method[onStoreAdd] * :param store: ``Ext.data.Store`` * :param records: ``Array(Ext.data.Record)`` * :param index: ``Number`` * :param node: ``Ext.tree.TreeNode`` * * Listener for the store's add event. */ onStoreAdd: function(store, records, index, node) { if(!this._reordering) { var nodeIndex = node.recordIndexToNodeIndex(index+records.length-1); for(var i=0; i 1) { // find index by neighboring node in the same container var searchIndex = (index === 0) ? index + 1 : index - 1; newRecordIndex = this.store.findBy(function(r) { return newParent.childNodes[searchIndex].layer === r.getLayer(); }); index === 0 && newRecordIndex++; } else if(oldParent.parentNode === newParent.parentNode){ // find index by last node of a container above var prev = newParent; do { prev = prev.previousSibling; } while (prev && !(prev instanceof GeoExt.tree.LayerContainer && prev.lastChild)); if(prev) { newRecordIndex = this.store.findBy(function(r) { return prev.lastChild.layer === r.getLayer(); }); } else { // find indext by first node of a container below var next = newParent; do { next = next.nextSibling; } while (next && !(next instanceof GeoExt.tree.LayerContainer && next.firstChild)); if(next) { newRecordIndex = this.store.findBy(function(r) { return next.firstChild.layer === r.getLayer(); }); } newRecordIndex++; } } if(newRecordIndex !== undefined) { this.store.insert(newRecordIndex, [record]); window.setTimeout(function() { newParent.reload(); oldParent.reload(); }); } else { this.store.insert(oldRecordIndex, [record]); } delete newParent.loader._reordering; } delete this._reordering; }, /** private: method[addStoreHandlers] * :param node: :class:`GeoExt.tree.LayerNode` */ addStoreHandlers: function(node) { if(!this._storeHandlers) { this._storeHandlers = { "add": this.onStoreAdd.createDelegate(this, [node], true), "remove": this.onStoreRemove.createDelegate(this, [node], true) }; for(var evt in this._storeHandlers) { this.store.on(evt, this._storeHandlers[evt], this); } } }, /** private: method[removeStoreHandlers] */ removeStoreHandlers: function() { if(this._storeHandlers) { for(var evt in this._storeHandlers) { this.store.un(evt, this._storeHandlers[evt], this); } delete this._storeHandlers; } }, /** api: method[createNode] * :param attr: ``Object`` attributes for the new node * * Override this function for custom TreeNode node implementation, or to * modify the attributes at creation time. */ createNode: function(attr) { if(this.baseAttrs){ Ext.apply(attr, this.baseAttrs); } if(typeof attr.uiProvider == 'string'){ attr.uiProvider = this.uiProviders[attr.uiProvider] || eval(attr.uiProvider); } attr.nodeType = attr.nodeType || "gx_layer"; return new Ext.tree.TreePanel.nodeTypes[attr.nodeType](attr); }, /** private: method[destroy] */ destroy: function() { this.removeStoreHandlers(); this.iconCls = null; } }); /** FILE: 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({ /** * APIProperty: id * {String} */ id: null, /** * APIProperty: name * {String} */ name: null, /** * APIProperty: div * {DOMElement} */ div: null, /** * APIProperty: opacity * {Float} The layer's opacity. Float number between 0.0 and 1.0. Default * is 1. */ opacity: 1, /** * 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. */ alwaysInRange: null, /** * Constant: RESOLUTION_PROPERTIES * {Array} The properties that are used for calculating resolutions * information. */ RESOLUTION_PROPERTIES: [ 'scales', 'resolutions', 'maxScale', 'minScale', 'maxResolution', 'minResolution', 'numZoomLevels', 'maxZoomLevel' ], /** * 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. */ events: null, /** * APIProperty: 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 * 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. */ inRange: false, /** * 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. */ options: null, /** * 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: 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 * {} 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. * */ 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 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 * {|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. */ 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. */ 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 * 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: * name - {String} The layer name * options - {Object} Hashtable of extra options to tag onto the layer */ 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); } } }, /** * 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; }, /** * Method: clone * * Parameters: * obj - {} The layer to be cloned * * Returns: * {} An exact clone of this */ 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 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" }); } } }, /** * 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) { 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; } } } }, /** * 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: * {Boolean} The layer was redrawn. */ 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; } } return redrawn; }, /** * Method: moveTo * * 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; } 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 - {} */ 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(); } }, /** * 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 - {} */ removeMap: function(map) { //to be overridden by subclasses }, /** * 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. */ 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 - {} */ 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)); } }, /** * 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" }); } 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"; } }, /** * 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 { if (this.map) { var resolution = this.map.getResolution(); inRange = ( (resolution >= this.minResolution) && (resolution <= this.maxResolution) ); } } return inRange; }, /** * APIMethod: setIsBaseLayer * * Parameters: * isBaseLayer - {Boolean} */ setIsBaseLayer: function(isBaseLayer) { if (isBaseLayer != this.isBaseLayer) { this.isBaseLayer = isBaseLayer; if (this.map != null) { this.map.events.triggerEvent("changebaselayer", { layer: this }); } } }, /********************************************************/ /* */ /* 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} 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(); }, /** * APIMethod: getZoomForExtent * * 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. */ 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: * {} */ getDataExtent: function () { //to be implemented by subclasses }, /** * APIMethod: getResolutionForZoom * * Parameters: * zoom - {Float} * * 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; }, /** * 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. */ 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= 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 minDiff) { break; } minDiff = diff; } else { if (this.resolutions[i] < resolution) { break; } } } zoom = Math.max(0, i-1); } return zoom; }, /** * APIMethod: getLonLatFromViewPortPx * * Parameters: * viewPortPx - {|Object} An OpenLayers.Pixel or * an object with a 'x' * and 'y' properties. * * Returns: * {} An OpenLayers.LonLat which is the passed-in * view port , translated into lon/lat by the layer. */ 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: 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)) ); } return px; }, /** * 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 - {} */ 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" }); /** FILE: GeoExt/widgets/tree/LayerNode.js **/ /** * Copyright (c) 2008-2012 The Open Source Geospatial Foundation * * Published under the BSD license. * See http://svn.geoext.org/core/trunk/geoext/license.txt for the full text * of the license. */ /** * @include GeoExt/widgets/MapPanel.js * @require OpenLayers/Layer.js */ Ext.namespace("GeoExt.tree"); /** private: constructor * .. class:: LayerNodeUI * * Place in a separate file if this should be documented. */ GeoExt.tree.LayerNodeUI = Ext.extend(Ext.tree.TreeNodeUI, { /** private: method[constructor] */ constructor: function(config) { GeoExt.tree.LayerNodeUI.superclass.constructor.apply(this, arguments); }, /** private: method[render] * :param bulkRender: ``Boolean`` */ render: function(bulkRender) { var a = this.node.attributes; if (a.checked === undefined) { a.checked = this.node.layer.getVisibility(); } /* Ext.tree.treeNodeUI render looks for and handles checked * attribute, but not the disabled attribute, so we set it * directly on the node object and not the attributes hash*/ if (a.disabled === undefined && this.node.autoDisable) { this.node.disabled = this.node.layer.inRange === false || !this.node.layer.calculateInRange(); } GeoExt.tree.LayerNodeUI.superclass.render.apply(this, arguments); var cb = this.checkbox; if(a.checkedGroup) { // replace the checkbox with a radio button var radio = Ext.DomHelper.insertAfter(cb, [''].join("")); radio.defaultChecked = cb.defaultChecked; Ext.get(cb).remove(); this.checkbox = radio; } this.enforceOneVisible(); }, /** private: method[onClick] * :param e: ``Object`` */ onClick: function(e) { if(e.getTarget('.x-tree-node-cb', 1)) { this.toggleCheck(this.isChecked()); } else { GeoExt.tree.LayerNodeUI.superclass.onClick.apply(this, arguments); } }, /** private: method[toggleCheck] * :param value: ``Boolean`` */ toggleCheck: function(value) { value = (value === undefined ? !this.isChecked() : value); GeoExt.tree.LayerNodeUI.superclass.toggleCheck.call(this, value); this.enforceOneVisible(); }, /** private: method[enforceOneVisible] * * Makes sure that only one layer is visible if checkedGroup is set. */ enforceOneVisible: function() { var attributes = this.node.attributes; var group = attributes.checkedGroup; // If we are in the baselayer group, the map will take care of // enforcing visibility. if(group && group !== "gx_baselayer") { var layer = this.node.layer; var checkedNodes = this.node.getOwnerTree().getChecked(); var checkedCount = 0; // enforce "not more than one visible" Ext.each(checkedNodes, function(n){ var l = n.layer; if(!n.hidden && n.attributes.checkedGroup === group) { checkedCount++; if(l != layer && attributes.checked) { l.setVisibility(false); } } }); // enforce "at least one visible" if(checkedCount === 0 && attributes.checked == false) { layer.setVisibility(true); } } }, /** private: method[appendDDGhost] * :param ghostNode ``DOMElement`` * * For radio buttons, makes sure that we do not use the option group of * the original, otherwise only the original or the clone can be checked */ appendDDGhost : function(ghostNode){ var n = this.elNode.cloneNode(true); var radio = Ext.DomQuery.select("input[type='radio']", n); Ext.each(radio, function(r) { r.name = r.name + "_clone"; }); ghostNode.appendChild(n); } }); /** api: (define) * module = GeoExt.tree * class = LayerNode * base_link = `Ext.tree.TreeNode `_ */ /** api: constructor * .. class:: LayerNode(config) * * A subclass of ``Ext.tree.TreeNode`` that is connected to an * ``OpenLayers.Layer`` by setting the node's layer property. Checking or * unchecking the checkbox of this node will directly affect the layer and * vice versa. The default iconCls for this node's icon is * "gx-tree-layer-icon", unless it has children. * * Setting the node's layer property to a layer name instead of an object * will also work. As soon as a layer is found, it will be stored as layer * property in the attributes hash. * * The node's text property defaults to the layer name. * * If the node has a checkedGroup attribute configured, it will be * rendered with a radio button instead of the checkbox. The value of * the checkedGroup attribute is a string, identifying the options group * for the node. * * To use this node type in a ``TreePanel`` config, set ``nodeType`` to * "gx_layer". */ GeoExt.tree.LayerNode = Ext.extend(Ext.tree.AsyncTreeNode, { /** api: config[layer] * ``OpenLayers.Layer or String`` * The layer that this layer node will * be bound to, or the name of the layer (has to match the layer's * name property). If a layer name is provided, ``layerStore`` also has * to be provided. */ /** api: property[layer] * ``OpenLayers.Layer`` * The layer this node is bound to. */ layer: null, /** api: property[autoDisable] * ``Boolean`` * Should this node automattically disable itself when the layer * is out of range and enable itself when the layer is in range. * Defaults to true, unless ``layer`` has ``isBaseLayer``==true * or ``alwaysInRange``==true. */ autoDisable: null, /** api: config[layerStore] * :class:`GeoExt.data.LayerStore` ``or "auto"`` * The layer store containing the layer that this node represents. If set * to "auto", the node will query the ComponentManager for a * :class:`GeoExt.MapPanel`, take the first one it finds and take its layer * store. This property is only required if ``layer`` is provided as a * string. */ layerStore: null, /** api: config[checkedGroup] * ``String`` If provided, nodes will be rendered with a radio button * instead of a checkbox. All layers represented by nodes with the same * checkedGroup are considered mutually exclusive - only one can be * visible at a time. */ /** api: config[loader] * ``Ext.tree.TreeLoader|Object`` If provided, subnodes will be added to * this LayerNode. Obviously, only loaders that process an * ``OpenLayers.Layer`` or :class:`GeoExt.data.LayerRecord` (like * :class:`GeoExt.tree.LayerParamsLoader`) will actually generate child * nodes here. If provided as ``Object``, a * :class:`GeoExt.tree.LayerParamLoader` instance will be created, with * the provided object as configuration. */ /** private: method[constructor] * Private constructor override. */ constructor: function(config) { config.leaf = config.leaf || !(config.children || config.loader); if(!config.iconCls && !config.children) { config.iconCls = "gx-tree-layer-icon"; } if(config.loader && !(config.loader instanceof Ext.tree.TreeLoader)) { config.loader = new GeoExt.tree.LayerParamLoader(config.loader); } this.defaultUI = this.defaultUI || GeoExt.tree.LayerNodeUI; Ext.apply(this, { layer: config.layer, layerStore: config.layerStore, autoDisable: config.autoDisable }); if (config.text) { this.fixedText = true; } GeoExt.tree.LayerNode.superclass.constructor.apply(this, arguments); }, /** private: method[render] * :param bulkRender: ``Boolean`` */ render: function(bulkRender) { var layer = this.layer instanceof OpenLayers.Layer && this.layer; if(!layer) { // guess the store if not provided if(!this.layerStore || this.layerStore == "auto") { this.layerStore = GeoExt.MapPanel.guess().layers; } // now we try to find the layer by its name in the layer store var i = this.layerStore.findBy(function(o) { return o.get("title") == this.layer; }, this); if(i != -1) { // if we found the layer, we can assign it and everything // will be fine layer = this.layerStore.getAt(i).getLayer(); } } if (!this.rendered || !layer) { var ui = this.getUI(); if(layer) { this.layer = layer; // no DD and radio buttons for base layers if(layer.isBaseLayer) { this.draggable = false; Ext.applyIf(this.attributes, { checkedGroup: "gx_baselayer" }); } //base layers & alwaysInRange layers should never be auto-disabled this.autoDisable = !(this.autoDisable===false || this.layer.isBaseLayer || this.layer.alwaysInRange); if(!this.text) { this.text = layer.name; } ui.show(); this.addVisibilityEventHandlers(); } else { ui.hide(); } if(this.layerStore instanceof GeoExt.data.LayerStore) { this.addStoreEventHandlers(layer); } } GeoExt.tree.LayerNode.superclass.render.apply(this, arguments); }, /** private: method[addVisibilityHandlers] * Adds handlers that sync the checkbox state with the layer's visibility * state */ addVisibilityEventHandlers: function() { this.layer.events.on({ "visibilitychanged": this.onLayerVisibilityChanged, scope: this }); this.on({ "checkchange": this.onCheckChange, scope: this }); if(this.autoDisable){ if (this.layer.map) { this.layer.map.events.register("moveend", this, this.onMapMoveend); } else { this.layer.events.register("added", this, function added() { this.layer.events.unregister("added", this, added); this.layer.map.events.register("moveend", this, this.onMapMoveend); }); } } }, /** private: method[onLayerVisiilityChanged * handler for visibilitychanged events on the layer */ onLayerVisibilityChanged: function() { if(!this._visibilityChanging) { this.getUI().toggleCheck(this.layer.getVisibility()); } }, /** private: method[onCheckChange] * :param node: ``GeoExt.tree.LayerNode`` * :param checked: ``Boolean`` * * handler for checkchange events */ onCheckChange: function(node, checked) { if(checked != this.layer.getVisibility()) { this._visibilityChanging = true; var layer = this.layer; if(checked && layer.isBaseLayer && layer.map) { layer.map.setBaseLayer(layer); } else { layer.setVisibility(checked); } delete this._visibilityChanging; } }, /** private: method[onMapMoveend] * :param evt: ``OpenLayers.Event`` * * handler for map moveend events to determine if node should be * disabled or enabled */ onMapMoveend: function(evt){ /* scoped to node */ if (this.autoDisable) { if (this.layer.inRange === false) { this.disable(); } else { this.enable(); } } }, /** private: method[addStoreEventHandlers] * Adds handlers that make sure the node disappeares when the layer is * removed from the store, and appears when it is re-added. */ addStoreEventHandlers: function() { this.layerStore.on({ "add": this.onStoreAdd, "remove": this.onStoreRemove, "update": this.onStoreUpdate, scope: this }); }, /** private: method[onStoreAdd] * :param store: ``Ext.data.Store`` * :param records: ``Array(Ext.data.Record)`` * :param index: ``Number`` * * handler for add events on the store */ onStoreAdd: function(store, records, index) { var l; for(var i=0; i`_ */ Ext.namespace("GeoExt"); /** api: example * Sample code to create a popup anchored to a feature: * * .. code-block:: javascript * * var popup = new GeoExt.Popup({ * title: "My Popup", * location: feature, * width: 200, * html: "
Popup content
", * collapsible: true * }); */ /** api: constructor * .. class:: Popup(config) * * Popups are a specialized Window that supports anchoring * to a particular location in a MapPanel. When a popup * is anchored to a location, that means that the popup * will visibly point to the location on the map, and move * accordingly when the map is panned or zoomed. */ GeoExt.Popup = Ext.extend(Ext.Window, { /** api: config[anchored] * ``Boolean`` The popup begins anchored to its location. Default is * ``true``. */ anchored: true, /** api: config[map] * ``OpenLayers.Map`` or :class:`GeoExt.MapPanel` * The map this popup will be anchored to (only required if ``anchored`` * is set to true and the map cannot be derived from the ``location``'s * layer. */ map: null, /** api: config[panIn] * ``Boolean`` The popup should pan the map so that the popup is * fully in view when it is rendered. Default is ``true``. */ panIn: true, /** api: config[unpinnable] * ``Boolean`` The popup should have a "unpin" tool that unanchors it from * its location. Default is ``true``. */ unpinnable: true, /** api: config[location] * ``OpenLayers.Feature.Vector`` or ``OpenLayers.LonLat`` or * ``OpenLayers.Pixel`` or ``OpenLayers.Geometry`` A location for this * popup's anchor. */ /** private: property[location] * ``OpenLayers.LonLat`` */ location: null, /** private: property[insideViewport] * ``Boolean`` Wether the popup is currently inside the map viewport. */ insideViewport: null, /** * Some Ext.Window defaults need to be overriden here * because some Ext.Window behavior is not currently supported. */ /** private: config[animCollapse] * ``Boolean`` Animate the transition when the panel is collapsed. * Default is ``false``. Collapsing animation is not supported yet for * popups. */ animCollapse: false, /** private: config[draggable] * ``Boolean`` Enable dragging of this Panel. Defaults to ``false`` * because the popup defaults to being anchored, and anchored popups * should not be draggable. */ draggable: false, /** private: config[shadow] * ``Boolean`` Give the popup window a shadow. Defaults to ``false`` * because shadows are not supported yet for popups (the shadow does * not look good with the anchor). */ shadow: false, /** api: config[popupCls] * ``String`` CSS class name for the popup DOM elements. Default is * "gx-popup". */ popupCls: "gx-popup", /** api: config[ancCls] * ``String`` CSS class name for the popup's anchor. */ ancCls: null, /** api: config[anchorPosition] * ``String`` Controls the anchor position for the popup. If set to * ``auto``, the anchor will be positioned on the top or the bottom of * the window, minimizing map movement. Supported values are ``bottom-left``, * ``bottom-right``, ``top-left``, ``top-right`` or ``auto``. * Defaults to ``auto``. */ anchorPosition: "auto", /** private: method[initComponent] * Initializes the popup. */ initComponent: function() { if(this.map instanceof GeoExt.MapPanel) { this.map = this.map.map; } if(!this.map && this.location instanceof OpenLayers.Feature.Vector && this.location.layer) { this.map = this.location.layer.map; } if (this.location instanceof OpenLayers.Feature.Vector) { this.location = this.location.geometry; } if (this.location instanceof OpenLayers.Geometry) { if (typeof this.location.getCentroid == "function") { this.location = this.location.getCentroid(); } this.location = this.location.getBounds().getCenterLonLat(); } else if (this.location instanceof OpenLayers.Pixel) { this.location = this.map.getLonLatFromViewPortPx(this.location); } if (!(this.location instanceof OpenLayers.LonLat)) { this.anchored = false; } var mapExtent = this.map.getExtent(); if (mapExtent && this.location) { this.insideViewport = mapExtent.containsLonLat(this.location); } if(this.anchored) { this.addAnchorEvents(); this.elements += ',anc'; } else { this.unpinnable = false; } this.baseCls = this.popupCls + " " + this.baseCls; GeoExt.Popup.superclass.initComponent.call(this); }, /** private: method[onRender] * Executes when the popup is rendered. */ onRender: function(ct, position) { GeoExt.Popup.superclass.onRender.call(this, ct, position); if (this.anchored) { this.ancCls = this.popupCls + "-anc"; //create anchor dom element. this.createElement("anc", this.el.dom); } else { this.makeDraggable(); } }, /** private: method[initTools] * Initializes the tools on the popup. In particular, * it adds the 'unpin' tool if the popup is unpinnable. */ initTools : function() { if(this.unpinnable) { this.addTool({ id: 'unpin', handler: this.unanchorPopup.createDelegate(this, []) }); } GeoExt.Popup.superclass.initTools.call(this); }, /** private: method[show] * Override. */ show: function() { GeoExt.Popup.superclass.show.apply(this, arguments); if(this.anchored) { this.position(); if(this.panIn && !this._mapMove) { this.panIntoView(); } } }, /** private: method[maximize] * Override. */ maximize: function() { if(!this.maximized && this.anc) { this.unanchorPopup(); } GeoExt.Popup.superclass.maximize.apply(this, arguments); }, /** api: method[setSize] * :param w: ``Integer`` * :param h: ``Integer`` * * Sets the size of the popup, taking into account the size of the anchor. */ setSize: function(w, h) { if(this.anc) { var ancSize = this.anc.getSize(); if(typeof w == 'object') { h = w.height - ancSize.height; w = w.width; } else if(!isNaN(h)){ h = h - ancSize.height; } } GeoExt.Popup.superclass.setSize.call(this, w, h); }, /** private: method[position] * Positions the popup relative to its location */ position: function() { var me = this; if(me._mapMove === true) { me.insideViewport = me.map.getExtent().containsLonLat(me.location); if(me.insideViewport !== me.isVisible()) { me.setVisible(me.insideViewport); } } if(me.isVisible()) { var locationPx = me.map.getPixelFromLonLat(me.location), mapBox = Ext.fly(me.map.div).getBox(true), y = locationPx.y + mapBox.y, x = locationPx.x + mapBox.x, elSize = me.el.getSize(), ancSize = me.anc.getSize(), ancPos = me.anchorPosition; if (ancPos.indexOf("right") > -1 || locationPx.x > mapBox.width / 2) { // right me.anc.addClass("right"); var ancRight = me.el.getX(true) + elSize.width - me.anc.getX(true) - ancSize.width; x -= elSize.width - ancRight - ancSize.width / 2; } else { // left me.anc.removeClass("right"); var ancX = me.anc.getLeft(true); x -= ancX + ancSize.width / 2; } if (ancPos.indexOf("bottom") > -1 || locationPx.y > mapBox.height / 2) { // bottom me.anc.removeClass("top"); y -= elSize.height + ancSize.height; } else { // top me.anc.addClass("top"); y += ancSize.height; // ok } // Needed to have the right position on the first display // (no flash on the center of the map). me.setPagePosition(x, y); // position in the next cycle - otherwise strange shifts can occur. window.setTimeout(function() { if (me.el.dom) { me.setPagePosition(x, y); } }, 0); } }, /** private: method[makeDraggable] * Make the window draggable */ makeDraggable: function() { this.draggable = true; this.header.addClass("x-window-draggable"); this.dd = new Ext.Window.DD(this); }, /** private: method[unanchorPopup] * Unanchors a popup from its location. This removes the popup from its * MapPanel and adds it to the page body. */ unanchorPopup: function() { this.removeAnchorEvents(); this.makeDraggable(); //remove anchor this.anc.remove(); this.anc = null; //hide unpin tool this.tools.unpin.hide(); }, /** private: method[panIntoView] * Pans the MapPanel's map so that an anchored popup can come entirely * into view, with padding specified as per normal OpenLayers.Map popup * padding. */ panIntoView: function() { var mapBox = Ext.fly(this.map.div).getBox(true); //assumed viewport takes up whole body element of map panel var popupPos = this.getPosition(true); popupPos[0] -= mapBox.x; popupPos[1] -= mapBox.y; var panelSize = [mapBox.width, mapBox.height]; // [X,Y] var popupSize = this.getSize(); var newPos = [popupPos[0], popupPos[1]]; //For now, using native OpenLayers popup padding. This may not be ideal. var padding = this.map.paddingForPopups; // X if(popupPos[0] < padding.left) { newPos[0] = padding.left; } else if(popupPos[0] + popupSize.width > panelSize[0] - padding.right) { newPos[0] = panelSize[0] - padding.right - popupSize.width; } // Y if(popupPos[1] < padding.top) { newPos[1] = padding.top; } else if(popupPos[1] + popupSize.height > panelSize[1] - padding.bottom) { newPos[1] = panelSize[1] - padding.bottom - popupSize.height; } var dx = popupPos[0] - newPos[0]; var dy = popupPos[1] - newPos[1]; this.map.pan(dx, dy); }, /** private: method[onMapMove] */ onMapMove: function() { if (!(this.hidden && this.insideViewport)){ this._mapMove = true; this.position(); delete this._mapMove; } }, /** private: method[addAnchorEvents] */ addAnchorEvents: function() { this.map.events.on({ "move" : this.onMapMove, scope : this }); this.on({ "resize": this.position, "collapse": this.position, "expand": this.position, scope: this }); }, /** private: method[removeAnchorEvents] */ removeAnchorEvents: function() { //stop position with location this.map.events.un({ "move" : this.onMapMove, scope : this }); this.un("resize", this.position, this); this.un("collapse", this.position, this); this.un("expand", this.position, this); }, /** private: method[beforeDestroy] * Cleanup events before destroying the popup. */ beforeDestroy: function() { if(this.anchored) { this.removeAnchorEvents(); } GeoExt.Popup.superclass.beforeDestroy.call(this); } }); /** api: xtype = gx_popup */ Ext.reg('gx_popup', GeoExt.Popup); /** FILE: GeoExt/widgets/MapPanel.js **/ /** * Copyright (c) 2008-2012 The Open Source Geospatial Foundation * * Published under the BSD license. * See http://svn.geoext.org/core/trunk/geoext/license.txt for the full text * of the license. */ /** * @include GeoExt/data/LayerStore.js * @require OpenLayers/Map.js * @require OpenLayers/BaseTypes/LonLat.js * @require OpenLayers/BaseTypes/Bounds.js */ /** api: (define) * module = GeoExt * class = MapPanel * base_link = `Ext.Panel `_ */ Ext.namespace("GeoExt"); /** api: example * Sample code to create a panel with a new map: * * .. code-block:: javascript * * var mapPanel = new GeoExt.MapPanel({ * border: false, * renderTo: "div-id", * map: { * maxExtent: new OpenLayers.Bounds(-90, -45, 90, 45) * } * }); * * Sample code to create a map panel with a bottom toolbar in a Window: * * .. code-block:: javascript * * var win = new Ext.Window({ * title: "My Map", * items: [{ * xtype: "gx_mappanel", * bbar: new Ext.Toolbar() * }] * }); */ /** api: constructor * .. class:: MapPanel(config) * * Create a panel container for a map. The map contained by this panel * will initially be zoomed to either the center and zoom level configured * by the ``center`` and ``zoom`` configuration options, or the configured * ``extent``, or - if neither are provided - the extent returned by the * map's ``getExtent()`` method. */ GeoExt.MapPanel = Ext.extend(Ext.Panel, { /** api: config[map] * ``OpenLayers.Map or Object`` A configured map or a configuration object * for the map constructor. A configured map will be available after * construction through the :attr:`map` property. */ /** api: property[map] * ``OpenLayers.Map`` or ``Object`` A map or map configuration. */ map: null, /** api: config[layers] * ``GeoExt.data.LayerStore or GeoExt.data.GroupingStore or Array(OpenLayers.Layer)`` * A store holding records. The layers provided here will be added to this * MapPanel's map when it is rendered. */ /** api: property[layers] * :class:`GeoExt.data.LayerStore` A store containing * :class:`GeoExt.data.LayerRecord` objects. */ layers: null, /** api: config[center] * ``OpenLayers.LonLat or Array(Number)`` A location for the initial map * center. If an array is provided, the first two items should represent * x & y coordinates. */ center: null, /** api: config[zoom] * ``Number`` An initial zoom level for the map. */ zoom: null, /** api: config[extent] * ``OpenLayers.Bounds or Array(Number)`` An initial extent for the map (used * if center and zoom are not provided. If an array, the first four items * should be minx, miny, maxx, maxy. */ extent: null, /** api: config[prettyStateKeys] * ``Boolean`` Set this to true if you want pretty strings in the MapPanel's * state keys. More specifically, layer.name instead of layer.id will be used * in the state keys if this option is set to true. But in that case you have * to make sure you don't have two layers with the same name. Defaults to * false. */ prettyStateKeys: false, /** private: property[stateEvents] * ``Array(String)`` Array of state events */ stateEvents: ["aftermapmove", "afterlayervisibilitychange", "afterlayeropacitychange", "afterlayerorderchange", "afterlayernamechange", "afterlayeradd", "afterlayerremove"], /** private: method[initComponent] * Initializes the map panel. Creates an OpenLayers map if * none was provided in the config options passed to the * constructor. */ initComponent: function(){ if(!(this.map instanceof OpenLayers.Map)) { this.map = new OpenLayers.Map( Ext.applyIf(this.map || {}, {allOverlays: true, fallThrough: true}) ); } var layers = this.layers; if(!layers || layers instanceof Array) { this.layers = new GeoExt.data.LayerStore({ layers: layers, map: this.map.layers.length > 0 ? this.map : null }); } if(typeof this.center == "string") { this.center = OpenLayers.LonLat.fromString(this.center); } else if(this.center instanceof Array) { this.center = new OpenLayers.LonLat(this.center[0], this.center[1]); } if(typeof this.extent == "string") { this.extent = OpenLayers.Bounds.fromString(this.extent); } else if(this.extent instanceof Array) { this.extent = OpenLayers.Bounds.fromArray(this.extent); } GeoExt.MapPanel.superclass.initComponent.call(this); this.addEvents( /** private: event[aftermapmove] * Fires after the map is moved. */ "aftermapmove", /** private: event[afterlayervisibilitychange] * Fires after a layer changed visibility. */ "afterlayervisibilitychange", /** private: event[afterlayeropacitychange] * Fires after a layer changed opacity. */ "afterlayeropacitychange", /** private: event[afterlayerorderchange] * Fires after a layer order changed. */ "afterlayerorderchange", /** private: event[afterlayernamechange] * Fires after a layer name changed. */ "afterlayernamechange", /** private: event[afterlayeradd] * Fires after a layer added to the map. */ "afterlayeradd", /** private: event[afterlayerremove] * Fires after a layer removed from the map. */ "afterlayerremove" ); this.map.events.on({ "moveend": this.onMoveend, "changelayer": this.onChangelayer, "addlayer": this.onAddlayer, "removelayer": this.onRemovelayer, scope: this }); //TODO This should be handled by a LayoutManager this.on("afterlayout", function() { //TODO remove function check when we require OpenLayers > 2.11 if (typeof this.map.getViewport === "function") { this.items.each(function(cmp) { if (typeof cmp.addToMapPanel === "function") { cmp.getEl().appendTo(this.map.getViewport()); } }, this); } }, this); }, /** private: method[onMoveend] * * The "moveend" listener. */ onMoveend: function() { this.fireEvent("aftermapmove"); }, /** private: method[onChangelayer] * :param e: ``Object`` * * The "changelayer" listener. */ onChangelayer: function(e) { if(e.property) { if(e.property === "visibility") { this.fireEvent("afterlayervisibilitychange"); } else if(e.property === "order") { this.fireEvent("afterlayerorderchange"); } else if(e.property === "name") { this.fireEvent("afterlayernamechange"); } else if(e.property === "opacity") { this.fireEvent("afterlayeropacitychange"); } } }, /** private: method[onAddlayer] */ onAddlayer: function() { this.fireEvent("afterlayeradd"); }, /** private: method[onRemovelayer] */ onRemovelayer: function() { this.fireEvent("afterlayerremove"); }, /** private: method[applyState] * :param state: ``Object`` The state to apply. * * Apply the state provided as an argument. */ applyState: function(state) { // if we get strings for state.x, state.y or state.zoom // OpenLayers will take care of converting them to the // appropriate types so we don't bother with that this.center = new OpenLayers.LonLat(state.x, state.y); this.zoom = state.zoom; // set layer visibility and opacity var i, l, layer, layerId, visibility, opacity; var layers = this.map.layers; for(i=0, l=layers.length; i 0) { this.setInitialExtent(); } else { this.layers.on("add", this.setInitialExtent, this, {single: true}); } }, /** private: method[setInitialExtent] * Sets the initial extent of this panel's map */ setInitialExtent: function() { var map = this.map; if(this.center || this.zoom != null) { // both do not have to be defined map.setCenter(this.center, this.zoom); } else if(this.extent) { map.zoomToExtent(this.extent); } else { map.zoomToMaxExtent(); } }, /** private: method[afterRender] * Private method called after the panel has been rendered. */ afterRender: function() { GeoExt.MapPanel.superclass.afterRender.apply(this, arguments); if(!this.ownerCt) { this.renderMap(); } else { this.ownerCt.on("move", this.updateMapSize, this); this.ownerCt.on({ "afterlayout": this.afterLayout, scope: this }); } }, /** private: method[afterLayout] * Private method called after owner container has been laid out until * this panel has dimensions greater than zero. */ afterLayout: function() { var width = this.getInnerWidth() - this.body.getBorderWidth("lr"); var height = this.getInnerHeight() - this.body.getBorderWidth("tb"); if (width > 0 && height > 0) { this.ownerCt.un("afterlayout", this.afterLayout, this); this.renderMap(); } }, /** private: method[onResize] * Private method called after the panel has been resized. */ onResize: function() { GeoExt.MapPanel.superclass.onResize.apply(this, arguments); this.updateMapSize(); }, /** private: method[onBeforeAdd] * Private method called before a component is added to the panel. */ onBeforeAdd: function(item) { if(typeof item.addToMapPanel === "function") { item.addToMapPanel(this); } GeoExt.MapPanel.superclass.onBeforeAdd.apply(this, arguments); }, /** private: method[remove] * Private method called when a component is removed from the panel. */ remove: function(item, autoDestroy) { if(typeof item.removeFromMapPanel === "function") { item.removeFromMapPanel(this); } GeoExt.MapPanel.superclass.remove.apply(this, arguments); }, /** private: method[beforeDestroy] * Private method called during the destroy sequence. */ beforeDestroy: function() { if(this.ownerCt) { this.ownerCt.un("move", this.updateMapSize, this); } if(this.map && this.map.events) { this.map.events.un({ "moveend": this.onMoveend, "changelayer": this.onChangelayer, "addlayer": this.onAddlayer, "removelayer": this.onRemovelayer, scope: this }); } // if the map panel was passed a map instance, this map instance // is under the user's responsibility if(!this.initialConfig.map || !(this.initialConfig.map instanceof OpenLayers.Map)) { // we created the map, we destroy it if(this.map && this.map.destroy) { this.map.destroy(); } } delete this.map; GeoExt.MapPanel.superclass.beforeDestroy.apply(this, arguments); } }); /** api: function[guess] * :return: ``GeoExt.MapPanel`` The first map panel found by the Ext * component manager. * * Convenience function for guessing the map panel of an application. This * can reliably be used for all applications that just have one map panel * in the viewport. */ GeoExt.MapPanel.guess = function() { return Ext.ComponentMgr.all.find(function(o) { return o instanceof GeoExt.MapPanel; }); }; /** api: xtype = gx_mappanel */ Ext.reg('gx_mappanel', GeoExt.MapPanel); /** FILE: GeoExt/data/LayerStore.js **/ /** * Copyright (c) 2008-2012 The Open Source Geospatial Foundation * * Published under the BSD license. * See http://svn.geoext.org/core/trunk/geoext/license.txt for the full text * of the license. */ /** * @include GeoExt/data/LayerReader.js * @include GeoExt/widgets/MapPanel.js */ /** api: (define) * module = GeoExt.data * class = LayerStore * base_link = `Ext.data.Store `_ */ Ext.namespace("GeoExt.data"); /** private: constructor * .. class:: LayerStoreMixin * A store that synchronizes a layers array of an {OpenLayers.Map} with a * layer store holding {} entries. * * This class can not be instantiated directly. Instead, it is meant to * extend ``Ext.data.Store`` or a subclass of it. */ /** private: example * Sample code to extend a store with the LayerStoreMixin. * * .. code-block:: javascript * * var store = new (Ext.extend(Ext.data.Store, new GeoExt.data.LayerStoreMixin))({ * map: myMap, * layers: myLayers * }); * * For convenience, a :class:`GeoExt.data.LayerStore` class is available as a * shortcut to the ``Ext.extend`` sequence in the above code snippet. */ GeoExt.data.LayerStoreMixin = function() { return { /** api: config[map] * ``OpenLayers.Map`` * Map that this store will be in sync with. If not provided, the * store will not be bound to a map. */ /** api: property[map] * ``OpenLayers.Map`` * Map that the store is synchronized with, if any. */ map: null, /** api: config[layers] * ``Array(OpenLayers.Layer)`` * Layers that will be added to the store (and the map, depending on the * value of the ``initDir`` option. */ /** api: config[initDir] * ``Number`` * Bitfields specifying the direction to use for the initial sync between * the map and the store, if set to 0 then no initial sync is done. * Defaults to ``GeoExt.data.LayerStore.MAP_TO_STORE|GeoExt.data.LayerStore.STORE_TO_MAP`` */ /** api: config[fields] * ``Array`` * If provided a custom layer record type with additional fields will be * used. Default fields for every layer record are `layer` * (``OpenLayers.Layer``) `title` (``String``). The value of this option is * either a field definition objects as passed to the * :meth:`GeoExt.data.LayerRecord.create` function or a * :class:`GeoExt.data.LayerRecord` constructor created using * :meth:`GeoExt.data.LayerRecord.create`. */ /** api: config[reader] * ``Ext.data.DataReader`` The reader used to produce * :class:`GeoExt.data.LayerRecord` objects from ``OpenLayers.Layer`` * objects. If not provided, a :class:`GeoExt.data.LayerReader` will be * used. */ reader: null, /** private: method[constructor] */ constructor: function(config) { config = config || {}; config.reader = config.reader || new GeoExt.data.LayerReader({}, config.fields); delete config.fields; // "map" option var map = config.map instanceof GeoExt.MapPanel ? config.map.map : config.map; delete config.map; // "layers" option - is an alias to "data" option if(config.layers) { config.data = config.layers; } delete config.layers; // "initDir" option var options = {initDir: config.initDir}; delete config.initDir; arguments.callee.superclass.constructor.call(this, config); this.addEvents( /** api:event[bind] * Fires when the store is bound to a map. * * Listener arguments: * * :class:`GeoExt.data.LayerStore` * * ``OpenLayers.Map`` */ "bind" ); if(map) { this.bind(map, options); } }, /** api: method[bind] * :param map: ``OpenLayers.Map`` The map instance. * :param options: ``Object`` * * Bind this store to a map instance, once bound the store * is synchronized with the map and vice-versa. */ bind: function(map, options) { if(this.map) { // already bound return; } this.map = map; options = options || {}; var initDir = options.initDir; if(options.initDir == undefined) { initDir = GeoExt.data.LayerStore.MAP_TO_STORE | GeoExt.data.LayerStore.STORE_TO_MAP; } // create a snapshot of the map's layers var layers = map.layers.slice(0); if(initDir & GeoExt.data.LayerStore.STORE_TO_MAP) { this.each(function(record) { this.map.addLayer(record.getLayer()); }, this); } if(initDir & GeoExt.data.LayerStore.MAP_TO_STORE) { this.loadData(layers, true); } map.events.on({ "changelayer": this.onChangeLayer, "addlayer": this.onAddLayer, "removelayer": this.onRemoveLayer, scope: this }); this.on({ "load": this.onLoad, "clear": this.onClear, "add": this.onAdd, "remove": this.onRemove, "update": this.onUpdate, scope: this }); this.data.on({ "replace" : this.onReplace, scope: this }); this.fireEvent("bind", this, map); }, /** api: method[unbind] * Unbind this store from the map it is currently bound. */ unbind: function() { if(this.map) { this.map.events.un({ "changelayer": this.onChangeLayer, "addlayer": this.onAddLayer, "removelayer": this.onRemoveLayer, scope: this }); this.un("load", this.onLoad, this); this.un("clear", this.onClear, this); this.un("add", this.onAdd, this); this.un("remove", this.onRemove, this); this.data.un("replace", this.onReplace, this); this.map = null; } }, /** private: method[onChangeLayer] * :param evt: ``Object`` * * Handler for layer changes. When layer order changes, this moves the * appropriate record within the store. */ onChangeLayer: function(evt) { var layer = evt.layer; var recordIndex = this.findBy(function(rec, id) { return rec.getLayer() === layer; }); if(recordIndex > -1) { var record = this.getAt(recordIndex); if(evt.property === "order") { if(!this._adding && !this._removing) { var layerIndex = this.map.getLayerIndex(layer); if(layerIndex !== recordIndex) { this._removing = true; this.remove(record); delete this._removing; this._adding = true; this.insert(layerIndex, [record]); delete this._adding; } } } else if(evt.property === "name") { record.set("title", layer.name); } else { this.fireEvent("update", this, record, Ext.data.Record.EDIT); } } }, /** private: method[onAddLayer] * :param evt: ``Object`` * * Handler for a map's addlayer event */ onAddLayer: function(evt) { if(!this._adding) { var layer = evt.layer; this._adding = true; this.loadData([layer], true); delete this._adding; } }, /** private: method[onRemoveLayer] * :param evt: ``Object`` * * Handler for a map's removelayer event */ onRemoveLayer: function(evt){ //TODO replace the check for undloadDestroy with a listener for the // map's beforedestroy event, doing unbind(). This can be done as soon // as http://trac.openlayers.org/ticket/2136 is fixed. if(this.map.unloadDestroy) { if(!this._removing) { var layer = evt.layer; this._removing = true; this.remove(this.getById(layer.id)); delete this._removing; } } else { this.unbind(); } }, /** private: method[onLoad] * :param store: ``Ext.data.Store`` * :param records: ``Array(Ext.data.Record)`` * :param options: ``Object`` * * Handler for a store's load event */ onLoad: function(store, records, options) { if (!Ext.isArray(records)) { records = [records]; } if (options && !options.add) { this._removing = true; for (var i = this.map.layers.length - 1; i >= 0; i--) { this.map.removeLayer(this.map.layers[i]); } delete this._removing; // layers has already been added to map on "add" event var len = records.length; if (len > 0) { var layers = new Array(len); for (var j = 0; j < len; j++) { layers[j] = records[j].getLayer(); } this._adding = true; this.map.addLayers(layers); delete this._adding; } } }, /** private: method[onClear] * :param store: ``Ext.data.Store`` * * Handler for a store's clear event */ onClear: function(store) { this._removing = true; for (var i = this.map.layers.length - 1; i >= 0; i--) { this.map.removeLayer(this.map.layers[i]); } delete this._removing; }, /** private: method[onAdd] * :param store: ``Ext.data.Store`` * :param records: ``Array(Ext.data.Record)`` * :param index: ``Number`` * * Handler for a store's add event */ onAdd: function(store, records, index) { if(!this._adding) { this._adding = true; var layer; for(var i=records.length-1; i>=0; --i) { layer = records[i].getLayer(); this.map.addLayer(layer); if(index !== this.map.layers.length-1) { this.map.setLayerIndex(layer, index); } } delete this._adding; } }, /** private: method[onRemove] * :param store: ``Ext.data.Store`` * :param record: ``Ext.data.Record`` * :param index: ``Number`` * * Handler for a store's remove event */ onRemove: function(store, record, index){ if(!this._removing) { var layer = record.getLayer(); if (this.map.getLayer(layer.id) != null) { this._removing = true; this.removeMapLayer(record); delete this._removing; } } }, /** private: method[onUpdate] * :param store: ``Ext.data.Store`` * :param record: ``Ext.data.Record`` * :param operation: ``Number`` * * Handler for a store's update event */ onUpdate: function(store, record, operation) { if(operation === Ext.data.Record.EDIT) { if (record.modified && record.modified.title) { var layer = record.getLayer(); var title = record.get("title"); if(title !== layer.name) { layer.setName(title); } } } }, /** private: method[removeMapLayer] * :param record: ``Ext.data.Record`` * * Removes a record's layer from the bound map. */ removeMapLayer: function(record){ this.map.removeLayer(record.getLayer()); }, /** private: method[onReplace] * :param key: ``String`` * :param oldRecord: ``Object`` In this case, a record that has been * replaced. * :param newRecord: ``Object`` In this case, a record that is replacing * oldRecord. * Handler for a store's data collections' replace event */ onReplace: function(key, oldRecord, newRecord){ this.removeMapLayer(oldRecord); }, /** api: method[getByLayer] * :param layer: ``OpenLayers.Layer`` * :return: :class:`GeoExt.data.LayerRecord` or undefined if not found * * Get the record for the specified layer */ getByLayer: function(layer) { var index = this.findBy(function(r) { return r.getLayer() === layer; }); if(index > -1) { return this.getAt(index); } }, /** private: method[destroy] */ destroy: function() { this.unbind(); GeoExt.data.LayerStore.superclass.destroy.call(this); } }; }; /** api: example * Sample to create a new store containing a cache of * :class:`GeoExt.data.LayerRecord` instances derived from map layers. * * .. code-block:: javascript * * var store = new GeoExt.data.LayerStore({ * map: myMap, * layers: myLayers * }); */ /** api: constructor * .. class:: LayerStore * * A store that contains a cache of :class:`GeoExt.data.LayerRecord` * objects. */ GeoExt.data.LayerStore = Ext.extend( Ext.data.Store, new GeoExt.data.LayerStoreMixin ); /** * Constant: GeoExt.data.LayerStore.MAP_TO_STORE * {Integer} Constant used to make the store be automatically updated * when changes occur in the map. */ GeoExt.data.LayerStore.MAP_TO_STORE = 1; /** * Constant: GeoExt.data.LayerStore.STORE_TO_MAP * {Integer} Constant used to make the map be automatically updated * when changes occur in the store. */ GeoExt.data.LayerStore.STORE_TO_MAP = 2; /** FILE: GeoExt/data/LayerReader.js **/ /** * Copyright (c) 2008-2012 The Open Source Geospatial Foundation * * Published under the BSD license. * See http://svn.geoext.org/core/trunk/geoext/license.txt for the full text * of the license. */ /** * @include GeoExt/data/LayerRecord.js */ /** api: (define) * module = GeoExt.data * class = LayerReader * base_link = `Ext.data.DataReader `_ */ Ext.namespace("GeoExt", "GeoExt.data"); /** api: example * Sample using a reader to create records from an array of layers: * * .. code-block:: javascript * * var reader = new GeoExt.data.LayerReader(); * var layerData = reader.readRecords(map.layers); * var numRecords = layerData.totalRecords; * var layerRecords = layerData.records; */ /** api: constructor * .. class:: LayerReader(meta, recordType) * * Data reader class to create an array of * :class:`GeoExt.data.LayerRecord` objects from an array of * ``OpenLayers.Layer`` objects for use in a * :class:`GeoExt.data.LayerStore` object. */ GeoExt.data.LayerReader = function(meta, recordType) { meta = meta || {}; if(!(recordType instanceof Function)) { recordType = GeoExt.data.LayerRecord.create( recordType || meta.fields || {}); } GeoExt.data.LayerReader.superclass.constructor.call( this, meta, recordType); }; Ext.extend(GeoExt.data.LayerReader, Ext.data.DataReader, { /** private: property[totalRecords] * ``Integer`` */ totalRecords: null, /** api: method[readRecords] * :param layers: ``Array(OpenLayers.Layer)`` List of layers for creating * records. * :return: ``Object`` An object with ``records`` and ``totalRecords`` * properties. * * From an array of ``OpenLayers.Layer`` objects create a data block * containing :class:`GeoExt.data.LayerRecord` objects. */ readRecords : function(layers) { var records = []; if(layers) { var recordType = this.recordType, fields = recordType.prototype.fields; var i, lenI, j, lenJ, layer, values, field, v; for(i = 0, lenI = layers.length; i < lenI; i++) { layer = layers[i]; values = {}; for(j = 0, lenJ = fields.length; j < lenJ; j++){ field = fields.items[j]; v = layer[field.mapping || field.name] || field.defaultValue; v = field.convert(v); values[field.name] = v; } values.layer = layer; records[records.length] = new recordType(values, layer.id); } } return { records: records, totalRecords: this.totalRecords != null ? this.totalRecords : records.length }; } }); /** FILE: GeoExt/data/ScaleStore.js **/ /** * Copyright (c) 2008-2012 The Open Source Geospatial Foundation * * Published under the BSD license. * See http://svn.geoext.org/core/trunk/geoext/license.txt for the full text * of the license. */ /** * @include GeoExt/widgets/MapPanel.js * @require OpenLayers/Util.js */ /** api: (define) * module = GeoExt.data * class = ScaleStore * base_link = `Ext.data.Store `_ */ Ext.namespace("GeoExt.data"); /** api: constructor * .. class:: ScaleStore * * A store that contains a cache of available zoom levels. The store can * optionally be kept synchronized with an ``OpenLayers.Map`` or * :class:`GeoExt.MapPanel` object. * * Records have the following fields: * * * level - ``Number`` The zoom level. * * scale - ``Number`` The scale denominator. * * resolution - ``Number`` The map units per pixel. */ GeoExt.data.ScaleStore = Ext.extend(Ext.data.Store, { /** api: config[map] * ``OpenLayers.Map`` or :class:`GeoExt.MapPanel` * Optional map or map panel from which to derive scale values. */ map: null, /** private: method[constructor] * Construct a ScaleStore from a configuration. The ScaleStore accepts * some custom parameters addition to the fields accepted by Ext.Store. */ constructor: function(config) { var map = (config.map instanceof GeoExt.MapPanel ? config.map.map : config.map); delete config.map; config = Ext.applyIf(config, {reader: new Ext.data.JsonReader({}, [ "level", "resolution", "scale" ])}); GeoExt.data.ScaleStore.superclass.constructor.call(this, config); if (map) { this.bind(map); } }, /** api: method[bind] * :param map: :class:`GeoExt.MapPanel` or ``OpenLayers.Map`` Panel or map * to which we should bind. * * Bind this store to a map; that is, maintain the zoom list in sync with * the map's current configuration. If the map does not currently have a * set scale list, then the store will remain empty until the map is * configured with one. */ bind: function(map, options) { this.map = (map instanceof GeoExt.MapPanel ? map.map : map); this.map.events.register('changebaselayer', this, this.populateFromMap); if (this.map.baseLayer) { this.populateFromMap(); } else { this.map.events.register('addlayer', this, this.populateOnAdd); } }, /** api: method[unbind] * Un-bind this store from the map to which it is currently bound. The * currently stored zoom levels will remain, but no further changes from * the map will affect it. */ unbind: function() { if (this.map) { this.map.events.unregister('addlayer', this, this.populateOnAdd); this.map.events.unregister('changebaselayer', this, this.populateFromMap); delete this.map; } }, /** private: method[populateOnAdd] * :param evt: ``Object`` * * This method handles the case where we have bind() called on a * not-fully-configured map so that the zoom levels can be detected when a * baselayer is finally added. */ populateOnAdd: function(evt) { if (evt.layer.isBaseLayer) { this.populateFromMap(); this.map.events.unregister('addlayer', this, this.populateOnAdd); } }, /** private: method[populateFromMap] * This method actually loads the zoom level information from the * OpenLayers.Map and converts it to Ext Records. */ populateFromMap: function() { var zooms = []; var resolutions = this.map.baseLayer.resolutions; var units = this.map.baseLayer.units; for (var i=resolutions.length-1; i >= 0; i--) { var res = resolutions[i]; zooms.push({ level: i, resolution: res, scale: OpenLayers.Util.getScaleFromResolution(res, units) }); } this.loadData(zooms); }, /** private: method[destroy] */ destroy: function() { this.unbind(); GeoExt.data.ScaleStore.superclass.destroy.apply(this, arguments); } }); /** FILE: GeoExt/widgets/form.js **/ /** * Copyright (c) 2008-2012 The Open Source Geospatial Foundation * * Published under the BSD license. * See http://svn.geoext.org/core/trunk/geoext/license.txt for the full text * of the license. */ /** * @require OpenLayers/Filter/Comparison.js * @require OpenLayers/Filter/Logical.js */ Ext.namespace("GeoExt.form"); /** private: function[toFilter] * :param form: ``Ext.form.BasicForm|Ext.form.FormPanel`` * :param logicalOp: ``String`` Either ``OpenLayers.Filter.Logical.AND`` or * ``OpenLayers.Filter.Logical.OR``, set to * ``OpenLayers.Filter.Logical.AND`` if null or undefined * :param wildcard: ``Integer`` Determines the wildcard behaviour of like * queries. This behaviour can either be: none, prepend, append or both. * * :return: ``OpenLayers.Filter`` * * Create an {OpenLayers.Filter} object from a {Ext.form.BasicForm} * or a {Ext.form.FormPanel} instance. */ GeoExt.form.toFilter = function(form, logicalOp, wildcard) { if(form instanceof Ext.form.FormPanel) { form = form.getForm(); } var filters = [], values = form.getValues(false); for(var prop in values) { var s = prop.split("__"); var value = values[prop], type; if(s.length > 1 && (type = GeoExt.form.toFilter.FILTER_MAP[s[1]]) !== undefined) { prop = s[0]; } else { type = OpenLayers.Filter.Comparison.EQUAL_TO; } if (type === OpenLayers.Filter.Comparison.LIKE) { switch(wildcard) { case GeoExt.form.ENDS_WITH: value = '.*' + value; break; case GeoExt.form.STARTS_WITH: value += '.*'; break; case GeoExt.form.CONTAINS: value = '.*' + value + '.*'; break; default: // do nothing, just take the value break; } } filters.push( new OpenLayers.Filter.Comparison({ type: type, value: value, property: prop }) ); } return filters.length == 1 && logicalOp != OpenLayers.Filter.Logical.NOT ? filters[0] : new OpenLayers.Filter.Logical({ type: logicalOp || OpenLayers.Filter.Logical.AND, filters: filters }); }; /** private: constant[FILTER_MAP] * An object mapping operator strings as found in field names to * ``OpenLayers.Filter.Comparison`` types. */ GeoExt.form.toFilter.FILTER_MAP = { "eq": OpenLayers.Filter.Comparison.EQUAL_TO, "ne": OpenLayers.Filter.Comparison.NOT_EQUAL_TO, "lt": OpenLayers.Filter.Comparison.LESS_THAN, "le": OpenLayers.Filter.Comparison.LESS_THAN_OR_EQUAL_TO, "gt": OpenLayers.Filter.Comparison.GREATER_THAN, "ge": OpenLayers.Filter.Comparison.GREATER_THAN_OR_EQUAL_TO, "like": OpenLayers.Filter.Comparison.LIKE }; GeoExt.form.ENDS_WITH = 1; GeoExt.form.STARTS_WITH = 2; GeoExt.form.CONTAINS = 3; /** private: function[recordToField] * :param record: ``Ext.data.Record``, typically from an attributeStore * :param options: ``Object``, optional object litteral. Valid options: * * * checkboxLabelProperty - ``String`` The name of the property used to set * the label in the checkbox. Only applies if the record is of the "boolean" * type. Possible values are "boxLabel" and "fieldLabel". Default is "boxLabel". * * mandatoryFieldLabelStyle - ``String`` A CSS style specification string * to apply to the field label if the field is not nillable (that is, * the corresponding record has the "nillable" attribute set to ``false``). * Default is ``"font-weigth: bold;"``. * * labelTpl - ``Ext.Template`` or ``String`` or ``Array`` If set, * the field label is obtained by applying the record's data hash to this * template. This allows for very customizable field labels. * See for instance : * * .. code-block:: javascript * * var formPanel = new Ext.form.FormPanel({ * autoScroll: true, * plugins: [ * new GeoExt.plugins.AttributeForm({ * attributeStore: store, * recordToFieldOptions: { * mandatoryFieldLabelStyle: 'font-style:italic;', * labelTpl: new Ext.XTemplate( * '{name}', { * compiled: true, * disableFormats: true, * getTip: function(v) { * if (!v.type) { * return ''; * } * var type = v.type.split(":").pop(); * return OpenLayers.i18n(type) + * (v.nillable ? '' : ' (required)'); * } * } * ) * } * }) * ] * }); * * :return: ``Object`` An object literal with a xtype property, use * ``Ext.ComponentMgr.create`` (or ``Ext.create`` in Ext 3) to create * an ``Ext.form.Field`` from this object. * * This function can be used to create an ``Ext.form.Field`` from * an ``Ext.data.Record`` containing name, type, restriction and * label fields. */ GeoExt.form.recordToField = function(record, options) { options = options || {}; var type = record.get("type"); if(typeof type === "object" && type.xtype) { // we have an xtype'd object literal in the type // field, just return it return type; } type = type.split(":").pop(); // remove ns prefix var field; var name = record.get("name"); var restriction = record.get("restriction") || {}; var nillable = record.get("nillable") || false; var label = record.get("label"); var labelTpl = options.labelTpl; if (labelTpl) { var tpl = (labelTpl instanceof Ext.Template) ? labelTpl : new Ext.XTemplate(labelTpl); label = tpl.apply(record.data); } else if (label == null) { // use name for label if label isn't defined in the record label = name; } var baseOptions = { name: name, labelStyle: nillable ? '' : options.mandatoryFieldLabelStyle != null ? options.mandatoryFieldLabelStyle : 'font-weight:bold;' }; var r = GeoExt.form.recordToField.REGEXES; if (restriction.enumeration) { field = Ext.apply({ xtype: "combo", fieldLabel: label, mode: "local", forceSelection: true, triggerAction: "all", editable: false, store: restriction.enumeration }, baseOptions); } else if(type.match(r["text"])) { var maxLength = restriction["maxLength"] !== undefined ? parseFloat(restriction["maxLength"]) : undefined; var minLength = restriction["minLength"] !== undefined ? parseFloat(restriction["minLength"]) : undefined; field = Ext.apply({ xtype: "textfield", fieldLabel: label, maxLength: maxLength, minLength: minLength }, baseOptions); } else if(type.match(r["number"])) { var maxValue = restriction["maxInclusive"] !== undefined ? parseFloat(restriction["maxInclusive"]) : undefined; var minValue = restriction["minInclusive"] !== undefined ? parseFloat(restriction["minInclusive"]) : undefined; field = Ext.apply({ xtype: "numberfield", fieldLabel: label, maxValue: maxValue, minValue: minValue }, baseOptions); } else if(type.match(r["boolean"])) { field = Ext.apply({ xtype: "checkbox" }, baseOptions); var labelProperty = options.checkboxLabelProperty || "boxLabel"; field[labelProperty] = label; } else if(type.match(r["date"])) { field = Ext.apply({ xtype: "datefield", fieldLabel: label, format: 'c' }, baseOptions); } return field; }; /** private: constant[REGEXES] * ``Object`` Regular expressions for determining what type * of field to create from an attribute record. */ GeoExt.form.recordToField.REGEXES = { "text": new RegExp( "^(text|string)$", "i" ), "number": new RegExp( "^(number|float|decimal|double|int|long|integer|short)$", "i" ), "boolean": new RegExp( "^(boolean)$", "i" ), "date": new RegExp( "^(date|dateTime)$", "i" ) }; /** FILE: GeoExt/widgets/LegendPanel.js **/ /** * Copyright (c) 2008-2012 The Open Source Geospatial Foundation * * Published under the BSD license. * See http://svn.geoext.org/core/trunk/geoext/license.txt for the full text * of the license. */ /** * @include GeoExt/widgets/MapPanel.js * @include GeoExt/widgets/LayerLegend.js */ /** api: (define) * module = GeoExt * class = LegendPanel * base_link = `Ext.Panel `_ */ Ext.namespace('GeoExt'); /** api: constructor * .. class:: LegendPanel(config) * * A panel showing legends of all layers in a layer store. * Depending on the layer type, a legend renderer will be chosen. * * The LegendPanel will include legends for all the layers in the * ``layerStore`` it is configured with, unless the layer is configured with * ``displayInLayerSwitcher: false``, or a layer record has a * ``hideInLegend`` field with a value of ``true``. Additional filtering can * be done by configuring a ``filter`` on the LegendPanel. */ GeoExt.LegendPanel = Ext.extend(Ext.Panel, { /** api: config[dynamic] * ``Boolean`` * If false the LegendPanel will not listen to the add, remove and change * events of the LayerStore. So it will load with the initial state of * the LayerStore and not change anymore. */ dynamic: true, /** api: config[layerStore] * ``GeoExt.data.LayerStore`` * The layer store containing layers to be displayed in the legend * container. If not provided it will be taken from the MapPanel. */ layerStore: null, /** api: config[preferredTypes] * ``Array(String)`` An array of preferred legend types. */ /** private: property[preferredTypes] */ preferredTypes: null, /** api: config[filter] * ``Function`` * A function, called in the scope of the legend panel, with a layer record * as argument. Is expected to return true for layers to be displayed, false * otherwise. By default, all layers will be displayed. * * .. code-block:: javascript * * filter: function(record) { * return record.getLayer().isBaseLayer; * } */ filter: function(record) { return true; }, /** private: method[onRender] * Private method called when the legend panel is being rendered. */ onRender: function() { GeoExt.LegendPanel.superclass.onRender.apply(this, arguments); if(!this.layerStore) { this.layerStore = GeoExt.MapPanel.guess().layers; } this.layerStore.each(function(record) { this.addLegend(record); }, this); if (this.dynamic) { this.layerStore.on({ "add": this.onStoreAdd, "remove": this.onStoreRemove, "clear": this.onStoreClear, scope: this }); } }, /** private: method[recordIndexToPanelIndex] * Private method to get the panel index for a layer represented by a * record. * * :param index ``Integer`` The index of the record in the store. * * :return: ``Integer`` The index of the sub panel in this panel. */ recordIndexToPanelIndex: function(index) { var store = this.layerStore; var count = store.getCount(); var panelIndex = -1; var legendCount = this.items ? this.items.length : 0; var record, layer; for(var i=count-1; i>=0; --i) { record = store.getAt(i); layer = record.getLayer(); var types = GeoExt.LayerLegend.getTypes(record); if(layer.displayInLayerSwitcher && types.length > 0 && (store.getAt(i).get("hideInLegend") !== true)) { ++panelIndex; if(index === i || panelIndex > legendCount-1) { break; } } } return panelIndex; }, /** private: method[getIdForLayer] * :arg layer: ``OpenLayers.Layer`` * :returns: ``String`` * * Generate an element id that is unique to this panel/layer combo. */ getIdForLayer: function(layer) { return this.id + "-" + layer.id; }, /** private: method[onStoreAdd] * Private method called when a layer is added to the store. * * :param store: ``Ext.data.Store`` The store to which the record(s) was * added. * :param record: ``Ext.data.Record`` The record object(s) corresponding * to the added layers. * :param index: ``Integer`` The index of the inserted record. */ onStoreAdd: function(store, records, index) { var panelIndex = this.recordIndexToPanelIndex(index+records.length-1); for (var i=0, len=records.length; i 0) { this.insert(index, { xtype: types[0], id: this.getIdForLayer(layer), layerRecord: record, hidden: !((!layer.map && layer.visibility) || (layer.getVisibility() && layer.calculateInRange())) }); } } }, /** private: method[onDestroy] * Private method called during the destroy sequence. */ onDestroy: function() { if(this.layerStore) { this.layerStore.un("add", this.onStoreAdd, this); this.layerStore.un("remove", this.onStoreRemove, this); this.layerStore.un("clear", this.onStoreClear, this); } GeoExt.LegendPanel.superclass.onDestroy.apply(this, arguments); } }); /** api: xtype = gx_legendpanel */ Ext.reg('gx_legendpanel', GeoExt.LegendPanel); /** FILE: GeoExt/widgets/LayerLegend.js **/ /** * Copyright (c) 2008-2012 The Open Source Geospatial Foundation * * Published under the BSD license. * See http://svn.geoext.org/core/trunk/geoext/license.txt for the full text * of the license. */ /** api: (define) * module = GeoExt * class = LayerLegend * base_link = `Ext.Container `_ */ Ext.namespace('GeoExt'); /** api: constructor * .. class:: LayerLegend(config) * * Base class for components of :class:`GeoExt.LegendPanel`. */ GeoExt.LayerLegend = Ext.extend(Ext.Container, { /** api: config[layerRecord] * :class:`GeoExt.data.LayerRecord` The layer record for the legend */ layerRecord: null, /** api: config[showTitle] * ``Boolean`` * Whether or not to show the title of a layer. This can be overridden * on the LayerStore record using the hideTitle property. */ showTitle: true, /** api: config[legendTitle] * ``String`` * Optional title to be displayed instead of the layer title. If this is * set, the value of ``showTitle`` will be ignored (assumed to be true). */ legendTitle: null, /** api: config[labelCls] * ``String`` * Optional css class to use for the layer title labels. */ labelCls: null, /** private: property[layerStore] * :class:`GeoExt.data.LayerStore` */ layerStore: null, /** private: method[initComponent] */ initComponent: function() { GeoExt.LayerLegend.superclass.initComponent.call(this); this.autoEl = {}; this.add({ xtype: "label", html: this.getLayerTitle(this.layerRecord), cls: 'x-form-item x-form-item-label' + (this.labelCls ? ' ' + this.labelCls : '') }); if (this.layerRecord && this.layerRecord.store) { this.layerStore = this.layerRecord.store; this.layerStore.on("update", this.onStoreUpdate, this); this.layerStore.on("add", this.onStoreAdd, this); this.layerStore.on("remove", this.onStoreRemove, this); } }, /** private: method[getText] * :returns: ``String`` * * Get the label text of the legend. */ getLabel: function() { var label = this.items.get(0); return label.rendered ? label.el.dom.innerHTML : label.html; }, /** private: method[onStoreRemove] * Handler for remove event of the layerStore * * :param store: ``Ext.data.Store`` The store from which the record was * removed. * :param record: ``Ext.data.Record`` The record object corresponding * to the removed layer. * :param index: ``Integer`` The index in the store at which the record * was remvoed. */ onStoreRemove: function(store, record, index) { // to be implemented by subclasses if needed }, /** private: method[onStoreAdd] * Handler for add event of the layerStore * * :param store: ``Ext.data.Store`` The store to which the record was * added. * :param record: ``Ext.data.Record`` The record object corresponding * to the added layer. * :param index: ``Integer`` The index in the store at which the record * was added. */ onStoreAdd: function(store, record, index) { // to be implemented by subclasses if needed }, /** private: method[onStoreUpdate] * Update a the legend. Gets called when the store fires the update event. * This usually means the visibility of the layer, its style or title * has changed. * * :param store: ``Ext.data.Store`` The store in which the record was * changed. * :param record: ``Ext.data.Record`` The record object corresponding * to the updated layer. * :param operation: ``String`` The type of operation. */ onStoreUpdate: function(store, record, operation) { // if we don't have items, we are already awaiting garbage // collection after being removed by LegendPanel::removeLegend, and // updating will cause errors if (record === this.layerRecord && this.items.getCount() > 0) { var layer = record.getLayer(); this.setVisible(layer.getVisibility() && layer.calculateInRange() && layer.displayInLayerSwitcher && !record.get('hideInLegend')); this.update(); } }, /** private: method[update] * Updates the legend. */ update: function() { var title = this.getLayerTitle(this.layerRecord); var item = this.items.itemAt(0); if (item instanceof Ext.form.Label && this.getLabel() !== title) { // we need to update the title item.setText(title, false); } }, /** private: method[getLayerTitle] * :arg record: :class:GeoExt.data.LayerRecord * :returns: ``String`` * * Get a title for the layer. If the record doesn't have a title, use the * name. */ getLayerTitle: function(record) { var title = this.legendTitle || ""; if (this.showTitle && !title) { if (record && !record.get("hideTitle")) { title = record.get("title") || record.get("name") || record.getLayer().name || ""; } } return title; }, /** private: method[beforeDestroy] */ beforeDestroy: function() { if (this.layerStore) { this.layerStore.un("update", this.onStoreUpdate, this); this.layerStore.un("remove", this.onStoreRemove, this); this.layerStore.un("add", this.onStoreAdd, this); } GeoExt.LayerLegend.superclass.beforeDestroy.apply(this, arguments); }, /** private: method[onDestroy] */ onDestroy: function() { this.layerRecord = null; this.layerStore = null; GeoExt.LayerLegend.superclass.onDestroy.apply(this, arguments); } }); /** class: method[getTypes] * :param layerRecord: class:`GeoExt.data.LayerRecord` A layer record to get * legend types for. If not provided, all registered types will be * returned. * :param preferredTypes: ``Array(String)`` Types that should be considered. * first. If not provided, all registered legend types will be returned * in the order of their score for support of the provided layerRecord. * :return: ``Array(String)`` xtypes of legend types that can be used with * the provided ``layerRecord``. * * Gets an array of legend xtypes that support the provided layer record, * with optionally provided preferred types listed first. */ GeoExt.LayerLegend.getTypes = function(layerRecord, preferredTypes) { var types = (preferredTypes || []).concat(), scoredTypes = [], score, type; for (type in GeoExt.LayerLegend.types) { score = GeoExt.LayerLegend.types[type].supports(layerRecord); if(score > 0) { // add to scoredTypes if not preferred if (types.indexOf(type) == -1) { scoredTypes.push({ type: type, score: score }); } } else { // preferred, but not supported types.remove(type); } } scoredTypes.sort(function(a, b) { return a.score < b.score ? 1 : (a.score == b.score ? 0 : -1); }); var len = scoredTypes.length, goodTypes = new Array(len); for (var i=0; i`_ */ Ext.namespace('GeoExt'); /** api: constructor * .. class:: LegendImage(config) * * Show a legend image in a BoxComponent and make sure load errors are * dealt with. */ GeoExt.LegendImage = Ext.extend(Ext.BoxComponent, { /** api: config[url] * ``String`` The url of the image to load */ url: null, /** api: config[defaultImgSrc] * ``String`` Path to image that will be used if the legend image fails * to load. Default is Ext.BLANK_IMAGE_URL. */ defaultImgSrc: null, /** api: config[imgCls] * ``String`` Optional css class to apply to img tag */ imgCls: null, /** private: config[noImgCls] * ``String`` CSS class applied to img tag when no image is available or * the default image was loaded. */ noImgCls: "gx-legend-noimage", /** private: method[initComponent] * Initializes the legend image component. */ initComponent: function() { GeoExt.LegendImage.superclass.initComponent.call(this); if(this.defaultImgSrc === null) { this.defaultImgSrc = Ext.BLANK_IMAGE_URL; } this.autoEl = { tag: "img", "class": (this.imgCls ? this.imgCls + " " + this.noImgCls : this.noImgCls), src: this.defaultImgSrc }; }, /** api: method[getImgEl] * :return: ``Ext.Element`` The image element. * * Returns the image element. * This method is supposed to be overriden in subclasses. */ getImgEl: function() { return this.getEl(); }, /** api: method[setUrl] * :param url: ``String`` The new URL. * * Sets the url of the legend image. */ setUrl: function(url) { this.url = url; var el = this.getImgEl(); if (el) { el.un("load", this.onImageLoad, this); el.on("load", this.onImageLoad, this, {single: true}); el.un("error", this.onImageLoadError, this); el.on("error", this.onImageLoadError, this, {single: true}); el.dom.src = url; } }, /** private: method[onRender] * Private method called when the legend image component is being * rendered. */ onRender: function(ct, position) { GeoExt.LegendImage.superclass.onRender.call(this, ct, position); if(this.url) { this.setUrl(this.url); } }, /** private: method[onDestroy] * Private method called during the destroy sequence. */ onDestroy: function() { var el = this.getImgEl(); if(el) { el.un("load", this.onImageLoad, this); el.un("error", this.onImageLoadError, this); } GeoExt.LegendImage.superclass.onDestroy.apply(this, arguments); }, /** private: method[onImageLoadError] * Private method called if the legend image fails loading. */ onImageLoadError: function() { var el = this.getImgEl(); el.addClass(this.noImgCls); el.dom.src = this.defaultImgSrc; }, /** private: method[onImageLoad] * Private method called after the legend image finished loading. */ onImageLoad: function() { var el = this.getImgEl(); if (!OpenLayers.Util.isEquivalentUrl(el.dom.src, this.defaultImgSrc)) { el.removeClass(this.noImgCls); } } }); /** api: xtype = gx_legendimage */ Ext.reg('gx_legendimage', GeoExt.LegendImage); /** FILE: GeoExt/widgets/PrintMapPanel.js **/ /** * Copyright (c) 2008-2012 The Open Source Geospatial Foundation * * Published under the BSD license. * See http://svn.geoext.org/core/trunk/geoext/license.txt for the full text * of the license. */ /** * @requires GeoExt/widgets/MapPanel.js * @include GeoExt/data/PrintProvider.js * @include GeoExt/data/PrintPage.js */ Ext.namespace("GeoExt"); /** api: (define) * module = GeoExt * class = PrintMapPanel */ /** api: (extends) * GeoExt/widgets/MapPanel.js */ /** api: example * A map with a "Print..." button. If clicked, a dialog containing a * PrintMapPanel will open, with a "Create PDF" button. * * .. code-block:: javascript * * var mapPanel = new GeoExt.MapPanel({ * renderTo: "map", * layers: [new OpenLayers.Layer.WMS("Tasmania State Boundaries", * "http://demo.opengeo.org/geoserver/wms", * {layers: "topp:tasmania_state_boundaries"}, {singleTile: true})], * center: [146.56, -41.56], * zoom: 6, * bbar: [{ * text: "Print...", * handler: function() { * var printDialog = new Ext.Window({ * autoHeight: true, * width: 350, * items: [new GeoExt.PrintMapPanel({ * sourceMap: mapPanel, * printProvider: { * capabilities: printCapabilities * } * })], * bbar: [{ * text: "Create PDF", * handler: function() { * printDialog.items.get(0).print(); * } * }] * }); * printDialog.show(); * } * }] * }); */ /** api: constructor * .. class:: PrintMapPanel * * A map panel that controls scale and center of a print page. Based on the * current view (i.e. layers and extent) of a source map, this panel will be * sized according to the aspect ratio of the print page. As the user zooms * and pans in the :class:`GeoExt.PrintMapPanel`, the print page will update * its scale and center accordingly. If the scale on the print page changes * (e.g. by setting it using a combo box with a * :class:`GeoExt.plugins.PrintPageField`), the extent of the * :class:`GeoExt.PrintMapPanel` will be updated to match the page bounds. * * .. note:: The ``zoom``, ``center`` and ``extent`` config options will have * no affect, as they will be determined by the ``sourceMap``. */ GeoExt.PrintMapPanel = Ext.extend(GeoExt.MapPanel, { /** api: config[map] * ``Object`` Optional configuration for the ``OpenLayers.Map`` object * that this PrintMapPanel creates. Useful e.g. to configure a map with a * custom set of controls, or to add a ``preaddlayer`` listener for * filtering out layer types that cannot be printed. */ /** api: config[sourceMap] * :class:`GeoExt.MapPanel` or ``OpenLayers.Map`` The map that is to be * printed. */ /** private: property[sourceMap] * ``OpenLayers.Map`` */ sourceMap: null, /** api: config[printProvider] * :class:`GeoExt.data.PrintProvider` or ``Object`` PrintProvider to use * for printing. If an ``Object`` is provided, a new PrintProvider will * be created and configured with the object. * * .. note:: The PrintMapPanel requires the printProvider's capabilities * to be available upon initialization. This means that a PrintMapPanel * configured with an ``Object`` as ``printProvider`` will only work * when ``capabilities`` is provided in the printProvider's * configuration object. If ``printProvider`` is provided as an instance * of :class:`GeoExt.data.PrintProvider`, the capabilities must be * loaded before PrintMapPanel initialization. */ /** api: property[printProvider] * :class:`GeoExt.data.PrintProvider` PrintProvider for this * PrintMapPanel. */ printProvider: null, /** api: property[printPage] * :class:`GeoExt.data.PrintPage` PrintPage for this PrintMapPanel. * Read-only. */ printPage: null, /** api: config[limitScales] * ``Boolean`` If set to true, the printPage cannot be set to scales that * would generate a preview in this :class:`GeoExt.PrintMapPanel` with a * completely different extent than the one that would appear on the * printed map. Default is false. */ /** api: property[previewScales] * ``Ext.data.Store`` A data store with a subset of the printProvider's * scales. By default, this contains all the scales of the printProvider. * If ``limitScales`` is set to true, it will only contain print scales * that can properly be previewed with this :class:`GeoExt.PrintMapPanel`. */ previewScales: null, /** api: config[center] * ``OpenLayers.LonLat`` or ``Array(Number)`` A location for the map * center. Do not set, as this will be overridden with the ``sourceMap`` * center. */ center: null, /** api: config[zoom] * ``Number`` An initial zoom level for the map. Do not set, because the * initial extent will be determined by the ``sourceMap``. */ zoom: null, /** api: config[extent] * ``OpenLayers.Bounds or Array(Number)`` An initial extent for the map. * Do not set, because the initial extent will be determined by the * ``sourceMap``. */ extent: null, /** private: property[currentZoom] * ``Number`` */ currentZoom: null, /** * private: method[initComponent] * private override */ initComponent: function() { if(this.sourceMap instanceof GeoExt.MapPanel) { this.sourceMap = this.sourceMap.map; } if (!this.map) { this.map = {}; } Ext.applyIf(this.map, { projection: this.sourceMap.getProjection(), maxExtent: this.sourceMap.getMaxExtent(), maxResolution: this.sourceMap.getMaxResolution(), units: this.sourceMap.getUnits() }); if(!(this.printProvider instanceof GeoExt.data.PrintProvider)) { this.printProvider = new GeoExt.data.PrintProvider( this.printProvider); } this.printPage = new GeoExt.data.PrintPage({ printProvider: this.printProvider }); this.previewScales = new Ext.data.Store(); this.previewScales.add(this.printProvider.scales.getRange()); this.layers = []; var layer; Ext.each(this.sourceMap.layers, function(layer) { if (layer.getVisibility() === true) { if (layer instanceof OpenLayers.Layer.Vector) { var features = layer.features, clonedFeatures = new Array(features.length), vector = new OpenLayers.Layer.Vector(layer.name); for (var i=0, ii=features.length; i targetHeight) { height = targetHeight; width = height * ratio; } else { width = targetWidth; } } else if (targetHeight) { width = targetHeight * ratio; height = targetHeight; } return {width: width, height: height}; }, /** private: method[fitZoom] * Fits this PrintMapPanel's zoom to the print scale. */ fitZoom: function() { if (!this._updating && this.printPage.scale) { this._updating = true; var printBounds = this.printPage.getPrintExtent(this.map); this.currentZoom = this.map.getZoomForExtent(printBounds); this.map.zoomToExtent(printBounds); delete this._updating; } }, /** private: method[updatePage] * updates the print page to match this PrintMapPanel's center and scale. */ updatePage: function() { if (!this._updating) { var zoom = this.map.getZoom(); this._updating = true; if (zoom === this.currentZoom) { this.printPage.setCenter(this.map.getCenter()); } else { this.printPage.fit(this.map); } delete this._updating; this.currentZoom = zoom; } }, /** private: method[calculatePreviewScales] */ calculatePreviewScales: function() { this.previewScales.removeAll(); this.printPage.suspendEvents(); var scale = this.printPage.scale; // group print scales by the zoom level they would be previewed at var viewSize = this.map.getSize(); var scalesByZoom = {}; var zooms = []; this.printProvider.scales.each(function(rec) { this.printPage.setScale(rec); var extent = this.printPage.getPrintExtent(this.map); var zoom = this.map.getZoomForExtent(extent); var idealResolution = Math.max( extent.getWidth() / viewSize.w, extent.getHeight() / viewSize.h ); var resolution = this.map.getResolutionForZoom(zoom); // the closer to the ideal resolution, the better the fit var diff = Math.abs(idealResolution - resolution); if (!(zoom in scalesByZoom) || scalesByZoom[zoom].diff > diff) { scalesByZoom[zoom] = { rec: rec, diff: diff }; zooms.indexOf(zoom) == -1 && zooms.push(zoom); } }, this); // add only the preview scales that closely fit print extents for (var i=0, ii=zooms.length; i 0) { var maxScale = this.previewScales.getAt(0); var minScale = this.previewScales.getAt(this.previewScales.getCount()-1); if (scale.get("value") < minScale.get("value")) { this.printPage.setScale(minScale); } else if (scale.get("value") > maxScale.get("value")) { this.printPage.setScale(maxScale); } } this.fitZoom(); }, /** api: method[print] * :param options: ``Object`` options for * the :class:`GeoExt.data.PrintProvider` :: ``print`` method. * * Convenience method for printing the map, without the need to * interact with the printProvider and printPage. */ print: function(options) { this.printProvider.print(this.map, [this.printPage], options); }, /** private: method[beforeDestroy] */ beforeDestroy: function() { this.map.events.unregister("moveend", this, this.updatePage); this.printPage.un("change", this.fitZoom, this); this.printProvider.un("layoutchange", this.syncSize, this); GeoExt.PrintMapPanel.superclass.beforeDestroy.apply(this, arguments); } }); /** api: xtype = gx_printmappanel */ Ext.reg('gx_printmappanel', GeoExt.PrintMapPanel); /** FILE: 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 * constructor. * * Inherits from: * - */ 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 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 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" }); /** FILE: 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 * constructor. * * Inherits from: * - */ 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 . If type is "Geometry", the input json * must represent a single geometry, and the return will be an * . If type is "Feature", the input json must * represent a single feature, and the return will be an * . */ 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. * * Parameters: * obj - {Object} An object created from a GeoJSON object * * Returns: * {} 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 . * * Parameters: * obj - {Object} An object created from a GeoJSON object * * Returns: * {} 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. * * Parameters: * array - {Object} The coordinates array from the GeoJSON fragment. * * Returns: * {} 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 * . * * Parameters: * array - {Object} The coordinates array from the GeoJSON fragment. * * Returns: * {} A geometry. */ "multipoint": function(array) { var points = []; var p = null; for(var i=0, len=array.length; i. * * Parameters: * array - {Object} The coordinates array from the GeoJSON fragment. * * Returns: * {} A geometry. */ "linestring": function(array) { var points = []; var p = null; for(var i=0, len=array.length; i. * * Parameters: * array - {Object} The coordinates array from the GeoJSON fragment. * * Returns: * {} A geometry. */ "multilinestring": function(array) { var lines = []; var l = null; for(var i=0, len=array.length; i. * * Returns: * {} A geometry. */ "polygon": function(array) { var rings = []; var r, l; for(var i=0, len=array.length; i. * * Parameters: * array - {Object} The coordinates array from the GeoJSON fragment. * * Returns: * {} A geometry. */ "multipolygon": function(array) { var polys = []; var p = null; for(var i=0, len=array.length; i. * * Parameters: * array - {Object} The coordinates array from the GeoJSON fragment. * * Returns: * {} 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 , , * 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} * * 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 - {} * * 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 - {} * * 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 - {} * * 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 - {} * * 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} * * Returns: * {Array} An array of coordinate arrays representing * the linestring. */ 'linestring': function(linestring) { var array = []; for(var i=0, len=linestring.components.length; i} * * Returns: * {Array} An array of linestring arrays representing * the multilinestring. */ 'multilinestring': function(multilinestring) { var array = []; for(var i=0, len=multilinestring.components.length; i} * * 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} * * Returns: * {Array} An array of polygon arrays representing * the multipolygon */ 'multipolygon': function(multipolygon) { var array = []; for(var i=0, len=multipolygon.components.length; i} * * 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`_ */ Ext.namespace("GeoExt.data"); /** api: example * Minimal code to print as much of the current map extent as possible as * soon as the print service capabilities are loaded, using the first layout * reported by the print service: * * .. code-block:: javascript * * var mapPanel = new GeoExt.MapPanel({ * renderTo: "mappanel", * layers: [new OpenLayers.Layer.WMS("wms", "/geoserver/wms", * {layers: "topp:tasmania_state_boundaries"})], * center: [146.56, -41.56], * zoom: 7 * }); * var printProvider = new GeoExt.data.PrintProvider({ * url: "/geoserver/pdf", * listeners: { * "loadcapabilities": function() { * var printPage = new GeoExt.data.PrintPage({ * printProvider: printProvider * }); * printPage.fit(mapPanel, true); * printProvider.print(mapPanel, printPage); * } * } * }); */ /** api: constructor * .. class:: PrintProvider * * Provides an interface to a Mapfish or GeoServer print module. For printing, * one or more instances of :class:`GeoExt.data.PrintPage` are also required * to tell the PrintProvider about the scale and extent (and optionally * rotation) of the page(s) we want to print. */ GeoExt.data.PrintProvider = Ext.extend(Ext.util.Observable, { /** api: config[url] * ``String`` Base url of the print service. Only required if * ``capabilities`` is not provided. This * is usually something like http://path/to/mapfish/print for Mapfish, * and http://path/to/geoserver/pdf for GeoServer with the printing * extension installed. This property requires that the print service is * at the same origin as the application (or accessible via proxy). */ /** private: property[url] * ``String`` Base url of the print service. Will always have a trailing * "/". */ url: null, /** api: config[autoLoad] * ``Boolean`` If set to true, the capabilities will be loaded upon * instance creation, and ``loadCapabilities`` does not need to be called * manually. Setting this when ``capabilities`` and no ``url`` is provided * has no effect. Default is false. */ /** api: config[capabilities] * ``Object`` Capabilities of the print service. Only required if ``url`` * is not provided. This is the object returned by the ``info.json`` * endpoint of the print service, and is usually obtained by including a * script tag pointing to * http://path/to/printservice/info.json?var=myvar in the head of the * html document, making the capabilities accessible as ``window.myvar``. * This property should be used when no local print service or proxy is * available, or when you do not listen for the ``loadcapabilities`` * events before creating components that require the PrintProvider's * capabilities to be available. */ /** private: property[capabilities] * ``Object`` Capabilities as returned from the print service. */ capabilities: null, /** api: config[method] * ``String`` Either ``POST`` or ``GET`` (case-sensitive). Method to use * when sending print requests to the servlet. If the print service is at * the same origin as the application (or accessible via proxy), then * ``POST`` is recommended. Use ``GET`` when accessing a remote print * service with no proxy available, but expect issues with character * encoding and URLs exceeding the maximum length. Default is ``POST``. */ /** private: property[method] * ``String`` Either ``POST`` or ``GET`` (case-sensitive). Method to use * when sending print requests to the servlet. */ method: "POST", /** api: config[encoding] * ``String`` The encoding to set in the headers when requesting the print * service. Prevent character encoding issues, especially when using IE. * Default is retrieved from document charset or characterSet if existing * or ``UTF-8`` if not. */ encoding: document.charset || document.characterSet || "UTF-8", /** api: config[timeout] * ``Number`` Timeout of the POST Ajax request used for the print request * (in milliseconds). Default of 30 seconds. Has no effect if ``method`` * is set to ``GET``. */ timeout: 30000, /** api: property[customParams] * ``Object`` Key-value pairs of custom data to be sent to the print * service. Optional. This is e.g. useful for complex layout definitions * on the server side that require additional parameters. */ customParams: null, /** api: config[baseParams] * ``Object`` Key-value pairs of base params to be add to every * request to the service. Optional. */ /** api: property[scales] * ``Ext.data.JsonStore`` read-only. A store representing the scales * available. * * Fields of records in this store: * * * name - ``String`` the name of the scale * * value - ``Float`` the scale denominator */ scales: null, /** api: property[dpis] * ``Ext.data.JsonStore`` read-only. A store representing the dpis * available. * * Fields of records in this store: * * * name - ``String`` the name of the dpi * * value - ``Float`` the dots per inch */ dpis: null, /** api: property[layouts] * ``Ext.data.JsonStore`` read-only. A store representing the layouts * available. * * Fields of records in this store: * * * name - ``String`` the name of the layout * * size - ``Object`` width and height of the map in points * * rotation - ``Boolean`` indicates if rotation is supported */ layouts: null, /** api: property[dpi] * ``Ext.data.Record`` the record for the currently used resolution. * Read-only, use ``setDpi`` to set the value. */ dpi: null, /** api: property[layout] * ``Ext.data.Record`` the record of the currently used layout. Read-only, * use ``setLayout`` to set the value. */ layout: null, /** private: method[constructor] * Private constructor override. */ constructor: function(config) { this.initialConfig = config; Ext.apply(this, config); if(!this.customParams) { this.customParams = {}; } this.addEvents( /** api: event[loadcapabilities] * Triggered when the capabilities have finished loading. This * event will only fire when ``capabilities`` is not configured. * * Listener arguments: * * * printProvider - :class:`GeoExt.data.PrintProvider` this * PrintProvider * * capabilities - ``Object`` the capabilities */ "loadcapabilities", /** api: event[layoutchange] * Triggered when the layout is changed. * * Listener arguments: * * * printProvider - :class:`GeoExt.data.PrintProvider` this * PrintProvider * * layout - ``Ext.data.Record`` the new layout */ "layoutchange", /** api: event[dpichange] * Triggered when the dpi value is changed. * * Listener arguments: * * * printProvider - :class:`GeoExt.data.PrintProvider` this * PrintProvider * * dpi - ``Ext.data.Record`` the new dpi record */ "dpichange", /** api: event[beforeprint] * Triggered when the print method is called. * TODO: rename this event to beforeencode * * Listener arguments: * * * printProvider - :class:`GeoExt.data.PrintProvider` this * PrintProvider * * map - ``OpenLayers.Map`` the map being printed * * pages - Array of :class:`GeoExt.data.PrintPage` the print * pages being printed * * options - ``Object`` the options to the print command */ "beforeprint", /** api: event[print] * Triggered when the print document is opened. * * Listener arguments: * * * printProvider - :class:`GeoExt.data.PrintProvider` this * PrintProvider * * url - ``String`` the url of the print document */ "print", /** api: event[printexception] * Triggered when using the ``POST`` method, when the print * backend returns an exception. * * Listener arguments: * * * printProvider - :class:`GeoExt.data.PrintProvider` this * PrintProvider * * response - ``Object`` the response object of the XHR */ "printexception", /** api: event[beforeencodelayer] * Triggered before a layer is encoded. This can be used to * exclude layers from the printing, by having the listener * return false. * * Listener arguments: * * * printProvider - :class:`GeoExt.data.PrintProvider` this * PrintProvider * * layer - ``OpenLayers.Layer`` the layer which is about to be * encoded. */ "beforeencodelayer", /** api: event[encodelayer] * Triggered when a layer is encoded. This can be used to modify * the encoded low-level layer object that will be sent to the * print service. * * Listener arguments: * * * printProvider - :class:`GeoExt.data.PrintProvider` this * PrintProvider * * layer - ``OpenLayers.Layer`` the layer which is about to be * encoded. * * encodedLayer - ``Object`` the encoded layer that will be * sent to the print service. */ "encodelayer", /** api: events[beforedownload] * Triggered before the PDF is downloaded. By returning false from * a listener, the default handling of the PDF can be cancelled * and applications can take control over downloading the PDF. * TODO: rename to beforeprint after the current beforeprint event * has been renamed to beforeencode. * * Listener arguments: * * printProvider - :class:`GeoExt.data.PrintProvider` this * PrintProvider * * url - ``String`` the url of the print document */ "beforedownload", /** api: event[beforeencodelegend] * Triggered before the legend is encoded. If the listener * returns false, the default encoding based on GeoExt.LegendPanel * will not be executed. This provides an option for application * to get legend info from a custom component other than * GeoExt.LegendPanel. * * Listener arguments: * * * printProvider - :class:`GeoExt.data.PrintProvider` this * PrintProvider * * jsonData - ``Object`` The data that will be sent to the print * server. Can be used to populate jsonData.legends. * * legend - ``Object`` The legend supplied in the options which were * sent to the print function. */ "beforeencodelegend" ); GeoExt.data.PrintProvider.superclass.constructor.apply(this, arguments); this.scales = new Ext.data.JsonStore({ root: "scales", sortInfo: {field: "value", direction: "DESC"}, fields: ["name", {name: "value", type: "float"}] }); this.dpis = new Ext.data.JsonStore({ root: "dpis", fields: ["name", {name: "value", type: "float"}] }); this.layouts = new Ext.data.JsonStore({ root: "layouts", fields: [ "name", {name: "size", mapping: "map"}, {name: "rotation", type: "boolean"} ] }); if(config.capabilities) { this.loadStores(); } else { if(this.url.split("/").pop()) { this.url += "/"; } this.initialConfig.autoLoad && this.loadCapabilities(); } }, /** api: method[setLayout] * :param layout: ``Ext.data.Record`` the record of the layout. * * Sets the layout for this printProvider. */ setLayout: function(layout) { this.layout = layout; this.fireEvent("layoutchange", this, layout); }, /** api: method[setDpi] * :param dpi: ``Ext.data.Record`` the dpi record. * * Sets the dpi for this printProvider. */ setDpi: function(dpi) { this.dpi = dpi; this.fireEvent("dpichange", this, dpi); }, /** api: method[print] * :param map: ``GeoExt.MapPanel`` or ``OpenLayers.Map`` The map to print. * :param pages: ``Array`` of :class:`GeoExt.data.PrintPage` or * :class:`GeoExt.data.PrintPage` page(s) to print. * :param options: ``Object`` of additional options, see below. * * Sends the print command to the print service and opens a new window * with the resulting PDF. * * Valid properties for the ``options`` argument: * * * ``legend`` - :class:`GeoExt.LegendPanel` If provided, the legend * will be added to the print document. For the printed result to * look like the LegendPanel, the following ``!legends`` block * should be included in the ``items`` of your page layout in the * print module's configuration file: * * .. code-block:: none * * - !legends * maxIconWidth: 0 * maxIconHeight: 0 * classIndentation: 0 * layerSpace: 5 * layerFontSize: 10 * * * ``overview`` - :class:`OpenLayers.Control.OverviewMap` If provided, * the layers for the overview map in the printout will be taken from * the OverviewMap control. If not provided, the print service will * use the main map's layers for the overview map. Applies only for * layouts configured to print an overview map. */ print: function(map, pages, options) { if(map instanceof GeoExt.MapPanel) { map = map.map; } pages = pages instanceof Array ? pages : [pages]; options = options || {}; if(this.fireEvent("beforeprint", this, map, pages, options) === false) { return; } var jsonData = Ext.apply({ units: map.getUnits(), srs: map.baseLayer.projection.getCode(), layout: this.layout.get("name"), dpi: this.dpi.get("value") }, this.customParams); var pagesLayer = pages[0].feature.layer; var encodedLayers = []; // ensure that the baseLayer is the first one in the encoded list var layers = map.layers.concat(); layers.remove(map.baseLayer); layers.unshift(map.baseLayer); Ext.each(layers, function(layer){ if(layer !== pagesLayer && layer.getVisibility() === true) { var enc = this.encodeLayer(layer); enc && encodedLayers.push(enc); } }, this); jsonData.layers = encodedLayers; var encodedPages = []; Ext.each(pages, function(page) { encodedPages.push(Ext.apply({ center: [page.center.lon, page.center.lat], scale: page.scale.get("value"), rotation: page.rotation }, page.customParams)); }, this); jsonData.pages = encodedPages; if (options.overview) { var encodedOverviewLayers = []; Ext.each(options.overview.layers, function(layer) { var enc = this.encodeLayer(layer); enc && encodedOverviewLayers.push(enc); }, this); jsonData.overviewLayers = encodedOverviewLayers; } if(options.legend && !(this.fireEvent("beforeencodelegend", this, jsonData, options.legend) === false)) { var legend = options.legend; var rendered = legend.rendered; if (!rendered) { legend = legend.cloneConfig({ renderTo: document.body, hidden: true }); } var encodedLegends = []; legend.items && legend.items.each(function(cmp) { if(!cmp.hidden) { var encFn = this.encoders.legends[cmp.getXType()]; // MapFish Print doesn't currently support per-page // legends, so we use the scale of the first page. encodedLegends = encodedLegends.concat( encFn.call(this, cmp, jsonData.pages[0].scale)); } }, this); if (!rendered) { legend.destroy(); } jsonData.legends = encodedLegends; } if(this.method === "GET") { var url = Ext.urlAppend(this.capabilities.printURL, "spec=" + encodeURIComponent(Ext.encode(jsonData))); this.download(url); } else { Ext.Ajax.request({ url: this.capabilities.createURL, timeout: this.timeout, jsonData: jsonData, headers: {"Content-Type": "application/json; charset=" + this.encoding}, success: function(response) { var url = Ext.decode(response.responseText).getURL; this.download(url); }, failure: function(response) { this.fireEvent("printexception", this, response); }, params: this.initialConfig.baseParams, scope: this }); } }, /** private: method[download] * :param url: ``String`` */ download: function(url) { if (this.fireEvent("beforedownload", this, url) !== false) { if (Ext.isOpera) { // Make sure that Opera don't replace the content tab with // the pdf window.open(url); } else { // This avoids popup blockers for all other browsers window.location.href = url; } } this.fireEvent("print", this, url); }, /** api: method[loadCapabilities] * * Loads the capabilities from the print service. If this instance is * configured with either ``capabilities`` or a ``url`` and ``autoLoad`` * set to true, then this method does not need to be called from the * application. */ loadCapabilities: function() { if (!this.url) { return; } var url = this.url + "info.json"; Ext.Ajax.request({ url: url, method: "GET", disableCaching: false, success: function(response) { this.capabilities = Ext.decode(response.responseText); this.loadStores(); }, params: this.initialConfig.baseParams, scope: this }); }, /** private: method[loadStores] */ loadStores: function() { this.scales.loadData(this.capabilities); this.dpis.loadData(this.capabilities); this.layouts.loadData(this.capabilities); this.setLayout(this.layouts.getAt(0)); this.setDpi(this.dpis.getAt(0)); this.fireEvent("loadcapabilities", this, this.capabilities); }, /** private: method[encodeLayer] * :param layer: ``OpenLayers.Layer`` * :return: ``Object`` * * Encodes a layer for the print service. */ encodeLayer: function(layer) { var encLayer; for(var c in this.encoders.layers) { if(OpenLayers.Layer[c] && layer instanceof OpenLayers.Layer[c]) { if(this.fireEvent("beforeencodelayer", this, layer) === false) { return; } encLayer = this.encoders.layers[c].call(this, layer); this.fireEvent("encodelayer", this, layer, encLayer); break; } } // only return the encLayer object when we have a type. Prevents a // fallback on base encoders like HTTPRequest. return (encLayer && encLayer.type) ? encLayer : null; }, /** private: method[getAbsoluteUrl] * :param url: ``String`` * :return: ``String`` * * Converts the provided url to an absolute url. */ getAbsoluteUrl: function(url) { if (Ext.isSafari) { url = url.replace(/{/g, '%7B'); url = url.replace(/}/g, '%7D'); } var a; if (Ext.isIE6 || Ext.isIE7 || Ext.isIE8) { a = document.createElement(""); a.style.display = "none"; document.body.appendChild(a); a.href = a.href; document.body.removeChild(a); } else { a = document.createElement("a"); a.href = url; } return a.href; }, /** private: property[encoders] * ``Object`` Encoders for all print content */ encoders: { "layers": { "Layer": function(layer) { var enc = {}; if (layer.options && layer.options.maxScale) { enc.minScaleDenominator = layer.options.maxScale; } if (layer.options && layer.options.minScale) { enc.maxScaleDenominator = layer.options.minScale; } return enc; }, "WMS": function(layer) { var enc = this.encoders.layers.HTTPRequest.call(this, layer); enc.singleTile = layer.singleTile; Ext.apply(enc, { type: 'WMS', layers: [layer.params.LAYERS].join(",").split(","), format: layer.params.FORMAT, styles: [layer.params.STYLES].join(",").split(","), singleTile: layer.singleTile }); var param; for(var p in layer.params) { param = p.toLowerCase(); if(layer.params[p] != null && !layer.DEFAULT_PARAMS[param] && "layers,styles,width,height,srs".indexOf(param) == -1) { if(!enc.customParams) { enc.customParams = {}; } enc.customParams[p] = layer.params[p]; } } return enc; }, "OSM": function(layer) { var enc = this.encoders.layers.TileCache.call(this, layer); return Ext.apply(enc, { type: 'OSM', baseURL: enc.baseURL.substr(0, enc.baseURL.indexOf("$")), extension: "png" }); }, "TMS": function(layer) { var enc = this.encoders.layers.TileCache.call(this, layer); return Ext.apply(enc, { type: 'TMS', format: layer.type }); }, "TileCache": function(layer) { var enc = this.encoders.layers.HTTPRequest.call(this, layer); return Ext.apply(enc, { type: 'TileCache', layer: layer.layername, maxExtent: layer.maxExtent.toArray(), tileSize: [layer.tileSize.w, layer.tileSize.h], extension: layer.extension, resolutions: layer.serverResolutions || layer.resolutions }); }, "WMTS": function(layer) { var enc = this.encoders.layers.HTTPRequest.call(this, layer); enc = Ext.apply(enc, { type: 'WMTS', layer: layer.layer, version: layer.version, requestEncoding: layer.requestEncoding, style: layer.style, dimensions: layer.dimensions, params: layer.params, matrixSet: layer.matrixSet }); if (layer.matrixIds) { if (layer.requestEncoding == "KVP") { enc.format = layer.format; } enc.matrixIds = [] Ext.each(layer.matrixIds, function(matrixId) { enc.matrixIds.push({ identifier: matrixId.identifier, matrixSize: [matrixId.matrixWidth, matrixId.matrixHeight], resolution: matrixId.scaleDenominator * 0.28E-3 / OpenLayers.METERS_PER_INCH / OpenLayers.INCHES_PER_UNIT[layer.units], tileSize: [matrixId.tileWidth, matrixId.tileHeight], topLeftCorner: [matrixId.topLeftCorner.lon, matrixId.topLeftCorner.lat] }); }) return enc; } else { return Ext.apply(enc, { formatSuffix: layer.formatSuffix, tileOrigin: [layer.tileOrigin.lon, layer.tileOrigin.lat], tileSize: [layer.tileSize.w, layer.tileSize.h], maxExtent: (layer.tileFullExtent != null) ? layer.tileFullExtent.toArray() : layer.maxExtent.toArray(), zoomOffset: layer.zoomOffset, resolutions: layer.serverResolutions || layer.resolutions }); } }, "KaMapCache": function(layer) { var enc = this.encoders.layers.KaMap.call(this, layer); return Ext.apply(enc, { type: 'KaMapCache', // group param is mandatory when using KaMapCache group: layer.params['g'], metaTileWidth: layer.params['metaTileSize']['w'], metaTileHeight: layer.params['metaTileSize']['h'] }); }, "KaMap": function(layer) { var enc = this.encoders.layers.HTTPRequest.call(this, layer); return Ext.apply(enc, { type: 'KaMap', map: layer.params['map'], extension: layer.params['i'], // group param is optional when using KaMap group: layer.params['g'] || "", maxExtent: layer.maxExtent.toArray(), tileSize: [layer.tileSize.w, layer.tileSize.h], resolutions: layer.serverResolutions || layer.resolutions }); }, "HTTPRequest": function(layer) { var enc = this.encoders.layers.Layer.call(this, layer); return Ext.apply(enc, { baseURL: this.getAbsoluteUrl(layer.url instanceof Array ? layer.url[0] : layer.url), opacity: (layer.opacity != null) ? layer.opacity : 1.0 }); }, "Image": function(layer) { var enc = this.encoders.layers.Layer.call(this, layer); return Ext.apply(enc, { type: 'Image', baseURL: this.getAbsoluteUrl(layer.getURL(layer.extent)), opacity: (layer.opacity != null) ? layer.opacity : 1.0, extent: layer.extent.toArray(), pixelSize: [layer.size.w, layer.size.h], name: layer.name }); }, "Vector": function(layer) { if(!layer.features.length) { return; } var encFeatures = []; var encStyles = {}; var features = layer.features; var featureFormat = new OpenLayers.Format.GeoJSON(); var styleFormat = new OpenLayers.Format.JSON(); var nextId = 1; var styleDict = {}; var feature, style, dictKey, dictItem, styleName; for(var i=0, len=features.length; i`_ */ Ext.namespace("GeoExt.data"); /** api: constructor * .. class:: PrintPage * * Provides a representation of a print page for * :class:`GeoExt.data.PrintProvider`. The extent of the page is stored as * ``OpenLayers.Feature.Vector``. Widgets can use this to display the print * extent on the map. */ GeoExt.data.PrintPage = Ext.extend(Ext.util.Observable, { /** api:config[printProvider] * :class:`GeoExt.data.PrintProvider` The print provider to use with * this page. */ /** private: property[printProvider] * :class:`GeoExt.data.PrintProvider` */ printProvider: null, /** api: property[feature] * ``OpenLayers.Feature.Vector`` Feature representing the page extent. To * get the extent of the print page for a specific map, use * ``getPrintExtent``. * Read-only. */ feature: null, /** api: property[center] * ``OpenLayers.LonLat`` The current center of the page. Read-only. */ center: null, /** api: property[scale] * ``Ext.data.Record`` The current scale record of the page. Read-only. */ scale: null, /** api: property[rotation] * ``Float`` The current rotation of the page. Read-only. */ rotation: 0, /** api:config[customParams] * ``Object`` Key-value pairs of additional parameters that the * printProvider will send to the print service for this page. */ /** api: property[customParams] * ``Object`` Key-value pairs of additional parameters that the * printProvider will send to the print service for this page. */ customParams: null, /** private: method[constructor] * Private constructor override. */ constructor: function(config) { this.initialConfig = config; Ext.apply(this, config); if(!this.customParams) { this.customParams = {}; } this.addEvents( /** api: event[change] * Triggered when any of the page properties have changed * * Listener arguments: * * * printPage - :class:`GeoExt.data.PrintPage` this printPage * * modifications - ``Object`` Object with one or more of * ``scale``, ``center`` and ``rotation``, notifying * listeners of the changed properties. */ "change" ); GeoExt.data.PrintPage.superclass.constructor.apply(this, arguments); this.feature = new OpenLayers.Feature.Vector( new OpenLayers.Geometry.Polygon([ new OpenLayers.Geometry.LinearRing([ new OpenLayers.Geometry.Point(-1, -1), new OpenLayers.Geometry.Point(1, -1), new OpenLayers.Geometry.Point(1, 1), new OpenLayers.Geometry.Point(-1, 1) ]) ]) ); if(this.printProvider.capabilities) { this.setScale(this.printProvider.scales.getAt(0)); } else { this.printProvider.on({ "loadcapabilities": function() { this.setScale(this.printProvider.scales.getAt(0)); }, scope: this, single: true }); } this.printProvider.on({ "layoutchange": this.onLayoutChange, scope: this }); }, /** api: method[getPrintExtent] * :param map: ``OpenLayers.Map`` or :class:`GeoExt.MapPanel` the map to * get the print extent for. * :returns: ``OpenLayers.Bounds`` * * Gets this page's print extent for the provided map. */ getPrintExtent: function(map) { map = map instanceof GeoExt.MapPanel ? map.map : map; return this.calculatePageBounds(this.scale, map.getUnits()); }, /** api: method[setScale] * :param scale: ``Ext.data.Record`` The new scale record. * :param units: ``String`` map units to use for the scale calculation. * Optional if the ``feature`` is on a layer which is added to a map. * If not found, "dd" will be assumed. * * Updates the page geometry to match a given scale. Since this takes the * current layout of the printProvider into account, this can be used to * update the page geometry feature when the layout has changed. */ setScale: function(scale, units) { var bounds = this.calculatePageBounds(scale, units); var geom = bounds.toGeometry(); var rotation = this.rotation; if(rotation != 0) { geom.rotate(-rotation, geom.getCentroid()); } this.updateFeature(geom, {scale: scale}); }, /** api: method[setCenter] * :param center: ``OpenLayers.LonLat`` The new center. * * Moves the page extent to a new center. */ setCenter: function(center) { var geom = this.feature.geometry; var oldCenter = geom.getBounds().getCenterLonLat(); var dx = center.lon - oldCenter.lon; var dy = center.lat - oldCenter.lat; geom.move(dx, dy); this.updateFeature(geom, {center: center}); }, /** api: method[setRotation] * :param rotation: ``Float`` The new rotation. * :param force: ``Boolean`` If set to true, the rotation will also be * set when the layout does not support it. Default is false. * * Sets a new rotation for the page geometry. */ setRotation: function(rotation, force) { if(force || this.printProvider.layout.get("rotation") === true) { var geom = this.feature.geometry; geom.rotate(this.rotation - rotation, geom.getCentroid()); this.updateFeature(geom, {rotation: rotation}); } }, /** api: method[fit] * :param fitTo: :class:`GeoExt.MapPanel` or ``OpenLayers.Map`` or ``OpenLayers.Feature.Vector`` * The map or feature to fit the page to. * :param options: ``Object`` Additional options to determine how to fit * * Fits the page layout to a map or feature extent. If the map extent has * not been centered yet, this will do nothing. * * Available options: * * * mode - ``String`` How to calculate the print extent? If "closest", * the closest matching print extent will be chosen. If "printer", the * chosen print extent will be the closest one that can show the entire * ``fitTo`` extent on the printer. If "screen", it will be the closest * one that is entirely visible inside the ``fitTo`` extent. Default is * "printer". * */ fit: function(fitTo, options) { options = options || {}; var map = fitTo, extent; if(fitTo instanceof GeoExt.MapPanel) { map = fitTo.map; } else if(fitTo instanceof OpenLayers.Feature.Vector) { map = fitTo.layer.map; extent = fitTo.geometry.getBounds(); } if(!extent) { extent = map.getExtent(); if(!extent) { return; } } this._updating = true; var center = extent.getCenterLonLat(); this.setCenter(center); var units = map.getUnits(); var scale = this.printProvider.scales.getAt(0); var closest = Number.POSITIVE_INFINITY; var mapWidth = extent.getWidth(); var mapHeight = extent.getHeight(); this.printProvider.scales.each(function(rec) { var bounds = this.calculatePageBounds(rec, units); if (options.mode == "closest") { var diff = Math.abs(bounds.getWidth() - mapWidth) + Math.abs(bounds.getHeight() - mapHeight); if (diff < closest) { closest = diff; scale = rec; } } else { var contains = options.mode == "screen" ? !extent.containsBounds(bounds) : bounds.containsBounds(extent); if (contains || (options.mode == "screen" && !contains)) { scale = rec; } return contains; } }, this); this.setScale(scale, units); delete this._updating; this.updateFeature(this.feature.geometry, { center: center, scale: scale }); }, /** private: method[updateFeature] * :param geometry: ``OpenLayers.Geometry`` New geometry for the feature. * If not provided, the existing geometry will be left unchanged. * :param mods: ``Object`` An object with one or more of ``scale``, * ``center`` and ``rotation``, reflecting the page properties to * update. * * Updates the page feature with a new geometry and notifies listeners * of changed page properties. */ updateFeature: function(geometry, mods) { var f = this.feature; var modified = f.geometry !== geometry; geometry.id = f.geometry.id; f.geometry = geometry; if(!this._updating) { for(var property in mods) { if(mods[property] === this[property]) { delete mods[property]; } else { this[property] = mods[property]; modified = true; } } Ext.apply(this, mods); f.layer && f.layer.drawFeature(f); modified && this.fireEvent("change", this, mods); } }, /** private: method[calculatePageBounds] * :param scale: ``Ext.data.Record`` Scale record to calculate the page * bounds for. * :param units: ``String`` Map units to use for the scale calculation. * Optional if ``feature`` is added to a layer which is added to a * map. If not provided, "dd" will be assumed. * :return: ``OpenLayers.Bounds`` * * Calculates the page bounds for a given scale. */ calculatePageBounds: function(scale, units) { var s = scale.get("value"); var f = this.feature; var geom = this.feature.geometry; var center = geom.getBounds().getCenterLonLat(); var size = this.printProvider.layout.get("size"); var units = units || (f.layer && f.layer.map && f.layer.map.getUnits()) || "dd"; var unitsRatio = OpenLayers.INCHES_PER_UNIT[units]; var w = size.width / 72 / unitsRatio * s / 2; var h = size.height / 72 / unitsRatio * s / 2; return new OpenLayers.Bounds(center.lon - w, center.lat - h, center.lon + w, center.lat + h); }, /** private: method[onLayoutChange] * Handler for the printProvider's layoutchange event. */ onLayoutChange: function() { if(this.printProvider.layout.get("rotation") === false) { this.setRotation(0, true); } // at init time the print provider triggers layoutchange // before loadcapabilities, i.e. before we set this.scale // to the first scale in the scales store, we need to // guard against that this.scale && this.setScale(this.scale); }, /** private: method[destroy] */ destroy: function() { this.printProvider.un("layoutchange", this.onLayoutChange, this); } }); /** FILE: 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. * */ 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: * {} An exact clone of this * */ 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 { 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 constructor, or a subclass. * * TBD 3.0 - remove reference to url in above paragraph * */ OpenLayers.Tile = OpenLayers.Class({ /** * 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: * 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 (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 . 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 * . 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 * {} 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 * {} null */ bounds: null, /** * Property: size * {} null */ size: null, /** * Property: position * {} 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 instance. * * Parameters: * layer - {} layer that the tile will go in. * position - {} * bounds - {} * url - {} * 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 and return the result from . * * 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: * {Boolean} Whether or not the tile should actually be drawn. Returns null * if a beforedraw listener returned false. */ 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; }, /** * 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; } } return withinMaxExtent || this.layer.displayOutsideMaxExtent; }, /** * Method: setBounds * Sets the bounds on this instance * * Parameters: * 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. * * Parameters: * bounds - {} * position - {} * redraw - {Boolean} Call draw method on tile after moving. * Default is true */ moveTo: function (bounds, position, redraw) { if (redraw == null) { redraw = true; } 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" }); /** FILE: 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 * constructor. * * Inherits from: * - */ 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; }, /** * 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() { this.events.triggerEvent('beforeload'); this.layer.div.appendChild(this.getTile()); this.events.triggerEvent(this._loadEvent); var img = this.getImage(); if (this.url && OpenLayers.Util.isEquivalentUrl(img.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 { 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. * * 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 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; }()); /** FILE: 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 constructor. * * Inherits from: * - */ OpenLayers.Layer.Grid = OpenLayers.Class(OpenLayers.Layer.HTTPRequest, { /** * APIProperty: tileSize * {} */ tileSize: 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". */ tileOriginCorner: "bl", /** * 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``. */ tileOrigin: null, /** APIProperty: tileOptions * {Object} optional configuration options for instances * created by this Layer, if supported by the tile class. */ tileOptions: null, /** * APIProperty: tileClass * {} The tile class to use for this layer. * Defaults is OpenLayers.Tile.Image. */ tileClass: OpenLayers.Tile.Image, /** * Property: grid * {Array(Array())} 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 layers, * 2500 for tiled layers. See 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 ), * 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. */ 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: * name - {String} * url - {String} * params - {Object} * options - {Object} Hashtable of extra options to tag onto the layer */ 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 - {} The map. */ setMap: function(map) { OpenLayers.Layer.HTTPRequest.prototype.setMap.call(this, map); OpenLayers.Element.addClass(this.div, this.className); }, /** * Method: removeMap * Called when the layer is removed from the map. * * Parameters: * 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: * 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 */ clearGrid:function() { if (this.grid) { for(var iRow=0, len=this.grid.length; iRow} 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 - {} * 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 - {} 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). */ 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; }, /** * 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; }, /** * 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); }, /** * 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; // 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'; }, /** * 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=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 - {} */ 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: * {} 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 - {} */ 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 - {|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; 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 * 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: getTileBoundsForGridIndex * * Parameters: * row - {Number} The row of the grid * col - {Number} The column of the grid * * 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 - {} */ 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 { 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} */ 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 - {} */ 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); } } 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 }); }, /** * 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 }); }, /** * 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; } } }, /** * Method: shiftRow * Shifty grid work * * 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 */ 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 rows) { var row = this.grid.pop(); for (i=0, l=row.length; i columns) { var row = this.grid[i]; var tile = row.pop(); this.destroyTile(tile); } } }, /** * 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(); } }, /** * 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" }); /** FILE: 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 * constructor. * * Inherits from: * - */ 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 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 ). * 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: * {} 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 - {} 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" }); /** FILE: GeoExt/widgets/WMSLegend.js **/ /** * Copyright (c) 2008-2012 The Open Source Geospatial Foundation * * Published under the BSD license. * See http://svn.geoext.org/core/trunk/geoext/license.txt for the full text * of the license. */ /** * @include GeoExt/widgets/LegendImage.js * @requires GeoExt/widgets/LayerLegend.js * @require OpenLayers/Util.js * @require OpenLayers/Layer/WMS.js */ /** api: (define) * module = GeoExt * class = WMSLegend */ /** api: (extends) * GeoExt/widgets/LayerLegend.js */ Ext.namespace('GeoExt'); /** api: constructor * .. class:: WMSLegend(config) * * Show a legend image for a WMS layer. The image can be read from the styles * field of a layer record (if the record comes e.g. from a * :class:`GeoExt.data.WMSCapabilitiesReader`). If not provided, a * GetLegendGraphic request will be issued to retrieve the image. */ GeoExt.WMSLegend = Ext.extend(GeoExt.LayerLegend, { /** api: config[defaultStyleIsFirst] * ``Boolean`` * The WMS spec does not say if the first style advertised for a layer in * a Capabilities document is the default style that the layer is * rendered with. We make this assumption by default. To be strictly WMS * compliant, set this to false, but make sure to configure a STYLES * param with your WMS layers, otherwise LegendURLs advertised in the * GetCapabilities document cannot be used. */ defaultStyleIsFirst: true, /** api: config[useScaleParameter] * ``Boolean`` * Should we use the optional SCALE parameter in the SLD WMS * GetLegendGraphic request? Defaults to true. */ useScaleParameter: true, /** api: config[baseParams] * ``Object`` * Optional parameters to add to the legend url, this can e.g. be used to * support vendor-specific parameters in a SLD WMS GetLegendGraphic * request. To override the default MIME type of image/gif use the * FORMAT parameter in baseParams. * * .. code-block:: javascript * * var legendPanel = new GeoExt.LegendPanel({ * map: map, * title: 'Legend Panel', * defaults: { * style: 'padding:5px', * baseParams: { * FORMAT: 'image/png', * LEGEND_OPTIONS: 'forceLabels:on' * } * } * }); */ baseParams: null, /** api: config[itemXType] * ``String`` * The xtype to be used for each item of this legend. Defaults to * `gx_legendimage`. */ itemXType: "gx_legendimage", /** private: method[initComponent] * Initializes the WMS legend. For group layers it will create multiple * image box components. */ initComponent: function() { GeoExt.WMSLegend.superclass.initComponent.call(this); var layer = this.layerRecord.getLayer(); this._noMap = !layer.map; layer.events.register("moveend", this, this.onLayerMoveend); this.update(); }, /** private: method[onLayerMoveend] * :param e: ``Object`` */ onLayerMoveend: function(e) { if ((e.zoomChanged === true && this.useScaleParameter === true) || this._noMap) { delete this._noMap; this.update(); } }, /** private: method[getLegendUrl] * :param layerName: ``String`` A sublayer. * :param layerNames: ``Array(String)`` The array of sublayers, * read from this.layerRecord if not provided. * :return: ``String`` The legend URL. * * Get the legend URL of a sublayer. */ getLegendUrl: function(layerName, layerNames) { var rec = this.layerRecord; var url; var styles = rec && rec.get("styles"); var layer = rec.getLayer(); layerNames = layerNames || [layer.params.LAYERS].join(",").split(","); var styleNames = layer.params.STYLES && [layer.params.STYLES].join(",").split(","); var idx = layerNames.indexOf(layerName); var styleName = styleNames && styleNames[idx]; // check if we have a legend URL in the record's // "styles" data field if(styles && styles.length > 0) { if(styleName) { Ext.each(styles, function(s) { url = (s.name == styleName && s.legend) && s.legend.href; return !url; }); } else { if(!styleNames && !layer.params.SLD && !layer.params.SLD_BODY) { // let's search for a style with a 'layerName' attribute Ext.each(styles, function(s) { url = (s.layerName == layerName && s.legend) && s.legend.href; return !url; }); if (!url && this.defaultStyleIsFirst === true) { url = styles[0].legend && styles[0].legend.href; } } } if (url) { url = decodeURIComponent(url); } } if(!url) { url = layer.getFullRequestString({ REQUEST: "GetLegendGraphic", WIDTH: null, HEIGHT: null, EXCEPTIONS: "application/vnd.ogc.se_xml", LAYER: layerName, LAYERS: null, STYLE: (styleName !== '') ? styleName: null, STYLES: null, SRS: null, FORMAT: null, TIME: null }); } var params = Ext.apply({}, this.baseParams); if (layer.params._OLSALT) { // update legend after a forced layer redraw params._OLSALT = layer.params._OLSALT; } var appendParams = Ext.urlEncode(params); var formatRegEx = /([&\?]?)format=[^&]*&?/i; if (formatRegEx.test(appendParams) && formatRegEx.test(url)) { url = url.replace(formatRegEx, '$1'); } url = OpenLayers.Util.urlAppend(url, appendParams); if (url.toLowerCase().indexOf("request=getlegendgraphic") != -1) { if (url.toLowerCase().indexOf("format=") == -1) { url = Ext.urlAppend(url, "FORMAT=image%2Fgif"); } // add scale parameter - also if we have the url from the record's // styles data field and it is actually a GetLegendGraphic request. if (this.useScaleParameter === true) { var scale = layer.map.getScale(); url = Ext.urlAppend(url, "SCALE=" + scale); } } return url; }, /** private: method[update] * Update the legend, adding, removing or updating * the per-sublayer box component. */ update: function() { var layer = this.layerRecord.getLayer(); // In some cases, this update function is called on a layer // that has just been removed, see ticket #238. // The following check bypass the update if map is not set. if(!(layer && layer.map)) { return; } GeoExt.WMSLegend.superclass.update.apply(this, arguments); var layerNames, layerName, i, len; layerNames = [layer.params.LAYERS].join(",").split(","); var destroyList = []; var textCmp = this.items.find(function(item){ return item.isXType('label'); }); this.items.each(function(cmp) { i = layerNames.indexOf(cmp.itemId); if(i < 0 && cmp != textCmp) { destroyList.push(cmp); } else if(cmp !== textCmp){ layerName = layerNames[i]; var newUrl = this.getLegendUrl(layerName, layerNames); if(!OpenLayers.Util.isEquivalentUrl(newUrl, cmp.url)) { cmp.setUrl(newUrl); } } }, this); for(i = 0, len = destroyList.length; i} */ 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 * {} */ size: null, /** * Property: resolution * {Float} cache of current map resolution */ resolution: null, /** * Property: 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 * ). */ featureDx: 0, /** * Constructor: OpenLayers.Renderer * * Parameters: * containerID - {} * 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 - {} * 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 - {} */ 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 - {} * style - {} * * 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 - {} Bounds of the feature * worldBounds - {} 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 - {} * style - {Object} * featureId - {} */ 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 - {} */ 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 - {} * * Returns: * {String} A feature id or undefined. */ getFeatureIdFromEvent: function(evt) {}, /** * 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} * 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 - {} 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] }; /** FILE: 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 {}, 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 - {} 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" }); /** FILE: 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 * constructor. * * Inherits from: * - */ OpenLayers.Layer.Vector = OpenLayers.Class(OpenLayers.Layer, { /** * 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 (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. */ /** * 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()} */ features: null, /** * Property: 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()} */ 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 * {} */ styleMap: null, /** * Property: strategies * {Array(})} Optional list of strategies for the layer. */ strategies: null, /** * Property: 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 * {} */ renderer: null, /** * APIProperty: rendererOptions * {Object} Options for the renderer. See {} 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: * {} 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} 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} */ 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} */ removeMap: function(map) { this.drawn = false; if(this.strategies) { var strategy, i, len; for(i=0, len=this.strategies.length; i} * 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)} * 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)} 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()} 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 - {} * 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()} */ 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: * {} 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: * {} 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} 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: * {} 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({}) 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 - {} */ 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 - {} */ preFeatureInsert: function(feature) { }, /** * APIMethod: getDataExtent * Calculates the max extent which includes all of the features. * * Returns: * {} 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 0) { var feature; for (var i=0, ii=this.layer.features.length; i= rule.minScaleDenominator); } if (rule.maxScaleDenominator) { applies = applies && (this.currentScaleDenominator < rule.maxScaleDenominator); } } return { xtype: "panel", layout: "column", border: false, hidden: !applies, bodyStyle: this.selectOnClick ? {cursor: "pointer"} : undefined, defaults: { border: false }, items: [ this.createRuleRenderer(rule), this.createRuleTitle(rule) ], listeners: { render: function(comp){ this.selectOnClick && comp.getEl().on({ click: function(comp){ this.selectRuleEntry(rule); }, scope: this }); if (this.enableDD == true) { this.addDD(comp); } }, scope: this } }; }, /** private: method[createRuleRenderer] * :arg rule: ``OpenLayers.Rule`` * :returns: ``GeoExt.FeatureRenderer`` * * Create a renderer for the rule. */ createRuleRenderer: function(rule) { var types = [this.symbolType, "Point", "Line", "Polygon"]; var type, haveType, i, len, ii; var symbolizers = rule.symbolizers; if (!symbolizers) { // TODO: remove this when OpenLayers.Symbolizer is used everywhere var symbolizer = rule.symbolizer; for (i=0, len=types.length; i targetPos) { cls = "gx-ruledrag-insert-above"; } else if (sourcePos < targetPos) { cls = "gx-ruledrag-insert-below"; } cls && target.addClass(cls); return Ext.dd.DragZone.prototype.onDragEnter.apply(this, arguments); }, onDragDrop: function(e, targetId) { panel.moveRule(ct.items.indexOf(component), ct.items.indexOf(Ext.getCmp(targetId))); return Ext.dd.DragZone.prototype.onDragDrop.apply(this, arguments); }, getDragData: function(e) { var sourceEl = e.getTarget(".x-column-inner"); if(sourceEl) { var d = sourceEl.cloneNode(true); d.id = Ext.id(); return { sourceEl: sourceEl, repairXY: Ext.fly(sourceEl).getXY(), ddel: d }; } } }); new Ext.dd.DropTarget(component.getEl(), { ddGroup: ct.id, notifyDrop: function() { return true; } }); }, /** api: method[update] * Update rule titles and symbolizers. */ update: function() { GeoExt.VectorLegend.superclass.update.apply(this, arguments); var i, ii; if (this.symbolType && this.rules) { if (this.rulesContainer.items) { var comp; for (i=this.rulesContainer.items.length-1; i>=0; --i) { comp = this.rulesContainer.getComponent(i); this.rulesContainer.remove(comp, true); } } for (i=0, ii=this.rules.length; i`_ */ Ext.namespace('GeoExt'); /** api: constructor * .. class:: FeatureRenderer(config) * * Create a box component for rendering a vector feature. */ GeoExt.FeatureRenderer = Ext.extend(Ext.BoxComponent, { /** api: config[feature] * ``OpenLayers.Feature.Vector`` * Optional vector to be drawn. If not provided, and if ``symbolizers`` * is configured with an array of plain symbolizer objects, ``symbolType`` * should be configured. */ feature: undefined, /** api: config[symbolizers] * ``Array(Object)`` * An array of ``OpenLayers.Symbolizer`` instances or plain symbolizer * objects (in painters order) for rendering a feature. If no * symbolizers are provided, the OpenLayers default will be used. If a * symbolizer is an instance of ``OpenLayers.Symbolizer``, its type will * override the symbolType for rendering. */ symbolizers: [OpenLayers.Feature.Vector.style["default"]], /** api: config[symbolType] * ``String`` * One of ``"Point"``, ``"Line"``, ``"Polygon"`` or ``"Text"``. Only * pertinent if OpenLayers.Symbolizer objects are not used. If ``feature`` * is provided, it will be preferred. The default is "Polygon". */ symbolType: "Polygon", /** api: config[labelText] * ``String`` * Label text to display for text features. */ labelText: null, /** private: property[resolution] * ``Number`` * The resolution for the renderer. */ resolution: 1, /** private: property[minWidth] * ``Number`` */ minWidth: 20, /** private: property[minHeight] * ``Number`` */ minHeight: 20, /** private: property[renderers] * ``Array(String)`` * List of supported Renderer classes. Add to this list to add support for * additional renderers. The first renderer in the list that returns * ``true`` for the ``supported`` method will be used, if not defined in * the ``renderer`` config property. */ renderers: ["SVG", "VML", "Canvas"], /** private: property[rendererOptions] * ``Object`` * Options for the renderer. See ``OpenLayers.Renderer`` for supported * options. */ rendererOptions: null, /** private: property[pointFeature] * ``OpenLayers.Feature.Vector`` * Feature with point geometry. */ pointFeature: undefined, /** private: property[lineFeature] * ``OpenLayers.Feature.Vector`` * Feature with LineString geometry. Default zig-zag is provided. */ lineFeature: undefined, /** private: property[polygonFeature] * ``OpenLayers.Feature.Vector`` * Feature with Polygon geometry. Default is a soft cornered rectangle. */ polygonFeature: undefined, /** private: property[textFeature] * ``OpenLayers.Feature.Vector`` * Feature with invisible Point geometry and text label. */ textFeature: undefined, /** private: property[renderer] * ``OpenLayers.Renderer`` */ renderer: null, /** private: method[initComponent] */ initComponent: function() { GeoExt.FeatureRenderer.superclass.initComponent.apply(this, arguments); Ext.applyIf(this, { pointFeature: new OpenLayers.Feature.Vector( new OpenLayers.Geometry.Point(0, 0) ), lineFeature: new OpenLayers.Feature.Vector( new OpenLayers.Geometry.LineString([ new OpenLayers.Geometry.Point(-8, -3), new OpenLayers.Geometry.Point(-3, 3), new OpenLayers.Geometry.Point(3, -3), new OpenLayers.Geometry.Point(8, 3) ]) ), polygonFeature: new OpenLayers.Feature.Vector( new OpenLayers.Geometry.Polygon([ new OpenLayers.Geometry.LinearRing([ new OpenLayers.Geometry.Point(-8, -4), new OpenLayers.Geometry.Point(-6, -6), new OpenLayers.Geometry.Point(6, -6), new OpenLayers.Geometry.Point(8, -4), new OpenLayers.Geometry.Point(8, 4), new OpenLayers.Geometry.Point(6, 6), new OpenLayers.Geometry.Point(-6, 6), new OpenLayers.Geometry.Point(-8, 4) ]) ]) ), textFeature: new OpenLayers.Feature.Vector( new OpenLayers.Geometry.Point(0, 0) ) }); if(!this.feature) { this.setFeature(null, {draw: false}); } this.addEvents( /** api: event[click] * Fires when the feature is clicked on. * * Listener arguments: * * * renderer - :class:`GeoExt.FeatureRenderer` This feature renderer. */ "click" ); }, /** private: method[initCustomEvents] */ initCustomEvents: function() { this.clearCustomEvents(); this.el.on("click", this.onClick, this); }, /** private: method[clearCustomEvents] */ clearCustomEvents: function() { if (this.el && this.el.removeAllListeners) { this.el.removeAllListeners(); } }, /** private: method[onClick] */ onClick: function() { this.fireEvent("click", this); }, /** private: method[onRender] */ onRender: function(ct, position) { if(!this.el) { this.el = document.createElement("div"); this.el.id = this.getId(); } if(!this.renderer || !this.renderer.supported()) { this.assignRenderer(); } // monkey-patch renderer so we always get a resolution this.renderer.map = { getResolution: (function() { return this.resolution; }).createDelegate(this) }; GeoExt.FeatureRenderer.superclass.onRender.apply(this, arguments); this.drawFeature(); }, /** private: method[afterRender] */ afterRender: function() { GeoExt.FeatureRenderer.superclass.afterRender.apply(this, arguments); this.initCustomEvents(); }, /** private: method[onResize] */ onResize: function(w, h) { this.setRendererDimensions(); GeoExt.FeatureRenderer.superclass.onResize.apply(this, arguments); }, /** private: method[setRendererDimensions] */ setRendererDimensions: function() { var gb = this.feature.geometry.getBounds(); var gw = gb.getWidth(); var gh = gb.getHeight(); /** * Determine resolution based on the following rules: * 1) always use value specified in config * 2) if not specified, use max res based on width or height of element * 3) if no width or height, assume a resolution of 1 */ var resolution = this.initialConfig.resolution; if(!resolution) { resolution = Math.max(gw / this.width || 0, gh / this.height || 0) || 1; } this.resolution = resolution; // determine height and width of element var width = Math.max(this.width || this.minWidth, gw / resolution); var height = Math.max(this.height || this.minHeight, gh / resolution); // determine bounds of renderer var center = gb.getCenterPixel(); var bhalfw = width * resolution / 2; var bhalfh = height * resolution / 2; var bounds = new OpenLayers.Bounds( center.x - bhalfw, center.y - bhalfh, center.x + bhalfw, center.y + bhalfh ); this.renderer.setSize(new OpenLayers.Size(Math.round(width), Math.round(height))); this.renderer.setExtent(bounds, true); }, /** private: method[assignRenderer] * Iterate through the available renderer implementations and selects * and assign the first one whose ``supported`` method returns ``true``. */ assignRenderer: function() { for(var i=0, len=this.renderers.length; i"); a.style.display = "none"; document.body.appendChild(a); a.href = a.href; document.body.removeChild(a); } else { a = document.createElement("a"); a.href = url; } return a.href; }, /** api: function[throttle] * :arg func: ``Function`` * :arg interval: ``Integer`` * :arg scope: ``Object`` * :return: ``Function`` * * Returns a function, that, when invoked, will only be triggered at * most once during a given window of time. */ throttle: (function() { // taken from ExtJS 4.1 // TODO remove when we upgrade to ExtJS 4.1 or higher. /** * Creates a throttled version of the passed function which, when called repeatedly and * rapidly, invokes the passed function only after a certain interval has elapsed since the * previous invocation. * * This is useful for wrapping functions which may be called repeatedly, such as * a handler of a mouse move event when the processing is expensive. * * @param {Function} fn The function to execute at a regular time interval. * @param {Number} interval The interval **in milliseconds** on which the passed function is executed. * @param {Object} scope (optional) The scope (`this` reference) in which * the passed function is executed. If omitted, defaults to the scope specified by the caller. * @returns {Function} A function which invokes the passed function at the specified interval. */ var createThrottled = function(fn, interval, scope) { var lastCallTime, elapsed, lastArgs, timer, execute = function() { fn.apply(scope || this, lastArgs); lastCallTime = new Date().getTime(); }; return function() { elapsed = new Date().getTime() - lastCallTime; lastArgs = arguments; clearTimeout(timer); if (!lastCallTime || (elapsed >= interval)) { execute(); } else { timer = setTimeout(execute, interval - elapsed); } }; }; return function(func, interval, scope) { return createThrottled(func, interval, scope); }; })(), /** api: function[md5] * :arg data: ``String`` * :returns: ``String`` md5 hash * * Encrypts the specified string using MD5. */ md5: (function() { /* md5.js - MD5 Message-Digest * Copyright (C) 1999,2002 Masanao Izumo * Version: 2.0.0 * LastModified: May 13 2002 * * This program is free software. You can redistribute it and/or modify * it without any warranty. This library calculates the MD5 based on RFC1321. * See RFC1321 for more information and algorism. */ /* Interface: * md5_128bits = MD5_hash(data); * md5_hexstr = MD5_hexhash(data); */ /* ChangeLog * 2002/05/13: Version 2.0.0 released * NOTICE: API is changed. * 2002/04/15: Bug fix about MD5 length. */ // md5_T[i] = parseInt(Math.abs(Math.sin(i)) * 4294967296.0); var MD5_T = [ 0x00000000, 0xd76aa478, 0xe8c7b756, 0x242070db, 0xc1bdceee, 0xf57c0faf, 0x4787c62a, 0xa8304613, 0xfd469501, 0x698098d8, 0x8b44f7af, 0xffff5bb1, 0x895cd7be, 0x6b901122, 0xfd987193, 0xa679438e, 0x49b40821, 0xf61e2562, 0xc040b340, 0x265e5a51, 0xe9b6c7aa, 0xd62f105d, 0x02441453, 0xd8a1e681, 0xe7d3fbc8, 0x21e1cde6, 0xc33707d6, 0xf4d50d87, 0x455a14ed, 0xa9e3e905, 0xfcefa3f8, 0x676f02d9, 0x8d2a4c8a, 0xfffa3942, 0x8771f681, 0x6d9d6122, 0xfde5380c, 0xa4beea44, 0x4bdecfa9, 0xf6bb4b60, 0xbebfbc70, 0x289b7ec6, 0xeaa127fa, 0xd4ef3085, 0x04881d05, 0xd9d4d039, 0xe6db99e5, 0x1fa27cf8, 0xc4ac5665, 0xf4292244, 0x432aff97, 0xab9423a7, 0xfc93a039, 0x655b59c3, 0x8f0ccc92, 0xffeff47d, 0x85845dd1, 0x6fa87e4f, 0xfe2ce6e0, 0xa3014314, 0x4e0811a1, 0xf7537e82, 0xbd3af235, 0x2ad7d2bb, 0xeb86d391 ]; var MD5_round1 = [ [ 0, 7, 1], [ 1,12, 2], [ 2,17, 3], [ 3,22, 4], [ 4, 7, 5], [ 5,12, 6], [ 6,17, 7], [ 7,22, 8], [ 8, 7, 9], [ 9,12,10], [10,17,11], [11,22,12], [12, 7,13], [13,12,14], [14,17,15], [15,22,16] ]; var MD5_round2 = [ [ 1, 5,17], [ 6, 9,18], [11,14,19], [ 0,20,20], [ 5, 5,21], [10, 9,22], [15,14,23], [ 4,20,24], [ 9, 5,25], [14, 9,26], [ 3,14,27], [ 8,20,28], [13, 5,29], [ 2, 9,30], [ 7,14,31], [12,20,32] ]; var MD5_round3 = [ [ 5, 4,33], [ 8,11,34], [11,16,35], [14,23,36], [ 1, 4,37], [ 4,11,38], [ 7,16,39], [10,23,40], [13, 4,41], [ 0,11,42], [ 3,16,43], [ 6,23,44], [ 9, 4,45], [12,11,46], [15,16,47], [ 2,23,48] ]; var MD5_round4 = [ [ 0, 6,49], [ 7,10,50], [14,15,51], [ 5,21,52], [12, 6,53], [ 3,10,54], [10,15,55], [ 1,21,56], [ 8, 6,57], [15,10,58], [ 6,15,59], [13,21,60], [ 4, 6,61], [11,10,62], [ 2,15,63], [ 9,21,64] ]; function MD5_F(x, y, z) { return (x & y) | (~x & z); } function MD5_G(x, y, z) { return (x & z) | (y & ~z); } function MD5_H(x, y, z) { return x ^ y ^ z; } function MD5_I(x, y, z) { return y ^ (x | ~z); } var MD5_round = [ [MD5_F, MD5_round1], [MD5_G, MD5_round2], [MD5_H, MD5_round3], [MD5_I, MD5_round4] ]; function MD5_pack(n32) { return String.fromCharCode(n32 & 0xff) + String.fromCharCode((n32 >>> 8) & 0xff) + String.fromCharCode((n32 >>> 16) & 0xff) + String.fromCharCode((n32 >>> 24) & 0xff); } function MD5_unpack(s4) { return s4.charCodeAt(0) | (s4.charCodeAt(1) << 8) | (s4.charCodeAt(2) << 16) | (s4.charCodeAt(3) << 24); } function MD5_number(n) { while (n < 0) { n += 4294967296; } while (n > 4294967295) { n -= 4294967296; } return n; } function MD5_apply_round(x, s, f, abcd, r) { var a, b, c, d; var kk, ss, ii; var t, u; a = abcd[0]; b = abcd[1]; c = abcd[2]; d = abcd[3]; kk = r[0]; ss = r[1]; ii = r[2]; u = f(s[b], s[c], s[d]); t = s[a] + u + x[kk] + MD5_T[ii]; t = MD5_number(t); t = ((t<>>(32-ss))); t += s[b]; s[a] = MD5_number(t); } function MD5_hash(data) { var abcd, x, state, s; var len, index, padLen, f, r; var i, j, k; var tmp; state = [0x67452301, 0xefcdab89, 0x98badcfe, 0x10325476]; len = data.length; index = len & 0x3f; padLen = (index < 56) ? (56 - index) : (120 - index); if(padLen > 0) { data += "\x80"; for(i = 0; i < padLen - 1; i++) { data += "\x00"; } } data += MD5_pack(len * 8); data += MD5_pack(0); len += padLen + 8; abcd = [0, 1, 2, 3]; x = [16]; s = [4]; for(k = 0; k < len; k += 64) { for(i = 0, j = k; i < 16; i++, j += 4) { x[i] = data.charCodeAt(j) | (data.charCodeAt(j + 1) << 8) | (data.charCodeAt(j + 2) << 16) | (data.charCodeAt(j + 3) << 24); } for(i = 0; i < 4; i++) { s[i] = state[i]; } for(i = 0; i < 4; i++) { f = MD5_round[i][0]; r = MD5_round[i][1]; for(j = 0; j < 16; j++) { MD5_apply_round(x, s, f, abcd, r[j]); tmp = abcd[0]; abcd[0] = abcd[3]; abcd[3] = abcd[2]; abcd[2] = abcd[1]; abcd[1] = tmp; } } for(i = 0; i < 4; i++) { state[i] += s[i]; state[i] = MD5_number(state[i]); } } return MD5_pack(state[0]) + MD5_pack(state[1]) + MD5_pack(state[2]) + MD5_pack(state[3]); } function MD5_hexhash(data) { var i, out, c; var bit128; bit128 = MD5_hash(data); out = ""; for(i = 0; i < 16; i++) { c = bit128.charCodeAt(i); out += "0123456789abcdef".charAt((c>>4) & 0xf); out += "0123456789abcdef".charAt(c & 0xf); } return out; } return function(data) { return MD5_hexhash(data); }; })() }; /** FILE: menu/LayerMenu.js **/ /** * Copyright (c) 2008-2011 The Open Planning Project * * Published under the GPL license. * See https://github.com/opengeo/gxp/raw/master/license.txt for the full text * of the license. */ /** api: (define) * module = gxp.menu * class = LayerMenu * base_link = `Ext.menu.Menu `_ */ Ext.namespace("gxp.menu"); /** api: constructor * .. class:: LayerMenu(config) * * A menu to control layer visibility. */ gxp.menu.LayerMenu = Ext.extend(Ext.menu.Menu, { /** api: config[layerText] * ``String`` * Text for added layer (i18n). */ layerText: "Layer", /** api: config[layers] * ``GeoExt.data.LayerStore`` * The store containing layer records to be viewed in this menu. */ layers: null, /** private: method[initComponent] * Private method called to initialize the component. */ initComponent: function() { gxp.menu.LayerMenu.superclass.initComponent.apply(this, arguments); this.layers.on("add", this.onLayerAdd, this); this.onLayerAdd(); }, /** private: method[onRender] * Private method called during the render sequence. */ onRender : function(ct, position) { gxp.menu.LayerMenu.superclass.onRender.apply(this, arguments); }, /** private: method[beforeDestroy] * Private method called during the destroy sequence. */ beforeDestroy: function() { if (this.layers && this.layers.on) { this.layers.un("add", this.onLayerAdd, this); } delete this.layers; gxp.menu.LayerMenu.superclass.beforeDestroy.apply(this, arguments); }, /** private: method[onLayerAdd] * Listener called when records are added to the layer store. */ onLayerAdd: function() { this.removeAll(); // this.getEl().addClass("gxp-layer-menu"); // this.getEl().applyStyles({ // width: '', // height: '' // }); this.add( { iconCls: "gxp-layer-visibility", text: this.layerText, canActivate: false }, "-" ); this.layers.each(function(record) { var layer = record.getLayer(); if(layer.displayInLayerSwitcher) { var item = new Ext.menu.CheckItem({ text: record.get("title"), checked: record.getLayer().getVisibility(), group: record.get("group"), listeners: { checkchange: function(item, checked) { record.getLayer().setVisibility(checked); } } }); if (this.items.getCount() > 2) { this.insert(2, item); } else { this.add(item); } } }, this); } }); Ext.reg('gxp_layermenu', gxp.menu.LayerMenu); /** FILE: widgets/CrumbPanel.js **/ /** * Copyright (c) 2008-2011 The Open Planning Project * * Published under the GPL license. * See https://github.com/opengeo/gxp/raw/master/license.txt for the full text * of the license. */ Ext.namespace("gxp"); /** api: (define) * module = gxp * class = CrumbPanel * base_link = `Ext.TabPanel `_ */ /** api: constructor * .. class:: CrumbPanel(config) * * Panel that accommodates modal dialogs and displays their hierarchy as * crumbs in tabs, from left to right. Clicking on a crumb left of the * rightmost one will close all dialogs that are hosted in tabs on its * right. * * If a component has a ``shortTitle`` configured, it will be used instead * of the ``title`` in the crumb path. * * Components intended to be reused after being removed from this panel need * the closeAction option set to "hide", like an ``Ext.Window``. */ gxp.CrumbPanel = Ext.extend(Ext.TabPanel, { /** private: property[widths] * {Object} widths of the panel's items, keyed by item id, so they can be * restored after closing an item that required a wider container. */ widths: null, enableTabScroll: true, /** private: method[initComponent] */ initComponent: function() { gxp.CrumbPanel.superclass.initComponent.apply(this, arguments); this.widths = {}; }, /** private: method[onBeforeAdd] * :arg cmp: ``Ext.Component`` */ onBeforeAdd: function(cmp) { gxp.CrumbPanel.superclass.onBeforeAdd.apply(this, arguments); if (cmp.shortTitle) { cmp.title = cmp.shortTitle; } }, /** private: method[onAdd] * :arg cmp: ``Ext.Component`` */ onAdd: function(cmp) { gxp.CrumbPanel.superclass.onAdd.apply(this, arguments); this.setActiveTab(this.items.getCount() - 1); cmp.on("hide", this.onCmpHide, this); //TODO investigate why hidden components are displayed again when // another crumb is activated - this just works around the issue cmp.getEl().dom.style.display = ""; }, /** private: method[onRemove] * :arg cmp: ``Ext.Component`` */ onRemove: function(cmp) { gxp.CrumbPanel.superclass.onRemove.apply(this, arguments); cmp.un("hide", this.onCmpHide, this); var previousWidth = this.widths[this.get(this.items.getCount()-1).id]; if (previousWidth && previousWidth < this.getWidth()) { this.setWidth(previousWidth); if (this.ownerCt) { this.ownerCt.syncSize(); } } //TODO investigate why hidden components are displayed again when // another crumb is activated - this just works around the issue cmp.getEl().dom.style.display = "none"; this.activeTab.doLayout(); }, /** private: method[onRender] * :arg cmp: ``Ext.Component`` */ onRender: function(cmp) { if (!this.initialConfig.itemTpl) { this.itemTpl = new Ext.Template( '
  • \u00BB
    ', '', '{text}', '
  • ' ); } gxp.CrumbPanel.superclass.onRender.apply(this, arguments); this.getEl().down("div").addClass("gxp-crumbpanel-header"); }, /** private: method[onCmpHide] * :arg cmp: ``Ext.Component`` The component that was hidden. * * Listener for the "hide" event of components that were added to the * CrumbPanel. */ onCmpHide: function(cmp) { var lastIndex = this.items.getCount() - 1; if (this.items.indexOf(cmp) === lastIndex) { this.setActiveTab(this.get(--lastIndex)); } }, /** private: method[setActiveTab] * :arg item: ``Number|Ext.Component`` */ setActiveTab: function(item) { var index; if (Ext.isNumber(item)) { index = item; item = this.get(index); } else { index = this.items.indexOf(item); } if (~index) { var cmp, i; for (i=this.items.getCount()-1; i>index; --i) { cmp = this.get(i); // remove, but don't destroy if component was configured with // {closeAction: "hide"} this.remove(cmp, cmp.closeAction !== "hide"); } } var width = item.initialConfig.minWidth || item.initialConfig.width, previousWidth = this.getWidth(); if (width > previousWidth) { this.widths[this.get(index - 1).id] = previousWidth; this.setWidth(width); if (this.ownerCt) { this.ownerCt.syncSize(); } } gxp.CrumbPanel.superclass.setActiveTab.apply(this, arguments); } }); /** api: xtype = gxp_crumbpanel */ Ext.reg('gxp_crumbpanel', gxp.CrumbPanel); /** FILE: widgets/LayerUploadPanel.js **/ /** * Copyright (c) 2008-2011 The Open Planning Project * * Published under the GPL license. * See https://github.com/opengeo/gxp/raw/master/license.txt for the full text * of the license. */ /** api: (define) * module = gxp * class = LayerUploadPanel * base_link = `Ext.FormPanel `_ */ Ext.namespace("gxp"); /** api: constructor * .. class:: LayerUploadPanel(config) * * A panel for uploading new layer data to GeoServer. */ gxp.LayerUploadPanel = Ext.extend(Ext.FormPanel, { /** i18n */ titleLabel: "Title", titleEmptyText: "Layer title", abstractLabel: "Description", abstractEmptyText: "Layer description", fileLabel: "Data", fieldEmptyText: "Browse for data archive...", uploadText: "Upload", uploadFailedText: "Upload failed", processingUploadText: "Processing upload...", waitMsgText: "Uploading your data...", invalidFileExtensionText: "File extension must be one of: ", optionsText: "Options", workspaceLabel: "Workspace", workspaceEmptyText: "Default workspace", dataStoreLabel: "Store", dataStoreEmptyText: "Choose a store", dataStoreNewText: "Create new store", crsLabel: "CRS", crsEmptyText: "Coordinate Reference System ID", invalidCrsText: "CRS identifier should be an EPSG code (e.g. EPSG:4326)", /** private: property[fileUpload] * ``Boolean`` */ fileUpload: true, /** api: config[validFileExtensions] * ``Array`` * List of valid file extensions. These will be used in validating the * file input value. Default is ``[".zip", ".tif", ".tiff", ".gz", ".tar.bz2", * ".tar", ".tgz", ".tbz2"]``. */ validFileExtensions: [".zip", ".tif", ".tiff", ".gz", ".tar.bz2", ".tar", ".tgz", ".tbz2"], /** api: config[url] * ``String`` * URL for GeoServer RESTConfig root. E.g. "http://example.com/geoserver/rest". */ /** private: property[defaultDataStore] * ``string`` */ defaultDataStore: null, /** private: method[constructor] */ constructor: function(config) { // Allow for a custom method to handle upload responses. config.errorReader = { read: config.handleUploadResponse || this.handleUploadResponse.createDelegate(this) }; gxp.LayerUploadPanel.superclass.constructor.call(this, config); }, /** private: property[selectedWorkspace] * {Ext.data.Record} */ selectedWorkspace: null, /** private: method[initComponent] */ initComponent: function() { this.items = [{ xtype: "textfield", name: "title", fieldLabel: this.titleLabel, emptyText: this.titleEmptyText, allowBlank: true }, { xtype: "textarea", name: "abstract", fieldLabel: this.abstractLabel, emptyText: this.abstractEmptyText, allowBlank: true }, { xtype: "fileuploadfield", id: "file", anchor: "90%", emptyText: this.fieldEmptyText, fieldLabel: this.fileLabel, name: "file", buttonText: "", buttonCfg: { iconCls: "gxp-icon-filebrowse" }, listeners: { "fileselected": function(cmp, value) { // remove the path from the filename - avoids C:/fakepath etc. cmp.setValue(value.split(/[/\\]/).pop()); } }, validator: this.fileNameValidator.createDelegate(this) }, { xtype: "fieldset", ref: "optionsFieldset", title: this.optionsText, checkboxToggle: true, collapsed: true, hidden: this.workspace != undefined && this.store != undefined && this.crs != undefined, hideMode: "offsets", defaults: { anchor: "97%" }, items: [ this.createWorkspacesCombo(), this.createDataStoresCombo(), { xtype: "textfield", name: "nativeCRS", // anchor: "90%", fieldLabel: this.crsLabel, emptyText: this.crsEmptyText, allowBlank: true, regex: /^epsg:\d+$/i, regexText: this.invalidCrsText } ], listeners: { collapse: function(fieldset) { // reset all combos fieldset.items.each(function(item) { item.reset(); }); } } }]; this.buttons = [{ text: this.uploadText, handler: function() { var form = this.getForm(); if (form.isValid()) { var fields = form.getFieldValues(), jsonData = {'import': {}}; if (fields.workspace) { jsonData["import"].targetWorkspace = {workspace: {name: fields.workspace}}; } if (Ext.isEmpty(fields.store) && this.defaultDataStore) { jsonData["import"].targetStore = {dataStore: {name: this.defaultDataStore}}; } else if (!Ext.isEmpty(fields.store) && fields.store !== this.dataStoreNewText) { jsonData["import"].targetStore = {dataStore: {name: fields.store}}; } Ext.Ajax.request({ url: this.getUploadUrl(), method: "POST", jsonData: jsonData, success: function(response) { this._import = response.getResponseHeader("Location"); this.optionsFieldset.expand(); form.submit({ url: this._import + "/tasks?expand=all", waitMsg: this.waitMsgText, waitMsgTarget: true, reset: true, scope: this }); }, scope: this }); } }, scope: this }]; this.addEvents( /** * Event: workspaceselected * Fires when a workspace is selected. * * Listener arguments: * panel - {' + this.validFileExtensions.join(", "); }, /** private: method[createWorkspacesCombo] * :returns: ``Object`` Combo config. */ createWorkspacesCombo: function() { return { xtype: "combo", name: "workspace", ref: "../workspace", fieldLabel: this.workspaceLabel, store: new Ext.data.JsonStore({ url: this.getWorkspacesUrl(), autoLoad: true, root: "workspaces.workspace", fields: ["name", "href"] }), displayField: "name", valueField: "name", mode: "local", allowBlank: true, triggerAction: "all", forceSelection: true, listeners: { select: function(combo, record, index) { this.getDefaultDataStore(record.get('name')); this.fireEvent("workspaceselected", this, record); }, scope: this } }; }, /** private: method[createDataStoresCombo] * :returns: ``Ext.form.ComboBox`` */ createDataStoresCombo: function() { // this store will be loaded whenever a workspace is selected var store = new Ext.data.JsonStore({ autoLoad: false, root: "dataStores.dataStore", fields: ["name", "href"] }); this.on({ workspaceselected: function(panel, record) { combo.reset(); var workspaceUrl = record.get("href"); store.removeAll(); store.proxy = new Ext.data.HttpProxy({ url: workspaceUrl.split(".json").shift() + "/datastores.json" }); store.proxy.on('loadexception', addDefault, this); store.load(); }, scope: this }); var addDefault = function() { var defaultData = { name: this.dataStoreNewText }; var r = new store.recordType(defaultData); store.insert(0, r); store.proxy && store.proxy.un('loadexception', addDefault, this); }; store.on('load', addDefault, this); var combo = new Ext.form.ComboBox({ name: "store", ref: "../dataStore", emptyText: this.dataStoreEmptyText, fieldLabel: this.dataStoreLabel, store: store, displayField: "name", valueField: "name", mode: "local", allowBlank: true, triggerAction: "all", forceSelection: true, listeners: { select: function(combo, record, index) { this.fireEvent("datastoreselected", this, record); }, scope: this } }); return combo; }, getDefaultDataStore: function(workspace) { Ext.Ajax.request({ url: this.url + '/workspaces/' + workspace + '/datastores/default.json', callback: function(options, success, response) { this.defaultDataStore = null; if (response.status === 200) { var json = Ext.decode(response.responseText); if (workspace === 'default' && json.dataStore && json.dataStore.workspace) { this.workspace.setValue(json.dataStore.workspace.name); var store = this.workspace.store; var data = { name: json.dataStore.workspace.name, href: json.dataStore.workspace.href }; var r = new store.recordType(data); this.fireEvent("workspaceselected", this, r); } //TODO Revisit this logic - currently we assume that stores // with the substring "file" in the type are file based, // and for file-based data stores we want to crate a new // store. if (json.dataStore && json.dataStore.enabled === true && !/file/i.test(json.dataStore.type)) { this.defaultDataStore = json.dataStore.name; this.dataStore.setValue(this.defaultDataStore); } } }, scope: this }); }, /** private: method[getUploadUrl] */ getUploadUrl: function() { return this.url + "/imports"; }, /** private: method[getWorkspacesUrl] */ getWorkspacesUrl: function() { return this.url + "/workspaces.json"; }, /** private: method[handleUploadResponse] * TODO: if response includes errors object, this can be removed * Though it should only be removed if the server always returns text/html! */ handleUploadResponse: function(response) { var obj = this.parseResponseText(response.responseText), records, tasks, task, msg, i, formData = this.getForm().getFieldValues(), success = !!obj; if (obj) { if (typeof obj === "string") { success = false; msg = obj; } else { tasks = obj.tasks || [obj.task]; if (tasks.length === 0) { success = false; msg = "Upload contains no suitable files."; } else { for (i=tasks.length-1; i>=0; --i) { task = tasks[i]; if (!task) { success = false; msg = "Unknown upload error"; } else if (task.state === 'NO_FORMAT') { success = false; msg = "Upload contains no suitable files."; } else if (task.state === 'NO_CRS' && !formData.nativeCRS) { success = false; msg = "Coordinate Reference System (CRS) of source file " + task.data.file + " could not be determined. Please specify manually."; } } } } } if (!success) { // mark the file field as invlid records = [{data: {id: "file", msg: msg || this.uploadFailedText}}]; } else { var itemModified = !!(formData.title || formData["abstract"] || formData.nativeCRS); // do not do this for coverages see https://github.com/boundlessgeo/suite/issues/184 if (itemModified && tasks[0].target.dataStore) { this.waitMsg = new Ext.LoadMask((this.ownerCt || this).getEl(), {msg: this.processingUploadText}); this.waitMsg.show(); // for now we only support a single task var payload = { title: formData.title || undefined, "abstract": formData["abstract"] || undefined, srs: formData.nativeCRS || undefined }; Ext.Ajax.request({ method: "PUT", url: tasks[0].layer.href, jsonData: payload, success: this.finishUpload, failure: function(response) { if (this.waitMsg) { this.waitMsg.hide(); } var errors = []; try { var json = Ext.decode(response.responseText); if (json.errors) { for (var i=0, ii=json.errors.length; i element (given non text/html response type). */ parseResponseText: function(text) { var obj; try { obj = Ext.decode(text); } catch (err) { // if response type was text/plain, the text will be wrapped in a
                var match = text.match(/^\s*]*>(.*)<\/pre>\s*/);
                if (match) {
                    try {
                        obj = Ext.decode(match[1]);
                    } catch (err) {
                        obj = match[1];
                    }
                }
            }
            return obj;
        },
        
        /** private: method[handleUploadSuccess]
         */
        handleUploadSuccess: function(response) {
            Ext.Ajax.request({
                method: "GET",
                url: this._import + '?expand=all',
                failure: this.handleFailure,
                success: function(response) {
                    if (this.waitMsg) {
                        this.waitMsg.hide();
                    }
                    this.getForm().reset();
                    var details = Ext.decode(response.responseText);
                    this.fireEvent("uploadcomplete", this, details);
                    delete this._import;
                },
                scope: this
            });
        },
        
        /** private: method[handleFailure]
         */
        handleFailure: function(response) {
            // see http://stackoverflow.com/questions/10046972/msie-returns-status-code-of-1223-for-ajax-request
            if (response && response.status === 1223) {
                this.handleUploadSuccess(response);
            } else {
                if (this.waitMsg) {
                    this.waitMsg.hide();
                }
                this.getForm().markInvalid([{file: this.uploadFailedText}]);
            }
        }
    
    });
    
    /** api: xtype = gxp_layeruploadpanel */
    Ext.reg("gxp_layeruploadpanel", gxp.LayerUploadPanel);
    
    /** FILE: OpenLayers/Format/WKT.js **/
    /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for
     * full list of contributors). Published 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
     * @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.WKT
     * Class for reading and writing Well-Known Text.  Create a new instance
     * with the  constructor.
     * 
     * Inherits from:
     *  - 
     */
    OpenLayers.Format.WKT = OpenLayers.Class(OpenLayers.Format, {
        
        /**
         * Constructor: OpenLayers.Format.WKT
         * Create a new parser for WKT
         *
         * Parameters:
         * options - {Object} An optional object whose properties will be set on
         *           this instance
         *
         * Returns:
         * {} A new WKT parser.
         */
        initialize: function(options) {
            this.regExes = {
                'typeStr': /^\s*(\w+)\s*\(\s*(.*)\s*\)\s*$/,
                'spaces': /\s+/,
                'parenComma': /\)\s*,\s*\(/,
                'doubleParenComma': /\)\s*\)\s*,\s*\(\s*\(/,  // can't use {2} here
                'trimParens': /^\s*\(?(.*?)\)?\s*$/
            };
            OpenLayers.Format.prototype.initialize.apply(this, [options]);
        },
    
        /**
         * APIMethod: read
         * Deserialize a WKT string and return a vector feature or an
         * array of vector features.  Supports WKT for POINT, MULTIPOINT,
         * LINESTRING, MULTILINESTRING, POLYGON, MULTIPOLYGON, and
         * GEOMETRYCOLLECTION.
         *
         * Parameters:
         * wkt - {String} A WKT string
         *
         * Returns:
         * {|Array} A feature or array of features for
         * GEOMETRYCOLLECTION WKT.
         */
        read: function(wkt) {
            var features, type, str;
            wkt = wkt.replace(/[\n\r]/g, " ");
            var matches = this.regExes.typeStr.exec(wkt);
            if(matches) {
                type = matches[1].toLowerCase();
                str = matches[2];
                if(this.parse[type]) {
                    features = this.parse[type].apply(this, [str]);
                }
                if (this.internalProjection && this.externalProjection) {
                    if (features && 
                        features.CLASS_NAME == "OpenLayers.Feature.Vector") {
                        features.geometry.transform(this.externalProjection,
                                                    this.internalProjection);
                    } else if (features &&
                               type != "geometrycollection" &&
                               typeof features == "object") {
                        for (var i=0, len=features.length; i|Array} A feature or array of
         *            features
         *
         * Returns:
         * {String} The WKT string representation of the input geometries
         */
        write: function(features) {
            var collection, geometry, isCollection;
            if (features.constructor == Array) {
                collection = features;
                isCollection = true;
            } else {
                collection = [features];
                isCollection = false;
            }
            var pieces = [];
            if (isCollection) {
                pieces.push('GEOMETRYCOLLECTION(');
            }
            for (var i=0, len=collection.length; i0) {
                    pieces.push(',');
                }
                geometry = collection[i].geometry;
                pieces.push(this.extractGeometry(geometry));
            }
            if (isCollection) {
                pieces.push(')');
            }
            return pieces.join('');
        },
    
        /**
         * Method: extractGeometry
         * Entry point to construct the WKT for a single Geometry object.
         *
         * Parameters:
         * geometry - {}
         *
         * Returns:
         * {String} A WKT string of representing the geometry
         */
        extractGeometry: function(geometry) {
            var type = geometry.CLASS_NAME.split('.')[2].toLowerCase();
            if (!this.extract[type]) {
                return null;
            }
            if (this.internalProjection && this.externalProjection) {
                geometry = geometry.clone();
                geometry.transform(this.internalProjection, this.externalProjection);
            }                       
            var wktType = type == 'collection' ? 'GEOMETRYCOLLECTION' : type.toUpperCase();
            var data = wktType + '(' + this.extract[type].apply(this, [geometry]) + ')';
            return data;
        },
        
        /**
         * Object with properties corresponding to the geometry types.
         * Property values are functions that do the actual data extraction.
         */
        extract: {
            /**
             * Return a space delimited string of point coordinates.
             * @param {OpenLayers.Geometry.Point} point
             * @returns {String} A string of coordinates representing the point
             */
            'point': function(point) {
                return point.x + ' ' + point.y;
            },
    
            /**
             * Return a comma delimited string of point coordinates from a multipoint.
             * @param {OpenLayers.Geometry.MultiPoint} multipoint
             * @returns {String} A string of point coordinate strings representing
             *                  the multipoint
             */
            'multipoint': function(multipoint) {
                var array = [];
                for(var i=0, len=multipoint.components.length; i
             * @param {OpenLayers.Geometry.Collection} collection
             * @returns {String} internal WKT representation of the collection
             */
            'collection': function(collection) {
                var array = [];
                for(var i=0, len=collection.components.length; i objects.  Write 
     *      objects to get CQL strings. Create a new parser with 
     *     the  constructor.
     *
     * Inherits from:
     *  - 
     */
    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 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:
             * {} 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 - {} 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"
    
        });
    })();
    
    
    /** FILE: widgets/FilterBuilder.js **/
    /**
     * Copyright (c) 2008-2011 The Open Planning Project
     * 
     * Published under the GPL license.
     * See https://github.com/opengeo/gxp/raw/master/license.txt for the full text
     * of the license.
     */
    
    /**
     * @include widgets/form/FilterField.js
     */
    
    /** api: (define)
     *  module = gxp
     *  class = FilterBuilder
     *  base_link = `Ext.Panel `_
     */
    Ext.namespace("gxp");
    
    /** api: constructor
     *  .. class:: FilterBuilder(config)
     *   
     *      Create a panel for assembling a filter.
     */
    gxp.FilterBuilder = Ext.extend(Ext.Container, {
    
        /** api: config[builderTypeNames]
         *  ``Array``
         *  A list of labels that correspond to builder type constants.
         *  These will be the option names available in the builder type combo.
         *  Default is ``["any", "all", "none", "not all"]``.
         */
        builderTypeNames: ["any", "all", "none", "not all"],
        
        /** api: config[allowedBuilderTypes]
         *  ``Array``
         *  List of builder type constants.  Default is
         *  ``[ANY_OF, ALL_OF, NONE_OF]``.
         */
        allowedBuilderTypes: null,
        
        /** api: config[allowBlank]
         *  ``Boolean`` Do we allow blank FilterFields? It is safe to say true
         *  here, but for compatibility reasons with old applications, the default
         *  is false.
         */
        allowBlank: false,
        
        /** api: config[caseInsensitiveMatch]
         *  ``Boolean``
         *  Should Comparison Filters for Strings do case insensitive matching?  Default is ``"false"``.
         */
        caseInsensitiveMatch: false,
    
        /** api: config[preComboText]
         *  ``String``
         *  String to display before filter type combo.  Default is ``"Match"``.
         */
        preComboText: "Match",
    
        /** api: config[postComboText]
         *  ``String``
         *  String to display after filter type combo.  Default is
         *  ``"of the following:"``.
         */
        postComboText: "of the following:",
    
        /** api: config[cls]
         *  ``String``
         *  The CSS class to be added to this panel's element (defaults to
         *  ``"gxp-filterbuilder"``).
         */
        cls: "gxp-filterbuilder",
        
        /** api: config[filter]
         *  ``OpenLayers.Filter``
         *  Filter to initialize the component with
         */
    
        /** private: property[builderType]
         */
        builderType: null,
    
        /** private: property[childFilterContainer]
         */
        childFilterContainer: null,
        
        /** private: property[customizeFilterOnInit]
         */
        customizeFilterOnInit: true,
        
        /** i18n */
        addConditionText: "add condition",
        addGroupText: "add group",
        removeConditionText: "remove condition",
    
        /** api: config[allowGroups]
         *  ``Boolean``
         *  Allow groups of conditions to be added.  Default is ``true``.
         *  If ``false``, only individual conditions (non-logical filters) can
         *  be added.
         */
        allowGroups: true,
    
        initComponent: function() {
            var defConfig = {
                defaultBuilderType: gxp.FilterBuilder.ANY_OF
            };
            Ext.applyIf(this, defConfig);
            
            if(this.customizeFilterOnInit) {
                this.filter = this.customizeFilter(this.filter);
            }
            
            this.builderType = this.getBuilderType();
            
            this.items = [{
                xtype: "container",
                layout: "form",
                ref: "form",
                defaults: {anchor: "100%"},
                hideLabels: true,
                items: [{
                    xtype: "compositefield",
                    style: "padding-left: 2px",
                    items: [{
                        xtype: "label",
                        style: "padding-top: 0.3em",
                        text: this.preComboText
                    }, this.createBuilderTypeCombo(), {
                        xtype: "label",
                        style: "padding-top: 0.3em",
                        text: this.postComboText
                    }]
                }, this.createChildFiltersPanel(), {
                    xtype: "toolbar",
                    items: this.createToolBar()
                }]
            
            }];
            
            this.addEvents(
                /**
                 * Event: change
                 * Fires when the filter changes.
                 *
                 * Listener arguments:
                 * builder - {gxp.FilterBuilder} This filter builder.  Call
                 *     ``getFilter`` to get the updated filter.
                 */
                "change"
            ); 
    
            gxp.FilterBuilder.superclass.initComponent.call(this);
        },
    
        /** private: method[createToolBar]
         */
        createToolBar: function() {
            var bar = [{
                text: this.addConditionText,
                iconCls: "add",
                handler: function() {
                    this.addCondition();
                },
                scope: this
            }];
            if(this.allowGroups) {
                bar.push({
                    text: this.addGroupText,
                    iconCls: "add",
                    handler: function() {
                        this.addCondition(true);
                    },
                    scope: this
                });
            }
            return bar;
        },
        
        /** api: method[getFilter]
         *  :return: ``OpenLayers.Filter``
         *  
         *  Returns a filter that fits the model in the Filter Encoding
         *  specification.  Use this method instead of directly accessing
         *  the ``filter`` property.  Return will be ``false`` if any child
         *  filter does not have a type, property, or value.
         */
        getFilter: function() {
            var filter;
            if(this.filter) {
                filter = this.filter.clone();
                if(filter instanceof OpenLayers.Filter.Logical) {
                    filter = this.cleanFilter(filter);
                }
            }
            return filter;
        },
        
        /** private: method[cleanFilter]
         *  :arg filter: ``OpenLayers.Filter.Logical``
         *  :return: ``OpenLayers.Filter`` An equivalent filter to the input, where
         *      all binary logical filters have more than one child filter.  Returns
         *      false if a filter doesn't have non-null type, property, or value.
         *  
         *  Ensures that binary logical filters have more than one child.
         */
        cleanFilter: function(filter) {
            if(filter instanceof OpenLayers.Filter.Logical) {
                if(filter.type !== OpenLayers.Filter.Logical.NOT &&
                   filter.filters.length === 1) {
                    filter = this.cleanFilter(filter.filters[0]);
                } else {
                    var child;
                    for(var i=0, len=filter.filters.length; i method to return a
         *  filter that meets the encoding spec.
         */
        customizeFilter: function(filter) {
            if(!filter) {
                filter = this.wrapFilter(this.createDefaultFilter());
            } else {
                filter = this.cleanFilter(filter);
                var child, i, len;
                switch(filter.type) {
                    case OpenLayers.Filter.Logical.AND:
                    case OpenLayers.Filter.Logical.OR:
                        if(!filter.filters || filter.filters.length === 0) {
                            // give the filter children if it has none
                            filter.filters = [this.createDefaultFilter()];
                        } else {
                            for(i=0, len=filter.filters.length; i 0) {
                                        filter = this.customizeFilter(child.filters[0]);
                                    } else {
                                        filter = this.wrapFilter(this.createDefaultFilter());
                                    }
                                }
                            } else {
                                // non-logical child of NOT should be wrapped
                                var type;
                                if(this.defaultBuilderType === gxp.FilterBuilder.NOT_ALL_OF) {
                                    type = OpenLayers.Filter.Logical.AND;
                                } else {
                                    type = OpenLayers.Filter.Logical.OR;
                                }
                                filter.filters = [
                                    new OpenLayers.Filter.Logical({
                                        type: type,
                                        filters: [child]
                                    })
                                ];
                            }
                        }
                        break;
                    default:
                        // non-logical filters get wrapped
                        filter = this.wrapFilter(filter);
                        break;
                }
            }
            return filter;
        },
    
        createDefaultFilter: function() {
            return new OpenLayers.Filter.Comparison({
                                matchCase: !this.caseInsensitiveMatch});
        },
        
        /** private: method[wrapFilter]
         *  :arg filter: ``OpenLayers.Filter`` A non-logical filter.
         *  :return: ``OpenLayers.Filter`` A wrapped version of the input filter.
         *  
         *  Given a non-logical filter, this creates parent filters depending on
         *  the ``defaultBuilderType``.
         */
        wrapFilter: function(filter) {
            var type;
            if(this.defaultBuilderType === gxp.FilterBuilder.ALL_OF) {
                type = OpenLayers.Filter.Logical.AND;
            } else {
                type = OpenLayers.Filter.Logical.OR;
            }
            return new OpenLayers.Filter.Logical({
                type: OpenLayers.Filter.Logical.OR,
                filters: [
                    new OpenLayers.Filter.Logical({
                        type: type, filters: [filter]
                    })
                ]
            });
        },
        
        /** private: method[addCondition]
         *  Add a new condition or group of conditions to the builder.  This
         *  modifies the filter and adds a panel representing the new condition
         *  or group of conditions.
         */
        addCondition: function(group) {
            var filter, type;
            if(group) {
                type = "gxp_filterbuilder";
                filter = this.wrapFilter(this.createDefaultFilter());
            } else {
                type = "gxp_filterfield";
                filter = this.createDefaultFilter();
            }
            var newChild = this.newRow({
                xtype: type,
                filter: filter,
                columnWidth: 1,
                attributes: this.attributes,
                allowBlank: group ? undefined : this.allowBlank,
                customizeFilterOnInit: group && false,
                caseInsensitiveMatch: this.caseInsensitiveMatch,
                listeners: {
                    change: function() {
                        this.fireEvent("change", this);
                    },
                    scope: this
                }
            });
            this.childFilterContainer.add(newChild);
            this.filter.filters[0].filters.push(filter);
            this.childFilterContainer.doLayout();
        },
        
        /** private: method[removeCondition]
         *  Remove a condition or group of conditions from the builder.  This
         *  modifies the filter and removes the panel representing the condition
         *  or group of conditions.
         */
        removeCondition: function(item, filter) {
            var parent = this.filter.filters[0].filters;
            if(parent.length > 0) {
                parent.remove(filter);
                this.childFilterContainer.remove(item, true);
            }
            if(parent.length === 0) {
                this.addCondition();
            }
            this.fireEvent("change", this);
        },
        
        createBuilderTypeCombo: function() {
            var types = this.allowedBuilderTypes || [
                gxp.FilterBuilder.ANY_OF, gxp.FilterBuilder.ALL_OF,
                gxp.FilterBuilder.NONE_OF
            ];
            var numTypes = types.length;
            var data = new Array(numTypes);
            var type;
            for(var i=0; i`_
     */
    Ext.namespace("gxp");
    
    /** api: constructor
     *  .. class:: WMSLayerPanel(config)
     *   
     *      Create a dialog for setting WMS layer properties like title, abstract,
     *      opacity, transparency and image format.
     */
    gxp.WMSLayerPanel = Ext.extend(Ext.TabPanel, {
        
        /** api: config[layerRecord]
         *  ``GeoExt.data.LayerRecord``
         *  Show properties for this layer record.
         */
        layerRecord: null,
    
        /** api: config[source]
         *  ``gxp.plugins.LayerSource``
         *  Source for the layer. Optional. If not provided, ``sameOriginStyling``
         *  will be ignored.
         */
        source: null,
        
        /** api: config[styling]
         *  ``Boolean``
         *  Show a "Styles" tab. Default is true.
         */
        styling: true,
        
        /** api: config[sameOriginStyling]
         *  ``Boolean``
         *  Only allow editing of styles for layers whose sources have a URL that
         *  matches the origin of this application.  It is strongly discouraged to 
         *  do styling through the proxy as all authorization headers and cookies 
         *  are shared with all remotesources.  Default is ``true``.
         */
        sameOriginStyling: true,
    
        /** api: config[rasterStyling]
         *  ``Boolean`` If set to true, single-band raster styling will be
         *  supported.  Default is ``false``.
         */
        rasterStyling: false,
    
        /** private: property[transparent]
         *  ``Boolean``
         *  Used to store the previous state of the transparent checkbox before
         *  changing the image format to jpeg (and automagically changing
         *  the checkbox to disabled and unchecked).
         */
        transparent: null,
        
        /** private: property[editableStyles]
         *  ``Boolean``
         */
        editableStyles: false,
        
        /** api: config[activeTab]
         *  ``String or Number``
         *  A string id or the numeric index of the tab that should be initially
         *  activated on render.  Defaults to ``0``.
         */
        activeTab: 0,
        
        /** api: config[border]
         *  ``Boolean``
         *  Display a border around the panel.  Defaults to ``false``.
         */
        border: false,
        
        /** api: config[imageFormats]
         *  ``RegEx`` Regular expression used to test browser friendly formats for
         *  GetMap requests.  The formats displayed will those from the record that
         *  match this expression.  Default is ``/png|gif|jpe?g/i``.
         */
        imageFormats: /png|gif|jpe?g/i,
        
        /** i18n */
        aboutText: "About",
        titleText: "Title",
        attributionText: "Attribution",
        nameText: "Name",
        descriptionText: "Description",
        displayText: "Display",
        opacityText: "Opacity",
        formatText: "Tile format",
        infoFormatText: "Info format",
        infoFormatEmptyText: "Select a format",
        transparentText: "Transparent",
        cacheText: "Caching",
        cacheFieldText: "Use cached tiles",
        stylesText: "Available Styles",
        displayOptionsText: "Display options",
        queryText: "Limit with filters",
        scaleText: "Limit by scale",
        minScaleText: "Min scale",
        maxScaleText: "Max scale",
        switchToFilterBuilderText: "Switch back to filter builder",
        cqlPrefixText: "or ",
        cqlText: "use CQL filter instead",
        singleTileText: "Single tile",
        singleTileFieldText: "Use a single tile",
    
        initComponent: function() {
            this.cqlFormat = new OpenLayers.Format.CQL();
            if (this.source) {
                this.source.getSchema(this.layerRecord, function(attributeStore) {
                    if (attributeStore !== false) {
                        var filter = this.layerRecord.getLayer().params.CQL_FILTER;
                        this.filterBuilder = new gxp.FilterBuilder({
                            filter: filter && this.cqlFormat.read(filter),
                            allowGroups: false,
                            listeners: {
                                afterrender: function() {
                                    this.filterBuilder.cascade(function(item) {
                                        if (item.getXType() === "toolbar") {
                                            item.addText(this.cqlPrefixText);
                                            item.addButton({
                                                text: this.cqlText,
                                                handler: this.switchToCQL,
                                                scope: this
                                            });
                                        }
                                    }, this);
                                },
                                change: function(builder) {
                                    var filter = builder.getFilter();
                                    var cql = null;
                                    if (filter !== false) {
                                        cql = this.cqlFormat.write(filter);
                                    }
                                    this.layerRecord.getLayer().mergeNewParams({
                                        CQL_FILTER: cql
                                    });
                                },
                                scope: this
                            },
                            attributes: attributeStore
                        });
                        this.filterFieldset.add(this.filterBuilder);
                        this.filterFieldset.doLayout();
                    }
                }, this);
            }
            this.addEvents(
                /** api: event[change]
                 *  Fires when the ``layerRecord`` is changed using this dialog.
                 */
                "change"
            );
            this.items = [
                this.createAboutPanel(),
                this.createDisplayPanel()
            ];
    
            // only add the Styles panel if we know for sure that we have styles
            if (this.styling && gxp.WMSStylesDialog && this.layerRecord.get("styles")) {
                // TODO: revisit this
                var url = this.layerRecord.get("restUrl");
                if (!url) {
                    url = (this.source || this.layerRecord.get("layer")).url.split(
                        "?").shift().replace(/\/(wms|ows)\/?$/, "/rest");
                }
                if (this.sameOriginStyling) {
                    // this could be made more robust
                    // for now, only style for sources with relative url
                    this.editableStyles = url.charAt(0) === "/";
                } else {
                    this.editableStyles = true;
                }
                this.items.push(this.createStylesPanel(url));
            }
    
            gxp.WMSLayerPanel.superclass.initComponent.call(this);
        },
    
        /** private: method[switchToCQL]
         *  Switch from filter builder to CQL.
         */
        switchToCQL: function() {
            var filter = this.filterBuilder.getFilter();
            var CQL = "";
            if (filter !== false) {
                CQL = this.cqlFormat.write(filter);
            }
            this.filterBuilder.hide();
            this.cqlField.setValue(CQL);
            this.cqlField.show();
            this.cqlToolbar.show();
        },
    
        /** private: method[switchToFilterBuilder]
         *  Switch from CQL field to filter builder.
         */
        switchToFilterBuilder: function() {
            var filter = null;
            // when parsing fails, we keep the previous filter in the filter builder
            try {
                filter = this.cqlFormat.read(this.cqlField.getValue());
            } catch(e) {
            }
            this.cqlField.hide();
            this.cqlToolbar.hide();
            this.filterBuilder.show();
            if (filter !== null) {
                this.filterBuilder.setFilter(filter);
            }
        },
    
        /** private: method[createStylesPanel]
         *  :arg url: ``String`` url to save styles to
         *
         *  Creates the Styles panel.
         */
        createStylesPanel: function(url) {
            var config = gxp.WMSStylesDialog.createGeoServerStylerConfig(
                this.layerRecord, url
            );
            if (this.rasterStyling === true) {
                config.plugins.push({
                    ptype: "gxp_wmsrasterstylesdialog"
                });
            }
            var ownerCt = this.ownerCt;
            if (!(ownerCt.ownerCt instanceof Ext.Window)) {
                config.dialogCls = Ext.Panel;
                config.showDlg = function(dlg) {
                    dlg.layout = "fit";
                    dlg.autoHeight = false;
                    ownerCt.add(dlg);
                };
            }
            return Ext.apply(config, {
                title: this.stylesText,
                style: "padding: 10px",
                editable: false
            });
        },
        
        /** private: method[createAboutPanel]
         *  Creates the about panel.
         */
        createAboutPanel: function() {
            return {
                title: this.aboutText,
                bodyStyle: {"padding": "10px"},
                defaults: {
                    border: false
                },
                items: [{
                    layout: "form",
                    labelWidth: 70,
                    items: [{
                        xtype: "textfield",
                        fieldLabel: this.titleText,
                        anchor: "99%",
                        value: this.layerRecord.get("title"),
                        listeners: {
                            change: function(field) {
                                this.layerRecord.set("title", field.getValue());
                                //TODO revisit when discussion on
                                // http://trac.geoext.org/ticket/110 is complete
                                this.layerRecord.commit();
                                this.fireEvent("change");
                            },
                            scope: this
                        }
                    }, {
                        xtype: "textfield",
                        fieldLabel: this.nameText,
                        anchor: "99%",
                        value: this.layerRecord.get("name"),
                        readOnly: true
                    }, {
                        xtype: "textfield",
                        fieldLabel: this.attributionText,
                        anchor: "99%",
                        listeners: {
                            change: function(field) {
                                var layer = this.layerRecord.getLayer();
                                layer.attribution = field.getValue();
                                layer.map.events.triggerEvent("changelayer", {
                                    layer: layer, property: "attribution"
                                });
                                this.fireEvent("change");
                            },
                            scope: this
                        },
                        value: this.layerRecord.getLayer().attribution
                    }]
                }, {
                    layout: "form",
                    labelAlign: "top",
                    items: [{
                        xtype: "textarea",
                        fieldLabel: this.descriptionText,
                        grow: true,
                        growMax: 150,
                        anchor: "99%",
                        value: this.layerRecord.get("abstract"),
                        readOnly: true
                    }]
                }]
            };
        },
    
        /** private: method[onFormatChange]
         *  Handler for when the image format is changed.
         */
        onFormatChange: function(combo) {
            var layer = this.layerRecord.getLayer();
            var format = combo.getValue();
            layer.mergeNewParams({
                format: format
            });
            var cb = this.transparentCb;
            if (format == "image/jpeg") {
                this.transparent = cb.getValue();
                cb.setValue(false);
            } else if (this.transparent !== null) {
                cb.setValue(this.transparent);
                this.transparent = null;
            }
            cb.setDisabled(format == "image/jpeg");
            this.fireEvent("change");
        },
    
        /** private: method[addScaleOptions]
         *  :arg layer: ``OpenLayers.Layer.WMS``
         *  :arg options: ``Object``
         *
         *  Apply the scale options to the layer and redraw.
         */
        addScaleOptions: function(layer, options) {
            // work around for https://github.com/openlayers/openlayers/issues/407
            layer.alwaysInRange = null;
            layer.addOptions(options);
            layer.display();
            layer.redraw();
        },
        
        /** private: method[createDisplayPanel]
         *  Creates the display panel.
         */
        createDisplayPanel: function() {
            var record = this.layerRecord;
            var layer = record.getLayer();
            var opacity = layer.opacity;
            if(opacity == null) {
                opacity = 1;
            }
            var formats = [];
            var currentFormat = layer.params["FORMAT"].toLowerCase();
            Ext.each(record.get("formats"), function(format) {
                if(this.imageFormats.test(format)) {
                    formats.push(format.toLowerCase());
                }
            }, this);
            if(formats.indexOf(currentFormat) === -1) {
                formats.push(currentFormat);
            }
            var transparent = layer.params["TRANSPARENT"];
            transparent = (transparent === "true" || transparent === true);
    
            return {
                title: this.displayText,
                layout: 'form',
                bodyStyle: {"padding": "10px"},
                defaults: {
                    labelWidth: 70
                },
                items: [{
                    xtype: "fieldset",
                    title: this.displayOptionsText,
                    items: [{
                        xtype: "gx_opacityslider",
                        name: "opacity",
                        anchor: "99%",
                        isFormField: true,
                        fieldLabel: this.opacityText,
                        listeners: {
                            change: function() {
                                this.fireEvent("change");
                            },
                            scope: this
                        },
                        layer: this.layerRecord
                    }, {
                        xtype: "compositefield",
                        fieldLabel: this.formatText,
                        anchor: "99%",
                        items: [{
                            xtype: "combo",
                            width: 90,
                            listWidth: 150,
                            store: formats,
                            value: currentFormat,
                            mode: "local",
                            triggerAction: "all",
                            editable: false,
                            listeners: {
                                select: this.onFormatChange,
                                scope: this
                            }
                        }, {
                            xtype: "checkbox",
                            ref: '../../../transparentCb',
                            checked: transparent,
                            listeners: {
                                check: function(checkbox, checked) {
                                    layer.mergeNewParams({
                                        transparent: checked ? "true" : "false"
                                    });
                                    this.fireEvent("change");
                                },
                                scope: this
                            }
                        }, {
                            xtype: "label",
                            cls: "gxp-layerproperties-label",
                            text: this.transparentText
                        }]
                    }, {
                        xtype: "compositefield",
                        fieldLabel: this.singleTileText,
                        anchor: "99%",
                        items: [{
                            xtype: "checkbox",
                            checked: this.layerRecord.get("layer").singleTile,
                            listeners: {
                                check: function(checkbox, checked) {
                                    layer.addOptions({singleTile: checked});
                                    this.fireEvent("change");
                                },
                                scope: this
                            }
                        }, {
                            xtype: "label",
                            cls: "gxp-layerproperties-label",
                            text: this.singleTileFieldText
                        }]
                    }, {
                        xtype: "compositefield",
                        anchor: "99%",
                        hidden: this.layerRecord.get("layer").params.TILED == null,
                        fieldLabel: this.cacheText,
                        items: [{
                            xtype: "checkbox",
                            checked: (this.layerRecord.get("layer").params.TILED === true),
                            listeners: {
                                check: function(checkbox, checked) {
                                    var layer = this.layerRecord.get("layer");
                                    layer.mergeNewParams({
                                        TILED: checked
                                    });
                                    this.fireEvent("change");
                                },
                                scope: this
                            }
                        }, {
                            xtype: "label",
                            cls: "gxp-layerproperties-label",
                            text: this.cacheFieldText
                        }]
                    }, {
                        xtype: "combo",
                        fieldLabel: this.infoFormatText,
                        emptyText: this.infoFormatEmptyText,
                        store: record.get("infoFormats"),
                        value: record.get("infoFormat"),
                        hidden: (record.get("infoFormats") === undefined),
                        mode: 'local',
                        listWidth: 150,
                        triggerAction: "all",
                        editable: false,
                        anchor: "99%", 
                        listeners: {
                            select: function(combo) {
                                var infoFormat = combo.getValue();
                                record.set("infoFormat", infoFormat);
                                this.fireEvent("change");
                            }
                        }, 
                        scope: this
                    }]
                }, {
                    xtype: "fieldset",
                    title: this.queryText,
                    hideLabels: true,
                    ref: "../filterFieldset",
                    listeners: {
                        expand: function() {
                            this.layerRecord.getLayer().mergeNewParams({CQL_FILTER: this.cqlFilter});
                        },
                        collapse: function() {
                            this.cqlFilter = this.layerRecord.getLayer().params.CQL_FILTER;
                            this.layerRecord.getLayer().mergeNewParams({CQL_FILTER: null});
                        },
                        scope: this
                    },
                    hidden: this.source === null,
                    checkboxToggle: true,
                    collapsed: !this.layerRecord.getLayer().params.CQL_FILTER,
                    items: [{
                        xtype: "textarea",
                        value: this.layerRecord.getLayer().params.CQL_FILTER,
                        grow: true,
                        anchor: '99%',
                        width: '100%',
                        growMax: 100,
                        ref: "../../cqlField",
                        hidden: true
                    }],
                    buttons: [{
                        ref: "../../../cqlToolbar",
                        hidden: true,
                        text: this.switchToFilterBuilderText,
                        handler: this.switchToFilterBuilder,
                        scope: this
                    }]
                }, {
                    xtype: "fieldset",
                    title: this.scaleText,
                    listeners: {
                        expand: function() {
                            var layer = this.layerRecord.getLayer();
                            if (this.minScale !== undefined || this.maxScale !== undefined) {
                                this.addScaleOptions(layer, {minScale: this.maxScale, maxScale: this.minScale});
                            }
                        },
                        collapse: function() {
                            var layer = this.layerRecord.getLayer();
                            this.minScale = layer.options.maxScale;
                            this.maxScale = layer.options.minScale;
                            this.addScaleOptions(layer, {minScale: null, maxScale: null});
                        },
                        scope: this
                    },
                    checkboxToggle: true,
                    collapsed: this.layerRecord.getLayer().options.maxScale == null &&
                        this.layerRecord.getLayer().options.minScale == null,
                    items: [{
                        xtype: "compositefield",
                        fieldLabel: this.minScaleText,
                        items: [{
                            xtype: "label",
                            text: "1:",
                            cls: "gxp-layerproperties-label"
                        }, {
                            xtype: "numberfield",
                            anchor: '99%',
                            width: '85%',
                            listeners: {
                                'change': function(field) {
                                    var options = {
                                        maxScale: parseInt(field.getValue())
                                    };
                                    var layer = this.layerRecord.getLayer();
                                    this.addScaleOptions(layer, options);
                                },
                                scope: this
                            },
                            value: this.layerRecord.getLayer().options.maxScale
                        }]
                    }, {
                        xtype: "compositefield",
                        fieldLabel: this.maxScaleText,
                        items: [{
                            xtype: "label",
                            text: "1:",
                            cls: "gxp-layerproperties-label"
                        }, {
                            xtype: "numberfield",
                            anchor: '99%',
                            width: '85%',
                            listeners: {
                                'change': function(field) {
                                    var options = {
                                        minScale: parseInt(field.getValue())
                                    };
                                    var layer = this.layerRecord.getLayer();
                                    this.addScaleOptions(layer, options);
                                },
                                scope: this
                            },
                            value: this.layerRecord.getLayer().options.minScale
                        }]
                    }]
                }]
            };
        }    
    
    });
    
    Ext.reg('gxp_wmslayerpanel', gxp.WMSLayerPanel); 
    
    /** FILE: widgets/form/FilterField.js **/
    /**
     * Copyright (c) 2008-2011 The Open Planning Project
     * 
     * Published under the GPL license.
     * See https://github.com/opengeo/gxp/raw/master/license.txt for the full text
     * of the license.
     */
    
    /**
     * @include widgets/form/ComparisonComboBox.js
     * @requires GeoExt/data/AttributeStore.js
     */
    
    /** api: (define)
     *  module = gxp.form
     *  class = FilterField
     *  base_link = `Ext.form.CompositeField `_
     */
    Ext.namespace("gxp.form");
    
    /** api: constructor
     *  .. class:: FilterField(config)
     *   
     *      A form field representing a comparison filter.
     */
    gxp.form.FilterField = Ext.extend(Ext.form.CompositeField, {
        
        /** api:config[lowerBoundaryTip]
         *  ``String`` tooltip for the lower boundary textfield (i18n)
         */
        lowerBoundaryTip: "lower boundary",
         
        /** api:config[upperBoundaryTip]
         *  ``String`` tooltip for the lower boundary textfield (i18n)
         */
        upperBoundaryTip: "upper boundary",
         
        /** api: config[caseInsensitiveMatch]
         *  ``Boolean``
         *  Should Comparison Filters for Strings do case insensitive matching?  Default is ``"false"``.
         */
        caseInsensitiveMatch: false,
    
        /**
         * Property: filter
         * {OpenLayers.Filter} Optional non-logical filter provided in the initial
         *     configuration.  To retreive the filter, use  instead
         *     of accessing this property directly.
         */
        filter: null,
        
        /**
         * Property: attributes
         * {GeoExt.data.AttributeStore} A configured attributes store for use in
         *     the filter property combo.
         */
        attributes: null,
    
        /** api:config[comparisonComboConfig]
         *  ``Object`` Config object for comparison combobox.
         */
    
        /** api:config[attributesComboConfig]
         *  ``Object`` Config object for attributes combobox.
         */
        
        /**
         * Property: attributesComboConfig
         * {Object}
         */
        attributesComboConfig: null,
    
        initComponent: function() {
                    
            if(!this.filter) {
                this.filter = this.createDefaultFilter();
            }
            // Maintain compatibility with QueryPanel, which relies on "remote"
            // mode and the filterBy filter applied in it's attributeStore's load
            // listener *after* the initial combo filtering.
            //TODO Assume that the AttributeStore is already loaded and always
            // create a new one without geometry fields.
            var mode = "remote", attributes = new GeoExt.data.AttributeStore();
            if (this.attributes) {
                if (this.attributes.getCount() != 0) {
                    mode = "local";
                    this.attributes.each(function(r) {
                        var match = /gml:((Multi)?(Point|Line|Polygon|Curve|Surface|Geometry)).*/.exec(r.get("type"));
                        match || attributes.add([r]);
                    });
                } else {
                    attributes = this.attributes;
                }
            }
    
            var defAttributesComboConfig = {
                xtype: "combo",
                store: attributes,
                editable: mode == "local",
                typeAhead: true,
                forceSelection: true,
                mode: mode,
                triggerAction: "all",
                ref: "property",
                allowBlank: this.allowBlank,
                displayField: "name",
                valueField: "name",
                value: this.filter.property,
                listeners: {
                    select: function(combo, record) {
                        this.items.get(1).enable();
                        this.filter.property = record.get("name");
                        this.fireEvent("change", this.filter, this);
                    },
                    // workaround for select event not being fired when tab is hit
                    // after field was autocompleted with forceSelection
                    "blur": function(combo) {
                        var index = combo.store.findExact("name", combo.getValue());
                        if (index != -1) {
                            combo.fireEvent("select", combo, combo.store.getAt(index));
                        } else if (combo.startValue != null) {
                            combo.setValue(combo.startValue);
                        }
                    },
                    scope: this
                },
                width: 120
            };
            this.attributesComboConfig = this.attributesComboConfig || {};
            Ext.applyIf(this.attributesComboConfig, defAttributesComboConfig);
    
            this.items = this.createFilterItems();
            
            this.addEvents(
                /**
                 * Event: change
                 * Fires when the filter changes.
                 *
                 * Listener arguments:
                 * filter - {OpenLayers.Filter} This filter.
                 * this - {gxp.form.FilterField} (TODO change sequence of event parameters)
                 */
                "change"
            ); 
    
            gxp.form.FilterField.superclass.initComponent.call(this);
        },
    
        /**
         * Method: validateValue
         * Performs validation checks on the filter field.
         *
         * Returns:
         * {Boolean} True if value is valid. 
         */
        validateValue: function(value, preventMark) {
            if (this.filter.type === OpenLayers.Filter.Comparison.BETWEEN) {
                return (this.filter.property !== null && this.filter.upperBoundary !== null &&
                    this.filter.lowerBoundary !== null);
            } else {
                return (this.filter.property !== null &&
                    this.filter.value !== null && this.filter.type !== null);
            }
        },
        
        /**
         * Method: createDefaultFilter
         * May be overridden to change the default filter.
         *
         * Returns:
         * {OpenLayers.Filter} By default, returns a comparison filter.
         */
        createDefaultFilter: function() {
            return new OpenLayers.Filter.Comparison({matchCase: !this.caseInsensitiveMatch});
        },
        
        /**
         * Method: createFilterItems
         * Creates a panel config containing filter parts.
         */
        createFilterItems: function() {
            var isBetween = this.filter.type === OpenLayers.Filter.Comparison.BETWEEN;
            return [
                this.attributesComboConfig, Ext.applyIf({
                    xtype: "gxp_comparisoncombo",
                    ref: "type",
                    disabled: this.filter.property == null,
                    allowBlank: this.allowBlank,
                    value: this.filter.type,
                    listeners: {
                        select: function(combo, record) {
                            this.items.get(2).enable();
                            this.items.get(3).enable();
                            this.items.get(4).enable();
                            this.setFilterType(record.get("value"));
                            this.fireEvent("change", this.filter, this);
                        },
                        scope: this
                    }
                }, this.comparisonComboConfig), {
                    xtype: "textfield",
                    disabled: this.filter.type == null,
                    hidden: isBetween,
                    ref: "value",
                    value: this.filter.value,
                    width: 50,
                    grow: true,
                    growMin: 50,
                    anchor: "100%",
                    allowBlank: this.allowBlank,
                    listeners: {
                        "change": function(field, value) {
                            this.filter.value = value;
                            this.fireEvent("change", this.filter, this);
                        },
                        scope: this
                    }
                }, {
                    xtype: "textfield",
                    disabled: this.filter.type == null,
                    hidden: !isBetween,
                    value: this.filter.lowerBoundary,
                    tooltip: this.lowerBoundaryTip,
                    grow: true,
                    growMin: 30,
                    ref: "lowerBoundary",
                    anchor: "100%",
                    allowBlank: this.allowBlank,
                    listeners: {
                        "change": function(field, value) {
                            this.filter.lowerBoundary = value;
                            this.fireEvent("change", this.filter, this);
                        },
                        "render": function(c) {
                            Ext.QuickTips.register({
                                target: c.getEl(),
                                text: this.lowerBoundaryTip
                            });
                        },
                        "autosize": function(field, width) {
                            field.setWidth(width);
                            field.ownerCt.doLayout();
                        },
                        scope: this
                    }
                }, {
                    xtype: "textfield",
                    disabled: this.filter.type == null,
                    hidden: !isBetween,
                    grow: true,
                    growMin: 30,
                    ref: "upperBoundary",
                    value: this.filter.upperBoundary,
                    allowBlank: this.allowBlank,
                    listeners: {
                        "change": function(field, value) {
                            this.filter.upperBoundary = value;
                            this.fireEvent("change", this.filter, this);
                        },
                        "render": function(c) {
                            Ext.QuickTips.register({
                                target: c.getEl(),
                                text: this.upperBoundaryTip
                            });
                        },
                        scope: this
                    }
                }
            ];
        },
        
        setFilterType: function(type) {
            this.filter.type = type;
            if (type === OpenLayers.Filter.Comparison.BETWEEN) {
                this.items.get(2).hide();
                this.items.get(3).show();
                this.items.get(4).show();
            } else {
                this.items.get(2).show();
                this.items.get(3).hide();
                this.items.get(4).hide();
            }
            this.doLayout();
        },
    
        /** api: method[setFilter]
         *  :arg filter: ``OpenLayers.Filter``` Change the filter object to be
         *  used.
         */
        setFilter: function(filter) {
            var previousType = this.filter.type;
            this.filter = filter;
            if (previousType !== filter.type) {
                this.setFilterType(filter.type);
            }
            this['property'].setValue(filter.property);
            this['type'].setValue(filter.type);
            if (filter.type === OpenLayers.Filter.Comparison.BETWEEN) {
                this['lowerBoundary'].setValue(filter.lowerBoundary);
                this['upperBoundary'].setValue(filter.upperBoundary);
            } else {
                this['value'].setValue(filter.value);
            }
            this.fireEvent("change", this.filter, this);
        }
    
    });
    
    Ext.reg('gxp_filterfield', gxp.form.FilterField);
    
    /** FILE: widgets/form/ComparisonComboBox.js **/
    /**
     * Copyright (c) 2008-2011 The Open Planning Project
     * 
     * Published under the GPL license.
     * See https://github.com/opengeo/gxp/raw/master/license.txt for the full text
     * of the license.
     */
    
    /**
     * @requires OpenLayers/Filter/Comparison.js
     */
    
    /** api: (define)
     *  module = gxp.form
     *  class = ComparisonComboBox
     *  base_link = `Ext.form.ComboBox `_
     */
    Ext.namespace("gxp.form");
    
    /** api: constructor
     *  .. class:: ComparisonComboBox(config)
     *   
     *      A combo box for selecting comparison operators available in OGC
     *      filters.
     */
    gxp.form.ComparisonComboBox = Ext.extend(Ext.form.ComboBox, {
        
        allowedTypes: [
            [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.LIKE, "like"],
            [OpenLayers.Filter.Comparison.BETWEEN, "between"]
        ],
    
        allowBlank: false,
    
        mode: "local",
    
        typeAhead: true,
    
        forceSelection: true,
    
        triggerAction: "all",
    
        width: 50,
    
        editable: true,
      
        initComponent: function() {
            var defConfig = {
                displayField: "name",
                valueField: "value",
                store: new Ext.data.SimpleStore({
                    data: this.allowedTypes,
                    fields: ["value", "name"]
                }),
                value: (this.value === undefined) ? this.allowedTypes[0][0] : this.value,
                listeners: {
                    // workaround for select event not being fired when tab is hit
                    // after field was autocompleted with forceSelection
                    "blur": function() {
                        var index = this.store.findExact("value", this.getValue());
                        if (index != -1) {
                            this.fireEvent("select", this, this.store.getAt(index));
                        } else if (this.startValue != null) {
                            this.setValue(this.startValue);
                        }
                    }
                }
            };
            Ext.applyIf(this, defConfig);
            
            gxp.form.ComparisonComboBox.superclass.initComponent.call(this);
        }
    });
    
    Ext.reg("gxp_comparisoncombo", gxp.form.ComparisonComboBox);
    
    /** FILE: GeoExt/widgets/LayerOpacitySlider.js **/
    /**
     * Copyright (c) 2008-2012 The Open Source Geospatial Foundation
     * Published under the BSD license.
     * See http://geoext.org/svn/geoext/core/trunk/license.txt for the full text
     * of the license.
     */
    
    /**
     * @include GeoExt/widgets/tips/LayerOpacitySliderTip.js
     * @include GeoExt/data/LayerRecord.js
     * @require OpenLayers/Layer.js
     */
    
    /** api: (define)
     *  module = GeoExt
     *  class = LayerOpacitySlider
     *  base_link = `Ext.slider.SingleSlider `_
     */
    Ext.namespace("GeoExt");
    
    /** api: example
     *  Sample code to render a slider outside the map viewport:
     *
     *  .. code-block:: javascript
     *
     *      var slider = new GeoExt.LayerOpacitySlider({
     *          renderTo: document.body,
     *          width: 200,
     *          layer: layer
     *      });
     *
     *  Sample code to add a slider to a map panel:
     *
     *  .. code-block:: javascript
     *
     *      var layer = new OpenLayers.Layer.WMS(
     *          "Global Imagery",
     *          "http://maps.opengeo.org/geowebcache/service/wms",
     *          {layers: "bluemarble"}
     *      );
     *      var panel = new GeoExt.MapPanel({
     *          renderTo: document.body,
     *          height: 300,
     *          width: 400,
     *          map: {
     *              controls: [new OpenLayers.Control.Navigation()]
     *          },
     *          layers: [layer],
     *          extent: [-5, 35, 15, 55],
     *          items: [{
     *              xtype: "gx_opacityslider",
     *              layer: layer,
     *              aggressive: true,
     *              vertical: true,
     *              height: 100,
     *              x: 10,
     *              y: 20
     *          }]
     *      });
     */
    
    /** api: constructor
     *  .. class:: LayerOpacitySlider(config)
     *
     *      Create a slider for controlling a layer's opacity.
     */
    GeoExt.LayerOpacitySlider = Ext.extend(Ext.slider.SingleSlider, {
    
        /** api: config[layer]
         *  ``OpenLayers.Layer`` or :class:`GeoExt.data.LayerRecord`
         *  The layer this slider changes the opacity of. (required)
         */
        /** private: property[layer]
         *  ``OpenLayers.Layer``
         */
        layer: null,
    
        /** api: config[complementaryLayer]
         *  ``OpenLayers.Layer`` or :class:`GeoExt.data.LayerRecord` 
         *  If provided, a layer that will be made invisible (its visibility is
         *  set to false) when the slider value is set to its max value. If this
         *  slider is used to fade visibility between to layers, setting
         *  ``complementaryLayer`` and ``changeVisibility`` will make sure that
         *  only visible tiles are loaded when the slider is set to its min or max
         *  value. (optional)
         */
        complementaryLayer: null,
    
        /** api: config[delay]
         *  ``Number`` Time in milliseconds before setting the opacity value to the
         *  layer. If the value change again within that time, the original value
         *  is not set. Only applicable if aggressive is true.
         */
        delay: 5,
    
        /** api: config[changeVisibilityDelay]
         *  ``Number`` Time in milliseconds before changing the layer's visibility.
         *  If the value changes again within that time, the layer's visibility
         *  change does not occur. Only applicable if changeVisibility is true.
         *  Defaults to 5.
         */
        changeVisibilityDelay: 5,
    
        /** api: config[aggressive]
         *  ``Boolean``
         *  If set to true, the opacity is changed as soon as the thumb is moved.
         *  Otherwise when the thumb is released (default).
         */
        aggressive: false,
    
        /** api: config[changeVisibility]
         *  ``Boolean``
         *  If set to true, the layer's visibility is handled by the
         *  slider, the slider makes the layer invisible when its
         *  value is changed to the min value, and makes the layer
         *  visible again when its value goes from the min value
         *  to some other value. The layer passed to the constructor
         *  must be visible, as its visibility is fully handled by
         *  the slider. Defaults to false.
         */
        changeVisibility: false,
    
        /** api: config[value]
         *  ``Number``
         *  The value to initialize the slider with. This value is
         *  taken into account only if the layer's opacity is null.
         *  If the layer's opacity is null and this value is not
         *  defined in the config object then the slider initializes
         *  it to the max value.
         */
        value: null,
    
        /** api: config[inverse]
         *  ``Boolean``
         *  If true, we will work with transparency instead of with opacity.
         *  Defaults to false.
         */
        /** private: property[inverse]
         *  ``Boolean``
         */
        inverse: false,
    
        /** private: method[constructor]
         *  Construct the component.
         */
        constructor: function(config) {
            if (config.layer) {
                this.layer = this.getLayer(config.layer);
                this.bind();
                this.complementaryLayer = this.getLayer(config.complementaryLayer);
                // before we call getOpacityValue inverse should be set
                if (config.inverse !== undefined) {
                    this.inverse = config.inverse;
                }
                config.value = (config.value !== undefined) ? 
                    config.value : this.getOpacityValue(this.layer);
                delete config.layer;
                delete config.complementaryLayer;
            }
            GeoExt.LayerOpacitySlider.superclass.constructor.call(this, config);
        },
    
        /** private: method[bind]
         */
        bind: function() {
            if (this.layer && this.layer.map) {
                this.layer.map.events.on({
                    changelayer: this.update,
                    scope: this
                });
            }
        },
    
        /** private: method[unbind]
         */
        unbind: function() {
            if (this.layer && this.layer.map && this.layer.map.events) {
                this.layer.map.events.un({
                    changelayer: this.update,
                    scope: this
                });
            }
        },
    
        /** private: method[update]
         *  Registered as a listener for opacity change.  Updates the value of the slider.
         */
        update: function(evt) {
            if (evt.property === "opacity" && evt.layer == this.layer &&
                !this._settingOpacity) {
                this.setValue(this.getOpacityValue(this.layer));
            }
        },
    
        /** api: method[setLayer]
         *  :param layer: ``OpenLayers.Layer`` or :class:`GeoExt.data.LayerRecord`
         *
         *  Bind a new layer to the opacity slider.
         */
        setLayer: function(layer) {
            this.unbind();
            this.layer = this.getLayer(layer);
            this.setValue(this.getOpacityValue(layer));
            this.bind();
        },
    
        /** private: method[getOpacityValue]
         *  :param layer: ``OpenLayers.Layer`` or :class:`GeoExt.data.LayerRecord`
         *  :return:  ``Integer`` The opacity for the layer.
         *
         *  Returns the opacity value for the layer.
         */
        getOpacityValue: function(layer) {
            var value;
            if (layer && layer.opacity !== null) {
                value = parseInt(layer.opacity * (this.maxValue - this.minValue));
            } else {
                value = this.maxValue;
            }
            if (this.inverse === true) {
                value = (this.maxValue - this.minValue) - value;
            }
            return value;
        },
    
        /** private: method[getLayer]
         *  :param layer: ``OpenLayers.Layer`` or :class:`GeoExt.data.LayerRecord`
         *  :return:  ``OpenLayers.Layer`` The OpenLayers layer object
         *
         *  Returns the OpenLayers layer object for a layer record or a plain layer 
         *  object.
         */
        getLayer: function(layer) {
            if (layer instanceof OpenLayers.Layer) {
                return layer;
            } else if (layer instanceof GeoExt.data.LayerRecord) {
                return layer.getLayer();
            }
        },
    
        /** private: method[initComponent]
         *  Initialize the component.
         */
        initComponent: function() {
    
            GeoExt.LayerOpacitySlider.superclass.initComponent.call(this);
    
            if (this.changeVisibility && this.layer &&
                (this.layer.opacity == 0 || 
                (this.inverse === false && this.value == this.minValue) || 
                (this.inverse === true && this.value == this.maxValue))) {
                this.layer.setVisibility(false);
            }
    
            if (this.complementaryLayer &&
                ((this.layer && this.layer.opacity == 1) ||
                 (this.inverse === false && this.value == this.maxValue) ||
                 (this.inverse === true && this.value == this.minValue))) {
                this.complementaryLayer.setVisibility(false);
            }
    
            if (this.aggressive === true) {
                this.on('change', this.changeLayerOpacity, this, {
                    buffer: this.delay
                });
            } else {
                this.on('changecomplete', this.changeLayerOpacity, this);
            }
    
            if (this.changeVisibility === true) {
                this.on('change', this.changeLayerVisibility, this, {
                    buffer: this.changeVisibilityDelay
                });
            }
    
            if (this.complementaryLayer) {
                this.on('change', this.changeComplementaryLayerVisibility, this, {
                    buffer: this.changeVisibilityDelay
                });
            }
            this.on("beforedestroy", this.unbind, this);
        },
    
        /** private: method[changeLayerOpacity]
         *  :param slider: :class:`GeoExt.LayerOpacitySlider`
         *  :param value: ``Number`` The slider value
         *
         *  Updates the ``OpenLayers.Layer`` opacity value.
         */
        changeLayerOpacity: function(slider, value) {
            if (this.layer) {
                value = value / (this.maxValue - this.minValue);
                if (this.inverse === true) {
                    value = 1 - value;
                }
                this._settingOpacity = true;
                this.layer.setOpacity(value);
                delete this._settingOpacity;
            }
        },
    
        /** private: method[changeLayerVisibility]
         *  :param slider: :class:`GeoExt.LayerOpacitySlider`
         *  :param value: ``Number`` The slider value
         *
         *  Updates the ``OpenLayers.Layer`` visibility.
         */
        changeLayerVisibility: function(slider, value) {
            var currentVisibility = this.layer.getVisibility();
            if ((this.inverse === false && value == this.minValue) ||
                (this.inverse === true && value == this.maxValue) &&
                currentVisibility === true) {
                this.layer.setVisibility(false);
            } else if ((this.inverse === false && value > this.minValue) ||
                (this.inverse === true && value < this.maxValue) &&
                       currentVisibility == false) {
                this.layer.setVisibility(true);
            }
        },
    
        /** private: method[changeComplementaryLayerVisibility]
         *  :param slider: :class:`GeoExt.LayerOpacitySlider`
         *  :param value: ``Number`` The slider value
         *
         *  Updates the complementary ``OpenLayers.Layer`` visibility.
         */
        changeComplementaryLayerVisibility: function(slider, value) {
            var currentVisibility = this.complementaryLayer.getVisibility();
            if ((this.inverse === false && value == this.maxValue) ||
                (this.inverse === true && value == this.minValue) &&
                currentVisibility === true) {
                this.complementaryLayer.setVisibility(false);
            } else if ((this.inverse === false && value < this.maxValue) ||
                (this.inverse === true && value > this.minValue) &&
                       currentVisibility == false) {
                this.complementaryLayer.setVisibility(true);
            }
        },
    
        /** private: method[addToMapPanel]
         *  :param panel: :class:`GeoExt.MapPanel`
         *
         *  Called by a MapPanel if this component is one of the items in the panel.
         */
        addToMapPanel: function(panel) {
            this.on({
                render: function() {
                    var el = this.getEl();
                    el.setStyle({
                        position: "absolute",
                        zIndex: panel.map.Z_INDEX_BASE.Control
                    });
                    el.on({
                        mousedown: this.stopMouseEvents,
                        click: this.stopMouseEvents
                    });
                },
                scope: this
            });
        },
    
        /** private: method[removeFromMapPanel]
         *  :param panel: :class:`GeoExt.MapPanel`
         *
         *  Called by a MapPanel if this component is one of the items in the panel.
         */
        removeFromMapPanel: function(panel) {
            var el = this.getEl();
            el.un({
                mousedown: this.stopMouseEvents,
                click: this.stopMouseEvents,
                scope: this
            });
            this.unbind();
        },
    
        /** private: method[stopMouseEvents]
         *  :param e: ``Object``
         */
        stopMouseEvents: function(e) {
            e.stopEvent();
        }
    });
    
    /** api: xtype = gx_opacityslider */
    Ext.reg('gx_opacityslider', GeoExt.LayerOpacitySlider);
    
    /** FILE: GeoExt/widgets/tips/SliderTip.js **/
    /**
     * Copyright (c) 2008-2012 The Open Source Geospatial Foundation
     * 
     * Published under the BSD license.
     * See http://svn.geoext.org/core/trunk/geoext/license.txt for the full text
     * of the license.
     */
    
    /** api: (define)
     *  module = GeoExt
     *  class = SliderTip
     *  base_link = `Ext.Tip `_
     */
    Ext.namespace("GeoExt");
    
    /** api: example
     *  Sample code to create a slider tip to display slider value on hover:
     * 
     *  .. code-block:: javascript
     *     
     *      var slider = new Ext.slider.SingleSlider({
     *          renderTo: document.body,
     *          width: 200,
     *          plugins: new GeoExt.SliderTip()
     *      });
     */
    
    /** api: constructor
     *  .. class:: SliderTip(config)
     *   
     *      Create a slider tip displaying ``Ext.slider.SingleSlider`` values over slider thumbs.
     */
    GeoExt.SliderTip = Ext.extend(Ext.slider.Tip, {
    
        /** api: config[hover]
         *  ``Boolean``
         *  Display the tip when hovering over the thumb.  If ``false``, tip will
         *  only be displayed while dragging.  Default is ``true``.
         */
        hover: true,
        
        /** api: config[minWidth]
         *  ``Number``
         *  Minimum width of the tip.  Default is 10.
         */
        minWidth: 10,
    
        /** api: config[offsets]
         *  ``Array(Number)``
         *  A two item list that provides x, y offsets for the tip.  Default is
         *  [0, -10].
         */
        offsets : [0, -10],
        
        /** private: property[dragging]
         *  ``Boolean``
         *  The thumb is currently being dragged.
         */
        dragging: false,
    
        /** private: method[init]
         *  :param slider: ``Ext.slider.SingleSlider``
         *  
         *  Called when the plugin is initialized.
         */
        init: function(slider) {
            GeoExt.SliderTip.superclass.init.apply(this, arguments);
            if (this.hover) {
                slider.on("render", this.registerThumbListeners, this);
            }
            this.slider = slider;
        },
    
        /** private: method[registerThumbListeners]
         *  Set as a listener for 'render' if hover is true.
         */
        registerThumbListeners: function() {
            var thumb, el;
            for (var i=0, ii=this.slider.thumbs.length; i`_
     */
    Ext.namespace("GeoExt");
    
    /** api: example
     *  Sample code to create a slider tip to display scale and resolution:
     *
     *  .. code-block:: javascript
     *
     *      var slider = new GeoExt.LayerOpacitySlider({
     *          renderTo: document.body,
     *          width: 200,
     *          layer: layer,
     *          plugins: new GeoExt.LayerOpacitySliderTip({
     *              template: "Opacity: {opacity}%"
     *          })
     *      });
     */
    
    /** api: constructor
     *  .. class:: LayerOpacitySliderTip(config)
     *
     *      Create a slider tip displaying :class:`GeoExt.LayerOpacitySlider` values.
     */
    GeoExt.LayerOpacitySliderTip = Ext.extend(GeoExt.SliderTip, {
    
        /** api: config[template]
         *  ``String``
         *  Template for the tip. Can be customized using the following keywords in
         *  curly braces:
         *
         *  * ``opacity`` - the opacity value in percent.
         */
        template: '
    {opacity}%
    ', /** private: property[compiledTemplate] * ``Ext.Template`` * The template compiled from the ``template`` string on init. */ compiledTemplate: null, /** private: method[init] * Called to initialize the plugin. */ init: function(slider) { this.compiledTemplate = new Ext.Template(this.template); GeoExt.LayerOpacitySliderTip.superclass.init.call(this, slider); }, /** private: method[getText] * :param slider: ``Ext.slider.SingleSlider`` The slider this tip is attached to. */ getText: function(thumb) { var data = { opacity: thumb.value }; return this.compiledTemplate.apply(data); } }); /** FILE: widgets/GoogleEarthPanel.js **/ /** * Copyright (c) 2008-2011 The Open Planning Project * * Published under the GPL license. * See https://github.com/opengeo/gxp/raw/master/license.txt for the full text * of the license. */ /** api: (define) * module = gxp * class = GoogleEarthPanel * base_link = `Ext.Panel `_ */ Ext.namespace("gxp"); /** api: constructor * .. class:: GoogleEarthPanel(config) * * Create a panel for showing a 3D visualization of * a map with the Google Earth plugin. * See http://code.google.com/apis/earth/ for plugin api * documentation. */ gxp.GoogleEarthPanel = Ext.extend(Ext.Panel, { /** * Google Earth's horizontal field of view, in radians. (30 degrees) * This was not pulled from any documentation; it was chosen simply * by it's nice, even number, as well as its appearance to actually * work. */ HORIZONTAL_FIELD_OF_VIEW: (30 * Math.PI) / 180, /** api: config[flyToSpeed] * ``Number`` * Specifies the speed (0.0 to 5.0) at which the camera moves to the * target extent. Set to null to use the Google Earth default. By default * we show the target extent immediately, without flying to it. */ /** private: property[map] * ``OpenLayers.Map`` * The OpenLayers map associated with this panel. Defaults * to the map of the configured MapPanel */ map: null, /** api: config[mapPanel] * ``GeoExt.MapPanel | String`` * The map panel associated with this panel. If a MapPanel instance is * not provided, a MapPanel id must be provided. */ mapPanel: null, /** private: property[layers] * :class:`GeoExt.data.LayerStore` A store containing * :class:`GeoExt.data.LayerRecord` objects. */ layers: null, /** private: property[earth] * The Google Earth object. */ earth: null, //currently always set to 4326? projection: null, layerCache: null, /** private: method[initComponent] * Initializes the Google Earth panel. */ initComponent: function() { this.addEvents( /** api: event[beforeadd] * Fires before a layer is added to the 3D view. If a listener * returns ``false``, the layer will not be added. Listeners * will be called with a single argument: the layer record. */ "beforeadd", /** api: event[pluginfailure] * Fires when there is a failure creating the instance. Listeners * will receive two arguments: this plugin and the failure code * (see the Google Earth API docs for details on the failure codes). */ "pluginfailure", /** api: event[pluginready] * Fires when the instance is ready. Listeners will receive one * argument: the GEPlugin instance. */ "pluginready" ); gxp.GoogleEarthPanel.superclass.initComponent.call(this); var mapPanel = this.mapPanel; if (mapPanel && !(mapPanel instanceof GeoExt.MapPanel)) { mapPanel = Ext.getCmp(mapPanel); } if (!mapPanel) { throw new Error("Could not get map panel from config: " + this.mapPanel); } this.map = mapPanel.map; this.layers = mapPanel.layers; this.projection = new OpenLayers.Projection("EPSG:4326"); this.on("render", this.onRenderEvent, this); this.on("show", this.onShowEvent, this); this.on("hide", function() { if (this.earth != null) { this.updateMap(); } // Remove the plugin from the dom. this.body.dom.innerHTML = ""; this.earth = null; }, this); }, /** private: method[onEarthReady] * Runs when Google Earth instance is ready. Adds layer * store handlers. */ onEarthReady: function(object){ this.earth = object; if (this.flyToSpeed === undefined) { // We don't want to fly. Just go to the right spot immediately. this.earth.getOptions().setFlyToSpeed(this.earth.SPEED_TELEPORT); } else if (this.flyToSpeed !== null) { this.earth.getOptions().setFlyToSpeed(this.flyToSpeed); } // Set the extent of the earth to be that shown in OpenLayers. this.resetCamera(); this.setExtent(this.map.getExtent()); // Show the navigation control, and make it so it is on the left. // Not actually sure how the second to fourth lines make that happen, // but hey -- it works. :) this.earth.getNavigationControl().setVisibility(this.earth.VISIBILITY_SHOW); var screenXY = this.earth.getNavigationControl().getScreenXY(); screenXY.setXUnits(this.earth.UNITS_PIXELS); screenXY.setYUnits(this.earth.UNITS_INSET_PIXELS); // Show the plugin. this.earth.getWindow().setVisibility(true); this.layers.each(function(record) { this.addLayer(record); }, this); this.layers.on("remove", this.updateLayers, this); this.layers.on("update", this.updateLayers, this); this.layers.on("add", this.updateLayers, this); this.fireEvent("pluginready", this.earth); // Set up events. Notice global google namespace. // google.earth.addEventListener(this.earth.getView(), // "viewchangeend", // this.updateMap.createDelegate(this)); }, /** private: method[onRenderEvent] * Unfortunately, Ext does not call show() if the component is initally * displayed, so we need to fake it. * We can't call this method onRender because Ext has already stolen * the name for internal use :-( */ onRenderEvent: function() { var isCard = this.ownerCt && this.ownerCt.layout instanceof Ext.layout.CardLayout; if (!this.hidden && !isCard) { this.onShowEvent(); } }, /** private: method[onShowEvent] * Unfortunately, the Google Earth plugin does not like to be hidden. * No matter whether you hide it through CSS visibility, CSS offsets, * or CSS display = none, the Google Earth plugin will show an error * message when it is re-enabled. To counteract this, we delete * the instance and create a new one each time. * We can't call this method onShow because Ext has already stolen * the name for internal use :-( */ onShowEvent: function() { if (this.rendered) { this.layerCache = {}; google.earth.createInstance( this.body.dom, this.onEarthReady.createDelegate(this), (function(code) { this.fireEvent("pluginfailure", this, code); }).createDelegate(this) ); } }, /** */ beforeDestroy: function() { this.layers.un("remove", this.updateLayers, this); this.layers.un("update", this.updateLayers, this); this.layers.un("add", this.updateLayers, this); gxp.GoogleEarthPanel.superclass.beforeDestroy.call(this); }, /** private: method[updateLayers] * Synchronizes the 3D visualization with the * configured layer store. */ updateLayers: function() { if (!this.earth) return; var features = this.earth.getFeatures(); var f = features.getFirstChild(); while (f != null) { features.removeChild(f); f = features.getFirstChild(); } this.layers.each(function(record) { this.addLayer(record); }, this); }, /** private: method[addLayer] * Adds a layer to the 3D visualization. */ addLayer: function(layer, order) { var lyr = layer.getLayer(); var ows = (lyr && lyr.url); if (this.earth && lyr instanceof OpenLayers.Layer.WMS && typeof ows == "string") { var add = this.fireEvent("beforeadd", layer); if (add !== false) { var name = lyr.id; var networkLink; if (this.layerCache[name]) { networkLink = this.layerCache[name]; } else { var link = this.earth.createLink('kl_' + name); ows = ows.replace(/\?.*/, ''); var params = lyr.params; var kmlPath = '/kml?mode=refresh&layers=' + params.LAYERS + "&styles=" + params.STYLES; link.setHref(ows + kmlPath); networkLink = this.earth.createNetworkLink('nl_' + name); networkLink.setName(name); networkLink.set(link, false, false); this.layerCache[name] = networkLink; } networkLink.setVisibility(lyr.getVisibility()); if (order !== undefined && order < this.earth.getFeatures().getChildNodes().getLength()) { this.earth.getFeatures(). insertBefore(this.earth.getFeatures().getChildNodes().item(order)); } else { this.earth.getFeatures().appendChild(networkLink); } } } }, /** private: method[setExtent] * Sets the view of the 3D visualization to approximate an OpenLayers extent. */ setExtent: function(extent) { extent = extent.transform(this.map.getProjectionObject(), this.projection); var center = extent.getCenterLonLat(); var width = this.getExtentWidth(extent); // Calculate height of the camera from the ground, in meters. var height = width / (2 * Math.tan(this.HORIZONTAL_FIELD_OF_VIEW)); var lookAt = this.earth.getView().copyAsLookAt(this.earth.ALTITUDE_RELATIVE_TO_GROUND); lookAt.setLatitude(center.lat); lookAt.setLongitude(center.lon); lookAt.setRange(height); this.earth.getView().setAbstractView(lookAt); }, resetCamera: function() { var camera = this.earth.getView().copyAsCamera(this.earth.ALTITUDE_RELATIVE_TO_GROUND); camera.setRoll(0); camera.setHeading(0); camera.setTilt(0); this.earth.getView().setAbstractView(camera); }, /** private: method[getExtent] * Gets an OpenLayers.Bounds that approximates the visable area of * 3D visualization. */ getExtent: function() { var geBounds = this.earth.getView().getViewportGlobeBounds(); var olBounds = new OpenLayers.Bounds( geBounds.getWest(), geBounds.getSouth(), geBounds.getEast(), geBounds.getNorth() ); return olBounds; }, /** private: method[updateMap] */ updateMap: function() { // Get the center of the map from GE. We let GE get the center (as opposed to getting // the extent and then finding the center) because it'll find the correct visual // center represented by the globe, taking into account spherical calculations. var lookAt = this.earth.getView().copyAsLookAt(this.earth.ALTITUDE_RELATIVE_TO_GROUND); var center = this.reprojectToMap( new OpenLayers.LonLat(lookAt.getLongitude(), lookAt.getLatitude()) ); // Zoom to the closest zoom level for the extent given by GE's getViewPortGlobeBounds(). // Then recenter based on the visual center shown in GE. var geExtent = this.reprojectToMap(this.getExtent()); this.map.zoomToExtent(geExtent, true); this.map.setCenter(center); // Slight dirty hack -- // GE's getViewPortGlobeBounds() function gives us an extent larger than what OL // should show, sometimes with more data. This extent works most of the time when OL // tries to find the closest zoom level, but on some edge cases it zooms out // one zoom level too far. To counteract this, we calculate the geodetic width that // we expect GE to show (note: this is the opposite of the setExtent() calculations), // and then compare that width to that of the current zoom level and one zoom level // closer. If the next zoom level shows a geodetic width that's nearer to the width // we expect, then we zoom to that zoom level. // // Big note: This expects a map that has fractional zoom disabled! var height = lookAt.getRange(); var width = 2 * height * Math.tan(this.HORIZONTAL_FIELD_OF_VIEW); var nextResolution = this.map.getResolutionForZoom(this.map.getZoom() + 1); var currentExtent = this.map.getExtent(); var nextExtent = new OpenLayers.Bounds( center.lon - (this.map.getSize().w / 2 * nextResolution), center.lat + (this.map.getSize().h / 2 * nextResolution), center.lon + (this.map.getSize().w / 2 * nextResolution), center.lat - (this.map.getSize().h / 2 * nextResolution) ); var currentWidthDiff = Math.abs(this.getExtentWidth(currentExtent) - width); var nextWidthDiff = Math.abs(this.getExtentWidth(nextExtent) - width); if (nextWidthDiff < currentWidthDiff) { this.map.zoomTo(this.map.getZoom() + 1); } }, /** private: method[getExentWidth] */ getExtentWidth: function(extent) { var center = extent.getCenterLonLat(); var middleLeft = new OpenLayers.LonLat(extent.left, center.lat); var middleRight = new OpenLayers.LonLat(extent.right, center.lat); return OpenLayers.Util.distVincenty(middleLeft, middleRight) * 1000; }, /** private: method[reprojectToGE] */ reprojectToGE: function(data) { return data.clone().transform(this.map.getProjectionObject(), this.projection); }, /** private: method[reprojectToMap] */ reprojectToMap: function(data) { return data.clone().transform(this.projection, this.map.getProjectionObject()); } }); /** api: xtype = gxp_googleearthpanel */ Ext.reg("gxp_googleearthpanel", gxp.GoogleEarthPanel); /** FILE: 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.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 . 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 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 . * * 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 * {} 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 * instances. * * Parameters: * target - {} 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; } }); /** FILE: 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.Panel = OpenLayers.Class(OpenLayers.Control, { /** * Property: controls * {Array()} */ controls: null, /** * APIProperty: autoActivate * {Boolean} Activate the control when it is added to a map. Default is * true. */ autoActivate: true, /** * APIProperty: defaultControl * {} The control which is activated when the control is * activated (turned on), which also happens at instantiation. * If is true, 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 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 method is called. * * Specific properties for controls on a panel: * type - {Number} One of , * , . * If not provided, is assumed. * title - {string} Text displayed when mouse is over the icon that * represents the control. * * The of a control determines the behavior when * clicking its icon: * - The control is activated and other * controls of this type in the same panel are deactivated. This is * the default type. * - The active state of the control is * toggled. * - The * method of the control is called, * but its active state is not changed. * * If a control is , 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=0; i--) { this.div.removeChild(this.div.childNodes[i]); } this.div.innerHTML = ""; if (this.active) { for (var i=0, len=this.controls.length; i} */ 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} 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} 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()} Controls to add into map. */ addControlsToMap: function (controls) { var control; for (var i=0, len=controls.length; 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()} 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()} 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()} 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" }); /** FILE: 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 * . * When clicked, the function trigger() is executed. * * Inherits from: * - * * 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" }); /** FILE: 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.ZoomIn = OpenLayers.Class(OpenLayers.Control.Button, { /** * Method: trigger */ trigger: function(){ if (this.map) { this.map.zoomIn(); } }, CLASS_NAME: "OpenLayers.Control.ZoomIn" }); /** FILE: 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.ZoomOut = OpenLayers.Class(OpenLayers.Control.Button, { /** * Method: trigger */ trigger: function(){ if (this.map) { this.map.zoomOut(); } }, CLASS_NAME: "OpenLayers.Control.ZoomOut" }); /** FILE: 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 * . * * Inherits from: * - */ 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" }); /** FILE: 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 * , a , and a * . 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) * * (end) * * Inherits from: * - */ 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" }); /** FILE: 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 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, 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; /** FILE: 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, { /** * 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 * {} The last pixel location of the drag. */ last: null, /** * Property: start * {} 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 - {} 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" }); /** FILE: 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.Box = OpenLayers.Class(OpenLayers.Handler, { /** * Property: dragHandler * {} */ 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 - {} * 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 - {} */ 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" }); /** FILE: 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.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 * 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 - {} or {} */ 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); } } }, CLASS_NAME: "OpenLayers.Control.ZoomBox" }); /** FILE: 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/Control.js * @requires OpenLayers/Handler/Drag.js */ /** * Class: OpenLayers.Control.DragPan * The DragPan control pans the map with a drag of the mouse. * * Inherits from: * - */ OpenLayers.Control.DragPan = OpenLayers.Class(OpenLayers.Control, { /** * Property: type * {OpenLayers.Control.TYPES} */ type: OpenLayers.Control.TYPE_TOOL, /** * Property: panned * {Boolean} The map moved. */ panned: false, /** * 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. */ interval: 0, /** * APIProperty: documentDrag * {Boolean} If set to true, mouse dragging will continue even if the * mouse cursor leaves the map viewport. Default is false. */ documentDrag: false, /** * Property: kinetic * {} The OpenLayers.Kinetic object. */ kinetic: 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 {} * constructor. Defaults to true. * To get kinetic dragging, ensure that OpenLayers/Kinetic.js is * included in your build config. */ enableKinetic: true, /** * APIProperty: kineticInterval * {Integer} Interval in milliseconds between 2 steps in the "kinetic * scrolling". Applies only if enableKinetic is set. Defaults * to 10 milliseconds. */ kineticInterval: 10, /** * Method: draw * Creates a Drag handler, using and * as callbacks. */ 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 } ); }, /** * Method: panMapStart */ panMapStart: function() { if(this.kinetic) { this.kinetic.begin(); } }, /** * Method: panMap * * Parameters: * xy - {} Pixel of the mouse position */ 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} ); }, /** * Method: panMapDone * Finish the panning operation. Only call setCenter (through ) * if the map has actually been moved. * * Parameters: * xy - {} Pixel of the mouse position */ 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" }); /** FILE: 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/Handler.js */ /** * Class: OpenLayers.Handler.MouseWheel * Handler for wheel up/down events. * * Inherits from: * - */ OpenLayers.Handler.MouseWheel = OpenLayers.Class(OpenLayers.Handler, { /** * Property: wheelListener * {function} */ wheelListener: null, /** * 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, /** * 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, /** * 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, /** * 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, /** * Constructor: OpenLayers.Handler.MouseWheel * * Parameters: * 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 ); }, /** * Method: destroy */ destroy: function() { OpenLayers.Handler.prototype.destroy.apply(this, arguments); this.wheelListener = null; }, /** * Mouse ScrollWheel code thanks to http://adomas.org/javascript-mouse-wheel/ */ /** * Method: onWheelEvent * Catch the wheel event and handle it xbrowserly * * Parameters: * e - {Event} */ 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 } } 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); } }, /** * 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.Handler.MouseWheel" }); /** FILE: 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/Handler.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 , controls can also ignore clicks * that include a drag. Create a new instance with the * constructor. * * Inherits from: * - */ 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, /** * 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, /** * APIProperty: double * {Boolean} Handle double-clicks. Default is false. */ 'double': false, /** * 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, /** * 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 . */ 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, /** * 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 * . */ rightclickTimerId: null, /** * Constructor: OpenLayers.Handler.Click * Create a new click handler. * * Parameters: * 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: * {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. * * 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 * * 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. * * 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. */ 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); } 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]); var delayedRightCall = OpenLayers.Function.bind( this.delayedRightCall, this, clickEvent ); this.rightclickTimerId = window.setTimeout( delayedRightCall, this.delay ); } } return !this.stopSingle; }, /** * Method: delayedRightCall * Sets to null. And optionally triggers the * rightclick callback if evt is set. */ delayedRightCall: function(evt) { this.rightclickTimerId = null; if (evt) { this.callback('rightclick', [evt]); } }, /** * 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; }, /** * 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(); } }, /** * 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); } } }, /** * 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 ); }, /** * 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 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 this.pixelTolerance) { passes = false; break; } } } } 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) ); }, /** * 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; }, /** * Method: clearTimer * Clear the timer and set 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; } }, /** * Method: delayedCall * Sets to null. And optionally triggers the click callback if * evt is set. */ delayedCall: function(evt) { this.timerId = null; if (evt) { this.callback("click", [evt]); } }, /** * 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 control. * * Note that this control is added to the map by default (if no controls * array is sent in the options object to the * constructor). * * Inherits: * - */ OpenLayers.Control.Navigation = OpenLayers.Class(OpenLayers.Control, { /** * Property: dragPan * {} */ dragPan: null, /** * APIProperty: dragPanOptions * {Object} Options passed to the DragPan control. */ dragPanOptions: null, /** * Property: pinchZoom * {} */ pinchZoom: null, /** * APIProperty: pinchZoomOptions * {Object} Options passed to the PinchZoom control. */ pinchZoomOptions: null, /** * APIProperty: documentDrag * {Boolean} Allow panning of the map by dragging outside map viewport. * Default is false. */ documentDrag: false, /** * Property: zoomBox * {} */ zoomBox: null, /** * APIProperty: zoomBoxEnabled * {Boolean} Whether the user can draw a box to zoom */ zoomBoxEnabled: true, /** * APIProperty: zoomWheelEnabled * {Boolean} Whether the mousewheel should zoom the map */ zoomWheelEnabled: true, /** * Property: mouseWheelOptions * {Object} Options passed to the MouseWheel control (only useful if * 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} 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: */ 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); }, /** * 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} */ 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); }, /** * Method: defaultDblRightClick * * Parameters: * evt - {Event} */ defaultDblRightClick: function (evt) { this.map.zoomTo(this.map.zoom - 1, evt.xy); }, /** * Method: wheelChange * * Parameters: * evt - {Event} * deltaZ - {Integer} */ 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: wheelUp * User spun scroll wheel up * * Parameters: * evt - {Event} * delta - {Integer} */ wheelUp: function(evt, delta) { this.wheelChange(evt, delta || 1); }, /** * Method: wheelDown * User spun scroll wheel down * * Parameters: * evt - {Event} * delta - {Integer} */ wheelDown: function(evt, delta) { this.wheelChange(evt, delta || -1); }, /** * 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" }); /** FILE: 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" }); /** FILE: 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 . * * Inherits from: * - */ 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 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 . 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; } } }, 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"; /** FILE: 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) * * (end) * * Inherits from: * - */ 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 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 . 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" }); /** FILE: GeoExt/widgets/ZoomSlider.js **/ /** * Copyright (c) 2008-2012 The Open Source Geospatial Foundation * * Published under the BSD license. * See http://svn.geoext.org/core/trunk/geoext/license.txt for the full text * of the license. */ /** * @include GeoExt/widgets/tips/ZoomSliderTip.js * @require OpenLayers/Util.js */ /** api: (define) * module = GeoExt * class = ZoomSlider * base_link = `Ext.slider.SingleSlider `_ */ Ext.namespace("GeoExt"); /** api: example * Sample code to render a slider outside the map viewport: * * .. code-block:: javascript * * var slider = new GeoExt.ZoomSlider({ * renderTo: document.body, * width: 200, * map: map * }); * * Sample code to add a slider to a map panel: * * .. code-block:: javascript * * var panel = new GeoExt.MapPanel({ * renderTo: document.body, * height: 300, * width: 400, * map: { * controls: [new OpenLayers.Control.Navigation()] * }, * layers: [new OpenLayers.Layer.WMS( * "Global Imagery", * "http://maps.opengeo.org/geowebcache/service/wms", * {layers: "bluemarble"} * )], * extent: [-5, 35, 15, 55], * items: [{ * xtype: "gx_zoomslider", * aggressive: true, * vertical: true, * height: 100, * x: 10, * y: 20 * }] * }); */ /** api: constructor * .. class:: ZoomSlider(config) * * Create a slider for controlling a map's zoom level. */ GeoExt.ZoomSlider = Ext.extend(Ext.slider.SingleSlider, { /** api: config[map] * ``OpenLayers.Map`` or :class:`GeoExt.MapPanel` * The map that the slider controls. */ map: null, /** api: config[baseCls] * ``String`` * The CSS class name for the slider elements. Default is "gx-zoomslider". */ baseCls: "gx-zoomslider", /** api: config[aggressive] * ``Boolean`` * If set to true, the map is zoomed as soon as the thumb is moved. Otherwise * the map is zoomed when the thumb is released (default). */ aggressive: false, /** private: property[updating] * ``Boolean`` * The slider position is being updated by itself (based on map zoomend). */ updating: false, /** private: method[initComponent] * Initialize the component. */ initComponent: function() { GeoExt.ZoomSlider.superclass.initComponent.call(this); if(this.map) { if(this.map instanceof GeoExt.MapPanel) { this.map = this.map.map; } this.bind(this.map); } if (this.aggressive === true) { this.on('change', this.changeHandler, this); } else { this.on('changecomplete', this.changeHandler, this); } this.on("beforedestroy", this.unbind, this); }, /** private: method[onRender] * Override onRender to set base css class. */ onRender: function() { GeoExt.ZoomSlider.superclass.onRender.apply(this, arguments); this.el.addClass(this.baseCls); }, /** private: method[afterRender] * Override afterRender because the render event is fired too early * to call update. */ afterRender : function(){ Ext.slider.SingleSlider.superclass.afterRender.apply(this, arguments); this.update(); }, /** private: method[addToMapPanel] * :param panel: :class:`GeoExt.MapPanel` * * Called by a MapPanel if this component is one of the items in the panel. */ addToMapPanel: function(panel) { this.on({ render: function() { var el = this.getEl(); el.setStyle({ position: "absolute", zIndex: panel.map.Z_INDEX_BASE.Control }); el.on({ mousedown: this.stopMouseEvents, click: this.stopMouseEvents }); }, afterrender: function() { this.bind(panel.map); }, scope: this }); }, /** private: method[stopMouseEvents] * :param e: ``Object`` */ stopMouseEvents: function(e) { e.stopEvent(); }, /** private: method[removeFromMapPanel] * :param panel: :class:`GeoExt.MapPanel` * * Called by a MapPanel if this component is one of the items in the panel. */ removeFromMapPanel: function(panel) { var el = this.getEl(); el.un("mousedown", this.stopMouseEvents, this); el.un("click", this.stopMouseEvents, this); this.unbind(); }, /** private: method[bind] * :param map: ``OpenLayers.Map`` */ bind: function(map) { this.map = map; this.map.events.on({ zoomend: this.onZoomEnd, updatesize: this.initZoomValues, changebaselayer: this.initZoomValues, scope: this }); this.initZoomValues(); }, /** private: method[onZoomEnd] * Registered as a listener for zoomend. */ onZoomEnd: function() { this.update(); }, /** private: method[unbind] */ unbind: function() { if(this.map && this.map.events) { this.map.events.un({ zoomend: this.onZoomEnd, updatesize: this.initZoomValues, changebaselayer: this.initZoomValues, scope: this }); } }, /** private: method[initZoomValues] * Set the min/max values for the slider if not set in the config. */ initZoomValues: function() { var layer = this.map.baseLayer; if (layer) { if(this.initialConfig.minValue === undefined) { //TODO remove check for getMinZoom when we require OpenLayers 2.12. var minZoom = this.map.getMinZoom ? this.map.getMinZoom() : 0; this.minValue = Math.max(minZoom, layer.minZoomLevel || 0); } if(this.initialConfig.maxValue === undefined) { this.maxValue = layer.minZoomLevel == null ? layer.numZoomLevels - 1 : layer.maxZoomLevel; } // reset the thumb value so it gets repositioned when we call update this.update(true); } }, /** api: method[getZoom] * :return: ``Number`` The map zoom level. * * Get the zoom level for the associated map based on the slider value. */ getZoom: function() { return this.getValue(); }, /** api: method[getScale] * :return: ``Number`` The map scale denominator. * * Get the scale denominator for the associated map based on the slider value. */ getScale: function() { return OpenLayers.Util.getScaleFromResolution( this.map.getResolutionForZoom(this.getValue()), this.map.getUnits() ); }, /** api: method[getResolution] * :return: ``Number`` The map resolution. * * Get the resolution for the associated map based on the slider value. */ getResolution: function() { return this.map.getResolutionForZoom(this.getValue()); }, /** private: method[changeHandler] * Registered as a listener for slider changecomplete. Zooms the map. */ changeHandler: function() { if(this.map && !this.updating) { this.map.zoomTo(this.getValue()); } }, /** private: method[update] * :param force: ``Boolean`` Force an update of the thumb position even * if the value may not have changed (but min/max or length have). * * Registered as a listener for map zoomend. Updates the value of the slider. */ update: function(force) { if(this.rendered && this.map) { this.updating = true; if (force) { /** * This triggers repositioning even if the value doesn't * change. We want this when the min/max values change but * the zoom level doesn't. */ this.thumbs[0].value = null; } this.setValue(this.map.getZoom()); this.updating = false; } } }); /** api: xtype = gx_zoomslider */ Ext.reg('gx_zoomslider', GeoExt.ZoomSlider); /** FILE: GeoExt/widgets/tips/ZoomSliderTip.js **/ /** * Copyright (c) 2008-2012 The Open Source Geospatial Foundation * * Published under the BSD license. * See http://svn.geoext.org/core/trunk/geoext/license.txt for the full text * of the license. */ /** * @requires GeoExt/widgets/tips/SliderTip.js */ /** api: (extends) * GeoExt/widgets/tips/SliderTip.js */ /** api: (define) * module = GeoExt * class = ZoomSliderTip * base_link = `Ext.Tip `_ */ Ext.namespace("GeoExt"); /** api: example * Sample code to create a slider tip to display scale and resolution: * * .. code-block:: javascript * * var slider = new GeoExt.ZoomSlider({ * renderTo: document.body, * width: 200, * map: map, * plugins: new GeoExt.ZoomSliderTip({ * template: "Scale: 1 : {scale}
    Resolution: {resolution}" * }) * }); */ /** api: constructor * .. class:: ZoomSliderTip(config) * * Create a slider tip displaying :class:`GeoExt.ZoomSlider` values. */ GeoExt.ZoomSliderTip = Ext.extend(GeoExt.SliderTip, { /** api: config[template] * ``String`` * Template for the tip. Can be customized using the following keywords in * curly braces: * * * ``zoom`` - the zoom level * * ``resolution`` - the resolution * * ``scale`` - the scale denominator */ template: '
    Zoom Level: {zoom}
    ' + '
    Resolution: {resolution}
    ' + '
    Scale: 1 : {scale}
    ', /** private: property[compiledTemplate] * ``Ext.Template`` * The template compiled from the ``template`` string on init. */ compiledTemplate: null, /** private: method[init] * Called to initialize the plugin. */ init: function(slider) { this.compiledTemplate = new Ext.Template(this.template); GeoExt.ZoomSliderTip.superclass.init.call(this, slider); }, /** private: method[getText] * :param slider: ``Ext.slider.SingleSlider`` The slider this tip is attached to. */ getText: function(thumb) { var data = { zoom: thumb.value, resolution: this.slider.getResolution(), scale: Math.round(this.slider.getScale()) }; return this.compiledTemplate.apply(data); } }); /** FILE: widgets/Viewer.js **/ /** * Copyright (c) 2008-2011 The Open Planning Project * * Published under the GPL license. * See https://github.com/opengeo/gxp/raw/master/license.txt for the full text * of the license. */ /** * @requires util.js * @requires OpenLayers/Control/Attribution.js * @requires OpenLayers/Control/ZoomPanel.js * @requires OpenLayers/Control/Navigation.js * @requires OpenLayers/Kinetic.js * @requires OpenLayers/Control/PanPanel.js * @requires GeoExt/widgets/MapPanel.js * @requires GeoExt/widgets/ZoomSlider.js * @requires GeoExt/widgets/tips/ZoomSliderTip.js */ /** api: (define) * module = gxp * class = Viewer * base_link = `Ext.util.Observable `_ */ Ext.namespace("gxp"); /** api: constructor * .. class:: Viewer(config) * * A map viewer application framework that can be extended with plugins * for layer sources and tools. Types of viewers that can be built with * this framework range from simple map viewers to complex web-based GIS * applications with capabilities like feature editing, styling and more. */ /** api: example * A viewer can be added to an HTML page with a script block containing * something like this for a minimal viewer with an OSM layer: * * .. code-block:: javascript * * var app = new gxp.Viewer({ * sources: { * osm: { * ptype: "gxp_osmsource" * } * }, * map: { * center: [0, 0], * zoom: 2, * layers: [{ * source: "osm", * name: "mapnik" * }] * } * }); */ gxp.Viewer = Ext.extend(Ext.util.Observable, { /** private: property[mapPanel] * ``GeoExt.MapPanel`` */ /** api: config[proxy] * ``String`` An optional proxy url which can be used to bypass the same * origin policy. This will be set as ``OpenLayers.ProxyHost``. */ /** api: config[mapItems] * ``Array(Ext.Component)`` * Any items to be added to the map panel. A typical item to put on a map * would be a ``GeoExt.ZoomSlider``. */ /** api: config[mapPlugins] * ``Array(Ext.util.Observable)`` * Any plugins to be added to the map panel. */ /** api: config[portalConfig] * ``Object`` Configuration object for the wrapping container of the * viewer. This will be an ``Ext.Panel`` if it has a ``renderTo`` * property, or an ``Ext.Viewport`` otherwise. */ /** api: config[portalItems] * ``Array`` Items for the portal. A MapPanel will automatically be added * to the portal, unless ``portalConfig`` has ``items`` configured. */ /** api: config[sources] * ``Object`` Layer source configurations for this viewer, keyed by source * id. The source id will be used to reference the layer source in the * ``layers`` array of the ``map`` object. */ /** api: config[map] * ``Object`` Map configuration for this viewer. This object is similar * to the ``GeoExt.MapPanel`` configuration, with the following * exceptions: * * * center: ``Array`` of lon (x) and lat (y) values * * items: not available - use ``mapItems`` instead * * tbar: not available - use :class:`gxp.Tool` plugins to populate * the tbar * * wrapDateLine: ``Boolean`` Should we wrap the dateline? Defaults to * true * * numZoomLevels: ``Integer`` The number of zoom levels to use. * * layers: ``Array(Object)``. Each object has a ``source`` property * referencing a :class:`gxp.plugins.LayerSource`. The viewer will call * the ``createLayerRecord`` of this source with the object as * argument, which will result in a layer being created with the * configuration provided here. * * Valid config options for all layer sources: * * * source: ``String`` referencing a source from ``sources`` * * name: ``String`` - the name from the source's ``store`` (only for * sources that maintain a store) * * visibility: ``Boolean`` - initial layer visibility * * opacity: ``Number`` - initial layer.opacity * * group: ``String`` - group for the layer when the viewer also uses a * :class:`gxp.plugins.LayerTree`. Set this to "background" to make * the layer a base layer * * fixed: ``Boolean`` - Set to true to prevent the layer from being * removed by a :class:`gxp.plugins.RemoveLayer` tool and from being * dragged in a :class:`gxp.plugins.LayerTree` * * selected: ``Boolean`` - Set to true to mark the layer selected * * map: not available, can be configured with ``maxExtent``, * ``numZoomLevels`` and ``theme``. * * restrictedExtent: ``Array`` to be consumed by * ``OpenLayers.Bounds.fromArray`` - the restrictedExtent of the map * * maxExtent: ``Array`` to be consumed by * ``OpenLayers.Bounds.fromArray`` - the maxExtent of the map * * numZoomLevels: ``Number`` - the number of zoom levels if not * available on the first layer * * theme: ``String`` - optional theme for the ``OpenLayers.Map``, as * in ``OpenLayers.Map.theme``. */ /** api: config[defaultToolType] * ``String`` * The default tool plugin type. Default is "gxp_tool" */ defaultToolType: "gxp_tool", /** api: config[tools] * ``Array(`` :class:`gxp.plugins.Tool` ``)`` * Any tools to be added to the viewer. Tools are plugins that will be * plugged into this viewer's ``portal``. The ``tools`` array is usually * populated with configuration objects for plugins (using a ptype), * rather than instances. A default ptype can be configured with this * viewer's ``defaultToolType`` option. */ /** api: property[tools] * ``Object`` Storage of tool instances for this viewer, keyed by id */ tools: null, /** api: config[defaultSourceType] * ``String`` * The default layer source plugin type. */ /** api: property[portalItems] * ``Array(Ext.Component)`` * Items that make up the portal. */ /** api: property[selectedLayer] * ``GeoExt.data.LayerRecord`` The currently selected layer */ selectedLayer: null, /** api: config[field] * :class:`gxp.form.ViewerField` Optional - set by * :class:`gxp.form.ViewerField` so plugins like * :class:`gxp.plugins.FeatureToField` can set the form field's value. */ /** api: property[field] * :class:`gxp.form.ViewerField` Used by plugins to access the form field. * Only available if this viewer is wrapped into an * :class:`Ext.form.ViewerField`. */ /** api: config[authenticate] * ``Function`` A global authentication function that is invoked by * :meth:`doAuthorized` if no user is logged in or the current user is not * authorized. That process is supposed to call :meth:`setAuthorizedRoles` * upon successful authentication, and :meth:`cancelAuthentication` if the * user cancels the login process. Typically this function creates and * opens a login window. Optional, default is null. */ /** api: property[authenticate] * ``Function`` Like the config option above, but this can be set after * configuration e.g. by a plugin that provides authentication. It can * also be accessed to check if an authentication mechanism is available. */ authenticate: null, /** api: property[authorizedRoles] * ``Array`` Roles the application is authorized for. This property is * usually set by the :meth:`setAuthorizedRoles` method, which is * typically called by a component that authenticates the user (e.g. a * login window. After authentication, if the client is authorized to do * everything, this should be set to ``["ROLE_ADMINISTRATOR"]``. * * If this property is undefined, the ``isAuthorized()`` method will * return undefined, so plugins can check for that to do their own auth * checks in this case. So if the application uses an authentication * component (e.g. a login window), it is recommended to set this to * ``[]`` (equivalent to "not authorized to do anything") initially. */ /** api: config[saveErrorText] * ``String`` */ saveErrorText: "Trouble saving: ", /** private: method[constructor] * Construct the viewer. */ constructor: function(config) { // add any custom application events this.addEvents( /** api: event[ready] * Fires when application is ready for user interaction. */ "ready", /** api: event[beforecreateportal] * Fires before the portal is created by the Ext ComponentManager. */ "beforecreateportal", /** api: event[portalready] * Fires after the portal is initialized. */ "portalready", /** api: event[beforelayerselectionchange] * Fired before the selected set of layers changes. Listeners * can return ``false`` to stop the selected layers from being * changed. * * Listeners arguments: * * * layerRecord - ``GeoExt.data.LayerRecord`` the record of the * selected layer, or null if no layer is selected. */ "beforelayerselectionchange", /** api: event[layerselectionchange] * Fired when the selected set of layers changes. * * Listeners arguments: * * * layerRecord - ``GeoExt.data.LayerRecord`` the record of the * selected layer, or null if no layer is selected. */ "layerselectionchange", /** api: event[featureedit] * Fired when features were edited. * * Listener arguments: * * * featureManager - :class:`gxp.plugins.FeatureManager` the * the feature manager that was used for editing * * layer - ``Object`` object with name and source of the layer * that was edited */ "featureedit", /** api: event[authorizationchange] * Fired when the authorizedRoles are changed, e.g. when a user * logs in or out. */ "authorizationchange", /** api: event[beforesave] * Fires before application saves a map. If the listener returns * false, the save is cancelled. * * Listeners arguments: * * * requestConfig - ``Object`` configuration object for the request, * which has the following properties: method, url and data. * * callback - ``Function`` Optional callback function which was * passed on to the save function. */ "beforesave", /** api: event[save] * Fires when the map has been saved. * * Listener arguments: * * id - ``Integer`` The identifier of the saved map */ "save", /** api: event[beforehashchange] * Fires before the hash is updated after saving a map. Return * false in the listener not to update the hash. * * Listeners arguments: * * hash - ``String`` The hash which will be set as * window.location.hash */ "beforehashchange" ); Ext.apply(this, { layerSources: {}, portalItems: [] }); // private array of pending getLayerRecord requests this.createLayerRecordQueue = []; (config.loadConfig || this.loadConfig).call(this, config, this.applyConfig); gxp.Viewer.superclass.constructor.apply(this, arguments); }, /** api: method[selectLayer] * :arg record: ``GeoExt.data.LayerRecord``` Layer record. Call with no * layer record to remove layer selection. * :returns: ``Boolean`` Layers were set as selected. * * TODO: change to selectLayers (plural) */ selectLayer: function(record) { record = record || null; var changed = false; var allow = this.fireEvent("beforelayerselectionchange", record); if (allow !== false) { changed = true; if (this.selectedLayer) { this.selectedLayer.set("selected", false); } this.selectedLayer = record; if (this.selectedLayer) { this.selectedLayer.set("selected", true); } this.fireEvent("layerselectionchange", record); } return changed; }, /** api: method[loadConfig] * :arg config: ``Object`` The config object passed to the constructor. * * Subclasses that load config asynchronously can override this to load * any configuration before applyConfig is called. */ loadConfig: function(config) { this.applyConfig(config); }, applyConfig: function(config) { this.initialConfig = Ext.apply({}, config); Ext.apply(this, this.initialConfig); this.load(); }, load: function() { // pass on any proxy config to OpenLayers if (this.proxy) { OpenLayers.ProxyHost = this.proxy; } this.initMapPanel(); this.initTools(); // initialize all layer source plugins var config, queue = []; for (var key in this.sources) { queue.push(this.createSourceLoader(key)); } // create portal when dom is ready queue.push(function(done) { Ext.onReady(function() { this.initPortal(); done(); }, this); }); gxp.util.dispatch(queue, this.activate, this); }, createSourceLoader: function(key) { return function(done) { var config = this.sources[key]; config.projection = this.initialConfig.map.projection; this.addLayerSource({ id: key, config: config, callback: done, fallback: function(source, msg, details) { // TODO: log these issues somewhere that the app can display // them after loading. // console.log(arguments); done(); }, scope: this }); }; }, addLayerSource: function(options) { var id = options.id || Ext.id(null, "gxp-source-"); var source; var config = options.config; config.id = id; try { source = Ext.ComponentMgr.createPlugin( config, this.defaultSourceType ); } catch (err) { throw new Error("Could not create new source plugin with ptype: " + options.config.ptype); } source.on({ ready: { fn: function() { var callback = options.callback || Ext.emptyFn; callback.call(options.scope || this, id); }, scope: this, single: true }, failure: { fn: function() { var fallback = options.fallback || Ext.emptyFn; delete this.layerSources[id]; fallback.apply(options.scope || this, arguments); }, scope: this, single: true } }); this.layerSources[id] = source; source.init(this); return source; }, initMapPanel: function() { var config = Ext.apply({}, this.initialConfig.map); var mapConfig = {}; var baseLayerConfig = { wrapDateLine: config.wrapDateLine !== undefined ? config.wrapDateLine : true, maxResolution: config.maxResolution, numZoomLevels: config.numZoomLevels, displayInLayerSwitcher: false }; // split initial map configuration into map and panel config if (this.initialConfig.map) { var props = "theme,controls,resolutions,projection,units,maxExtent,restrictedExtent,maxResolution,numZoomLevels,panMethod".split(","); var prop; for (var i=props.length-1; i>=0; --i) { prop = props[i]; if (prop in config) { mapConfig[prop] = config[prop]; delete config[prop]; } } } this.mapPanel = Ext.ComponentMgr.create(Ext.applyIf({ xtype: config.xtype || "gx_mappanel", map: Ext.applyIf({ theme: mapConfig.theme || null, controls: mapConfig.controls || [ new OpenLayers.Control.Navigation({ zoomWheelOptions: {interval: 250}, dragPanOptions: {enableKinetic: true} }), new OpenLayers.Control.PanPanel(), new OpenLayers.Control.ZoomPanel(), new OpenLayers.Control.Attribution() ], maxExtent: mapConfig.maxExtent && OpenLayers.Bounds.fromArray(mapConfig.maxExtent), restrictedExtent: mapConfig.restrictedExtent && OpenLayers.Bounds.fromArray(mapConfig.restrictedExtent), numZoomLevels: mapConfig.numZoomLevels || 20 }, mapConfig), center: config.center && new OpenLayers.LonLat(config.center[0], config.center[1]), resolutions: config.resolutions, forceInitialExtent: true, layers: [new OpenLayers.Layer(null, baseLayerConfig)], items: this.mapItems, plugins: this.mapPlugins, tbar: config.tbar || new Ext.Toolbar({ hidden: true }) }, config)); this.mapPanel.getTopToolbar().on({ afterlayout: this.mapPanel.map.updateSize, show: this.mapPanel.map.updateSize, hide: this.mapPanel.map.updateSize, scope: this.mapPanel.map }); this.mapPanel.layers.on({ "add": function(store, records) { // check selected layer status var record; for (var i=records.length-1; i>= 0; i--) { record = records[i]; if (record.get("selected") === true) { this.selectLayer(record); } } }, "remove": function(store, record) { if (record.get("selected") === true) { this.selectLayer(); } }, scope: this }); }, initTools: function() { this.tools = {}; if (this.initialConfig.tools && this.initialConfig.tools.length > 0) { var tool; for (var i=0, len=this.initialConfig.tools.length; i=0; --i) { if (~this.authorizedRoles.indexOf(roles[i])) { authorized = true; break; } } } return authorized; }, /** api: method[setAuthorizedRoles] * :arg authorizedRoles: ``Array`` * * Change the authorized roles. */ setAuthorizedRoles: function(authorizedRoles) { this.authorizedRoles = authorizedRoles; this.fireEvent("authorizationchange"); }, /** api: method[cancelAuthentication] * Cancel an authentication process. */ cancelAuthentication: function() { if (this._authFn) { this.un("authorizationchange", this._authFn, this); } this.fireEvent("authorizationchange"); }, /** api: method[isAuthenticated] * :returns: ``Boolean`` The user has authenticated. * * Determines whether a user has logged in. In cases where the application * doesn't provide a login dialog, the user will be considered logged in. * In this same case, where components require authentication, the browser * will prompt for credentials when needed. */ isAuthenticated: function(role) { /** * If the application supports authentication, we expect a list of * authorized roles to be set (length zero if user has not logged in). * If the application does not support authentication, authorizedRoles * should be undefined. In this case, we return true so that components * that require authentication can still be enabled. This leaves the * authentication challenge up to the browser. */ return !this.authorizedRoles || this.authorizedRoles.length > 0; }, /** api: method[doAuthorized] * :param roles: ``Array`` Roles required for invoking the action * :param callback: ``Function`` The action to perform * :param scope: ``Object`` The execution scope for the callback * * Performs an action (defined as ``callback`` function), but only if * the user is authorized to perform it. If no user is logged in or the * logged in user is not authorized, the viewer's :meth:`authenticate` * function will be invoked. This method is usually called by plugins. */ doAuthorized: function(roles, callback, scope) { if (this.isAuthorized(roles) || !this.authenticate) { window.setTimeout(function() { callback.call(scope); }, 0); } else { this.authenticate(); this._authFn = function authFn() { delete this._authFn; this.doAuthorized(roles, callback, scope, true); }; this.on("authorizationchange", this._authFn, this, {single: true}); } }, /** private: method[save] * * Saves the map config and displays the URL in a window. */ save: function(callback, scope) { var configStr = Ext.util.JSON.encode(this.getState()); var method, url; if (this.id) { method = "PUT"; url = "../maps/" + this.id; } else { method = "POST"; url = "../maps/"; } var requestConfig = { method: method, url: url, data: configStr }; if (this.fireEvent("beforesave", requestConfig, callback) !== false) { OpenLayers.Request.issue(Ext.apply(requestConfig, { callback: function(request) { this.handleSave(request); if (callback) { callback.call(scope || this, request); } }, scope: this })); } }, /** private: method[handleSave] * :arg: ``XMLHttpRequest`` */ handleSave: function(request) { if (request.status == 200) { var config = Ext.util.JSON.decode(request.responseText); var mapId = config.id; if (mapId) { this.id = mapId; var hash = "#maps/" + mapId; if (this.fireEvent("beforehashchange", hash) !== false) { window.location.hash = hash; } this.fireEvent("save", this.id); } } else { if (window.console) { console.warn(this.saveErrorText + request.responseText); } } }, /** api: method[destroy] */ destroy: function() { //TODO there is probably more that needs to be destroyed this.mapPanel.destroy(); this.portal && this.portal.destroy(); } }); (function() { // OGC "standardized rendering pixel size" OpenLayers.DOTS_PER_INCH = 25.4 / 0.28; })(); /** FILE: widgets/EmbedMapDialog.js **/ /** * Copyright (c) 2008-2011 The Open Planning Project * * Published under the GPL license. * See https://github.com/opengeo/gxp/raw/master/license.txt for the full text * of the license. */ /** * @requires util.js */ Ext.namespace("gxp"); /** api: (define) * module = gxp * class = EmbedMapDialog * base_link = `Ext.Container `_ */ /** api: example * Show a :class:`gxp.EmbedMapDialog` in a window, using "viewer.html" in the * current path as url: * * .. code-block:: javascript * * new Ext.Window({ * title: "Export Map", * layout: "fit", * width: 380, * autoHeight: true, * items: [{ * xtype: "gxp_embedmapdialog", * url: "viewer.html" * }] * }).show(); */ /** api: constructor * .. class:: EmbedMapDialog(config) * * A dialog for configuring a map iframe to embed on external web pages. */ gxp.EmbedMapDialog = Ext.extend(Ext.Container, { /** api: config[url] * ``String`` the url to use as the iframe's src of the embed snippet. Can * be a url relative to the current href and will be converted to an * absolute one. */ url: null, /** api: property[url] * ``String`` the url to use as the iframe's src of the embed snippet. Can * be a url relative to the current href and will be converted to an * absolute one. */ url: null, /* begin i18n */ /** api: config[publishMessage] ``String`` i18n */ publishMessage: "Your map is ready to be published to the web! Simply copy the following HTML to embed the map in your website:", /** api: config[heightLabel] ``String`` i18n */ heightLabel: 'Height', /** api: config[widthLabel] ``String`` i18n */ widthLabel: 'Width', /** api: config[mapSizeLabel] ``String`` i18n */ mapSizeLabel: 'Map Size', /** api: config[miniSizeLabel] ``String`` i18n */ miniSizeLabel: 'Mini', /** api: config[smallSizeLabel] ``String`` i18n */ smallSizeLabel: 'Small', /** api: config[premiumSizeLabel] ``String`` i18n */ premiumSizeLabel: 'Premium', /** api: config[largeSizeLabel] ``String`` i18n */ largeSizeLabel: 'Large', /* end i18n */ /** private: property[snippetArea] */ snippetArea: null, /** private: property[heightField] */ heightField: null, /** private: property[widthField] */ widthField: null, /** private: method[initComponent] */ initComponent: function() { Ext.apply(this, this.getConfig()); gxp.EmbedMapDialog.superclass.initComponent.call(this); }, /** api: method[getIframeHTML] * :returns: ``String`` the HTML needed to create the iframe * * Get the HTML needed to create the iframe. */ getIframeHTML: function() { return this.snippetArea.getValue(); }, /** private: method[updateSnippet] */ updateSnippet: function() { this.snippetArea.setValue( ''); if (this.snippetArea.isVisible() === true) { this.snippetArea.focus(true, 100); } }, /** private: method[getConfig] */ getConfig: function() { this.snippetArea = new Ext.form.TextArea({ height: 70, selectOnFocus: true, readOnly: true }); var numFieldListeners = { "change": this.updateSnippet, "specialkey": function(f, e) { e.getKey() == e.ENTER && this.updateSnippet(); }, scope: this }; this.heightField = new Ext.form.NumberField({ width: 50, value: 400, listeners: numFieldListeners }); this.widthField = new Ext.form.NumberField({ width: 50, value: 600, listeners: numFieldListeners }); var adjustments = new Ext.Container({ layout: "column", defaults: { border: false, xtype: "box" }, items: [ {autoEl: {cls: "gxp-field-label", html: this.mapSizeLabel}}, new Ext.form.ComboBox({ editable: false, width: 75, store: new Ext.data.SimpleStore({ fields: ["name", "height", "width"], data: [ [this.miniSizeLabel, 100, 100], [this.smallSizeLabel, 200, 300], [this.largeSizeLabel, 400, 600], [this.premiumSizeLabel, 600, 800] ] }), triggerAction: 'all', displayField: 'name', value: this.largeSizeLabel, mode: 'local', listeners: { "select": function(combo, record, index) { this.widthField.setValue(record.get("width")); this.heightField.setValue(record.get("height")); this.updateSnippet(); }, scope: this } }), {autoEl: {cls: "gxp-field-label", html: this.heightLabel}}, this.heightField, {autoEl: {cls: "gxp-field-label", html: this.widthLabel}}, this.widthField ] }); return { border: false, defaults: { border: false, cls: "gxp-export-section", xtype: "container", layout: "fit" }, items: [{ items: [adjustments] }, { xtype: "box", autoEl: { tag: "p", html: this.publishMessage } }, { items: [this.snippetArea] }], listeners: { "afterrender": this.updateSnippet, scope: this } }; } }); /** api: xtype = gxp_embedmapdialog */ Ext.reg('gxp_embedmapdialog', gxp.EmbedMapDialog); /** FILE: widgets/RulePanel.js **/ /** * Copyright (c) 2008-2011 The Open Planning Project * * Published under the GPL license. * See https://github.com/opengeo/gxp/raw/master/license.txt for the full text * of the license. */ /** * @include widgets/ScaleLimitPanel.js * @include widgets/TextSymbolizer.js * @include widgets/PolygonSymbolizer.js * @include widgets/LineSymbolizer.js * @include widgets/PointSymbolizer.js * @include widgets/FilterBuilder.js */ /** api: (define) * module = gxp * class = RulePanel * base_link = `Ext.TabPanel `_ */ Ext.namespace("gxp"); /** api: constructor * .. class:: RulePanel(config) * * Create a panel for assembling SLD rules. */ gxp.RulePanel = Ext.extend(Ext.TabPanel, { /** api: property[fonts] * ``Array(String)`` List of fonts for the font combo. If not set, * defaults to the list provided by the . */ fonts: undefined, /** api: property[symbolType] * ``String`` One of "Point", "Line", or "Polygon". If no rule is * provided, default is "Point". */ symbolType: "Point", /** api: config[rule] * ``OpenLayers.Rule`` Optional rule provided in the initial * configuration. If a rule is provided and no `symbolType` is provided, * the symbol type will be derived from the first symbolizer found in the * rule. */ rule: null, /** private: property[attributes] * ``GeoExt.data.AttributeStore`` A configured attributes store for use * in the filter property combo. */ attributes: null, /** private: property[pointGraphics] * ``Array`` A list of objects to be used as the root of the data for a * JsonStore. These will become records used in the selection of * a point graphic. If an object in the list has no "value" property, * the user will be presented with an input to provide their own URL * for an external graphic. By default, names of well-known marks are * provided. In addition, the default list will produce a record with * display of "external" that create an input for an external graphic * URL. * * Fields: * * * display - ``String`` The name to be displayed to the user. * * preview - ``String`` URL to a graphic for preview. * * value - ``String`` Value to be sent to the server. * * mark - ``Boolean`` The value is a well-known name for a mark. If * false, the value will be assumed to be a url for an external graphic. */ /** private: property[nestedFilters] * ``Boolean`` Allow addition of nested logical filters. This sets the * allowGroups property of the filter builder. Default is true. */ nestedFilters: true, /** private: property[minScaleDenominatorLimit] * ``Number`` Lower limit for scale denominators. Default is what you get * when you assume 20 zoom levels starting with the world in Spherical * Mercator on a single 256 x 256 tile at zoom 0 where the zoom factor is * 2. */ minScaleDenominatorLimit: Math.pow(0.5, 19) * 40075016.68 * 39.3701 * OpenLayers.DOTS_PER_INCH / 256, /** private: property[maxScaleDenominatorLimit] * ``Number`` Upper limit for scale denominators. Default is what you get * when you project the world in Spherical Mercator onto a single * 256 x 256 pixel tile and assume OpenLayers.DOTS_PER_INCH (this * corresponds to zoom level 0 in Google Maps). */ maxScaleDenominatorLimit: 40075016.68 * 39.3701 * OpenLayers.DOTS_PER_INCH / 256, /** private: property [scaleLevels] * ``Number`` Number of scale levels to assume. This is only for scaling * values exponentially along the slider. Scale values are not * required to one of the discrete levels. Default is 20. */ scaleLevels: 20, /** private: property[scaleSliderTemplate] * ``String`` Template for the tip displayed by the scale threshold slider. * * Can be customized using the following keywords in curly braces: * * * zoom - the zoom level * * scale - the scale denominator * * type - "Max" or "Min" denominator * * scaleType - "Min" or "Max" scale (sense is opposite of type) * * Default is "{scaleType} Scale 1:{scale}". */ scaleSliderTemplate: "{scaleType} Scale 1:{scale}", /** private: method[modifyScaleTipContext] * Called from the multi-slider tip's getText function. The function * will receive two arguments - a reference to the panel and a data * object. The data object will have scale, zoom, and type properties * already calculated. Other properties added to the data object * are available to the . */ modifyScaleTipContext: Ext.emptyFn, /** i18n */ labelFeaturesText: "Label Features", labelsText: "Labels", basicText: "Basic", advancedText: "Advanced", limitByScaleText: "Limit by scale", limitByConditionText: "Limit by condition", symbolText: "Symbol", nameText: "Name", /** private */ initComponent: function() { var defConfig = { plain: true, border: false }; Ext.applyIf(this, defConfig); if(!this.rule) { this.rule = new OpenLayers.Rule({ name: this.uniqueRuleName() }); } else { if (!this.initialConfig.symbolType) { this.symbolType = this.getSymbolTypeFromRule(this.rule) || this.symbolType; } } this.activeTab = 0; this.textSymbolizer = new gxp.TextSymbolizer({ symbolizer: this.getTextSymbolizer(), attributes: this.attributes, fonts: this.fonts, listeners: { change: function(symbolizer) { this.fireEvent("change", this, this.rule); }, scope: this } }); /** * The interpretation here is that scale values of zero are equivalent to * no scale value. If someone thinks that a scale value of zero should have * a different interpretation, this needs to be changed. */ this.scaleLimitPanel = new gxp.ScaleLimitPanel({ maxScaleDenominator: this.rule.maxScaleDenominator || undefined, limitMaxScaleDenominator: !!this.rule.maxScaleDenominator, maxScaleDenominatorLimit: this.maxScaleDenominatorLimit, minScaleDenominator: this.rule.minScaleDenominator || undefined, limitMinScaleDenominator: !!this.rule.minScaleDenominator, minScaleDenominatorLimit: this.minScaleDenominatorLimit, scaleLevels: this.scaleLevels, scaleSliderTemplate: this.scaleSliderTemplate, modifyScaleTipContext: this.modifyScaleTipContext, listeners: { change: function(comp, min, max) { this.rule.minScaleDenominator = min; this.rule.maxScaleDenominator = max; this.fireEvent("change", this, this.rule); }, scope: this } }); this.filterBuilder = new gxp.FilterBuilder({ allowGroups: this.nestedFilters, filter: this.rule && this.rule.filter && this.rule.filter.clone(), attributes: this.attributes, listeners: { change: function(builder) { var filter = builder.getFilter(); this.rule.filter = filter; this.fireEvent("change", this, this.rule); }, scope: this } }); this.items = [{ title: this.labelsText, autoScroll: true, bodyStyle: {"padding": "10px"}, items: [{ xtype: "fieldset", title: this.labelFeaturesText, autoHeight: true, checkboxToggle: true, collapsed: !this.hasTextSymbolizer(), items: [ this.textSymbolizer ], listeners: { collapse: function() { OpenLayers.Util.removeItem(this.rule.symbolizers, this.getTextSymbolizer()); this.fireEvent("change", this, this.rule); }, expand: function() { this.setTextSymbolizer(this.textSymbolizer.symbolizer); this.fireEvent("change", this, this.rule); }, scope: this } }] }]; if (this.getSymbolTypeFromRule(this.rule) || this.symbolType) { this.items = [{ title: this.basicText, autoScroll: true, items: [this.createHeaderPanel(), this.createSymbolizerPanel()] }, this.items[0], { title: this.advancedText, defaults: { style: { margin: "7px" } }, autoScroll: true, items: [{ xtype: "fieldset", title: this.limitByScaleText, checkboxToggle: true, collapsed: !(this.rule && (this.rule.minScaleDenominator || this.rule.maxScaleDenominator)), autoHeight: true, items: [this.scaleLimitPanel], listeners: { collapse: function() { delete this.rule.minScaleDenominator; delete this.rule.maxScaleDenominator; this.fireEvent("change", this, this.rule); }, expand: function() { /** * Start workaround for * http://projects.opengeo.org/suite/ticket/676 */ var tab = this.getActiveTab(); this.activeTab = null; this.setActiveTab(tab); /** * End workaround for * http://projects.opengeo.org/suite/ticket/676 */ var changed = false; if (this.scaleLimitPanel.limitMinScaleDenominator) { this.rule.minScaleDenominator = this.scaleLimitPanel.minScaleDenominator; changed = true; } if (this.scaleLimitPanel.limitMaxScaleDenominator) { this.rule.maxScaleDenominator = this.scaleLimitPanel.maxScaleDenominator; changed = true; } if (changed) { this.fireEvent("change", this, this.rule); } }, scope: this } }, { xtype: "fieldset", title: this.limitByConditionText, checkboxToggle: true, collapsed: !(this.rule && this.rule.filter), autoHeight: true, items: [this.filterBuilder], listeners: { collapse: function(){ delete this.rule.filter; this.fireEvent("change", this, this.rule); }, expand: function(){ var changed = false; this.rule.filter = this.filterBuilder.getFilter(); this.fireEvent("change", this, this.rule); }, scope: this } }] }]; } this.items[0].autoHeight = true; this.addEvents( /** api: events[change] * Fires when any rule property changes. * * Listener arguments: * * panel - :class:`gxp.RulePanel` This panel. * * rule - ``OpenLayers.Rule`` The updated rule. */ "change" ); this.on({ tabchange: function(panel, tab) { tab.doLayout(); }, scope: this }); gxp.RulePanel.superclass.initComponent.call(this); }, /** private: method[hasTextSymbolizer] */ hasTextSymbolizer: function() { var candidate, symbolizer; for (var i=0, ii=this.rule.symbolizers.length; i`_ */ Ext.namespace("gxp"); /** api: constructor * .. class:: StylePropertiesDialog(config) * * Create a dialog for editing properties of a UserStyle. */ gxp.StylePropertiesDialog = Ext.extend(Ext.Container, { /* i18n */ titleText: "General", nameFieldText: "Name", titleFieldText: "Title", abstractFieldText: "Abstract", /* ~i18n */ /** api: config[userStyle] * ``OpenLayers.Style`` */ /** api: property[userStyle] * ``OpenLayers.Style`` */ userStyle: null, /** api: config[nameEditable] * ``Boolean`` Set to false if the name of the style should not be * editable. */ /** private: method[initComponent] */ initComponent: function() { var listeners = { "change": function(field, value) { this.userStyle[field.name] = value; this.fireEvent("change", this, this.userStyle); }, scope: this }; var defConfig = { layout: "form", items: [{ xtype: "fieldset", title: this.titleText, labelWidth: 75, defaults: { xtype: "textfield", anchor: "100%", listeners: listeners }, items: [{ xtype: this.initialConfig.nameEditable ? "textfield" : "displayfield", fieldLabel: this.nameFieldText, name: "name", value: this.userStyle.name, maskRe: /[A-Za-z0-9_]/ }, { fieldLabel: this.titleFieldText, name: "title", value: this.userStyle.title }, { xtype: "textarea", fieldLabel: this.abstractFieldText, name: "description", value: this.userStyle.description }] }] }; Ext.applyIf(this, defConfig); this.addEvents( /** api: events[change] * Fires when any style property changes. * * Listener arguments: * * component - ``gxp.StylePropertiesDialog`` This dialog. * * userStyle - ``OpenLayers.Style`` The updated style. */ "change" ); gxp.StylePropertiesDialog.superclass.initComponent.apply(this, arguments); } }); /** api: xtype = gxp_styleproperties */ Ext.reg('gxp_stylepropertiesdialog', gxp.StylePropertiesDialog); /** FILE: 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} 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 - {} * 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 - {} * 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 - {} * 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.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 method compares this value (which is the one * from the previous 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 * {} 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 - {} * 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 - {} * 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 - {} * 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} * 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 - {} * 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 - {} * 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 - {} * * 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 - {} * * 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 - {} * * 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 - {} * * 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 - {} * * 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 - {} * * 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 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 - {} * 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} 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" }); /** FILE: 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.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 - {} * 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 - {} 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 - {} * 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 - {} * * 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 - {} * 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 - {} * * 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 - {} * * 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 - {} * * 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} * * 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 - {} */ 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()} 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 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 - {} original geometry of the * invalid point * goodComponent - {} 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 - {} * * 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 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 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); }; /** FILE: 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/Renderer/Elements.js */ /** * Class: OpenLayers.Renderer.VML * Render vector features in browsers with VML capability. Construct a new * VML renderer with the 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.Renderer.VML = OpenLayers.Class(OpenLayers.Renderer.Elements, { /** * Property: xmlns * {String} XML Namespace URN */ xmlns: "urn:schemas-microsoft-com:vml", /** * 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: {}, /** * Property: offset * {Object} Hash with "x" and "y" properties */ offset: null, /** * Constructor: OpenLayers.Renderer.VML * Create a new VML renderer. * * Parameters: * 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: * {Boolean} The browser supports the VML renderer */ supported: function() { return !!(document.namespaces); }, /** * Method: setExtent * Set the renderer's extent * * 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. */ 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} the size of the drawing surface */ 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} * 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 = "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: 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 - {} */ 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); } } // 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"; if (!(style.graphicWidth && style.graphicHeight)) { fill.aspect = "atmost"; } } if (fill.parentNode != node) { node.appendChild(fill); } } // 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; } } // 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; }, /** * 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; 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; // will be called again by the onreadystate handler 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"; // 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 rot = rotation * Math.PI / 180; var sintheta = Math.sin(rot); var costheta = Math.cos(rot); // 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; // 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"; }, /** * 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; } }, /** * 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 - {} */ 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); // 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(); } }, /** * 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"; } }, /** * 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; } // 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 */ 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); } return (subType == nodeName); }, /** * 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"); }, /** * 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"); }, /************************************** * * * GEOMETRY DRAWING FUNCTIONS * * * **************************************/ /** * Method: drawPoint * Render a point * * Parameters: * node - {DOMElement} * geometry - {} * * Returns: * {DOMElement} or false if the point could not be drawn */ drawPoint: function(node, geometry) { return this.drawCircle(node, geometry, 1); }, /** * Method: drawCircle * Render a circle. * Size and Center a circle given geometry (x,y center) and radius * * Parameters: * node - {DOMElement} * 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(); 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; }, /** * Method: drawLineString * Render a linestring. * * Parameters: * node - {DOMElement} * geometry - {} * * Returns: * {DOMElement} */ drawLineString: function(node, geometry) { return this.drawLine(node, geometry, false); }, /** * Method: drawLinearRing * Render a linearring * * Parameters: * node - {DOMElement} * geometry - {} * * Returns: * {DOMElement} */ drawLinearRing: function(node, geometry) { return this.drawLine(node, geometry, true); }, /** * Method: DrawLine * Render a line. * * Parameters: * node - {DOMElement} * geometry - {} * closeLine - {Boolean} Close the line? (make it a ring?) * * Returns: * {DOMElement} */ 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; }, /** * Method: drawPolygon * Render a polygon * * Parameters: * node - {DOMElement} * 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} * * 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 - {} */ 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: moveRoot * moves this renderer's root to a different renderer. * * Parameters: * 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(); }, /** * 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; // 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 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.Renderer.VML.LABEL_SHIFT * {Object} */ OpenLayers.Renderer.VML.LABEL_SHIFT = { "l": 0, "c": .5, "r": 1, "t": 0, "m": .5, "b": 1 }; /** FILE: 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.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 - {} * 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 - {} * 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 - {} * 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 - {} */ 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 - {} * style - {} * * 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 - {} * 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 - {} * 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 - {} * 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= 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 - {} 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 - {} * 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 - {} * 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 - {} * 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} * 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} * 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 (} */ 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 - {} * * Returns: * {)} */ eraseFeatures: function(features) { if(!(OpenLayers.Util.isArray(features))) { features = [features]; } for(var i=0; i */ 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" }); /** FILE: widgets/WMSStylesDialog.js **/ /** * Copyright (c) 2008-2011 The Open Planning Project * * Published under the GPL license. * See https://github.com/opengeo/gxp/raw/master/license.txt for the full text * of the license. */ /** * @require util.js * @require widgets/RulePanel.js * @require widgets/StylePropertiesDialog.js * @requires OpenLayers/Renderer/SVG.js * @requires OpenLayers/Renderer/VML.js * @requires OpenLayers/Renderer/Canvas.js * @require OpenLayers/Style2.js * @require OpenLayers/Format/SLD/v1_0_0_GeoServer.js * @require GeoExt/data/AttributeStore.js * @require GeoExt/widgets/WMSLegend.js * @require GeoExt/widgets/VectorLegend.js */ /** api: (define) * module = gxp * class = WMSStylesDialog * base_link = `Ext.Container `_ */ Ext.namespace("gxp"); /** api: constructor * .. class:: WMSStylesDialog(config) * * Create a dialog for selecting and layer styles. If the WMS supports * GetStyles, styles can also be edited. The dialog does not provide any * means of writing modified styles back to the server. To save styles, * configure the dialog with a :class:`gxp.plugins.StyleWriter` plugin * and call the ``saveStyles`` method. * * Note: when this component is included in a build, * ``OpenLayers.Renderer.defaultSymbolizer`` will be set to the SLD * defaults. In addition, the OpenLayers SLD v1 parser will be patched * to support vendor specific extensions added to SLD by GeoTools. */ gxp.WMSStylesDialog = Ext.extend(Ext.Container, { /** api: config[addStyleText] (i18n) */ addStyleText: "Add", /** api: config[addStyleTip] (i18n) */ addStyleTip: "Add a new style", /** api: config[chooseStyleText] (i18n) */ chooseStyleText: "Choose style", /** api: config[addStyleText] (i18n) */ deleteStyleText: "Remove", /** api: config[addStyleTip] (i18n) */ deleteStyleTip: "Delete the selected style", /** api: config[addStyleText] (i18n) */ editStyleText: "Edit", /** api: config[addStyleTip] (i18n) */ editStyleTip: "Edit the selected style", /** api: config[addStyleText] (i18n) */ duplicateStyleText: "Duplicate", /** api: config[addStyleTip] (i18n) */ duplicateStyleTip: "Duplicate the selected style", /** api: config[addStyleText] (i18n) */ addRuleText: "Add", /** api: config[addStyleTip] (i18n) */ addRuleTip: "Add a new rule", /** api: config[newRuleText] (i18n) */ newRuleText: "New Rule", /** api: config[addStyleText] (i18n) */ deleteRuleText: "Remove", /** api: config[addStyleTip] (i18n) */ deleteRuleTip: "Delete the selected rule", /** api: config[addStyleText] (i18n) */ editRuleText: "Edit", /** api: config[addStyleTip] (i18n) */ editRuleTip: "Edit the selected rule", /** api: config[addStyleText] (i18n) */ duplicateRuleText: "Duplicate", /** api: config[addStyleTip] (i18n) */ duplicateRuleTip: "Duplicate the selected rule", /** api: config[cancelText] (i18n) */ cancelText: "Cancel", /** api: config[saveText] (i18n) */ saveText: "Save", /** api: config[stylePropertiesWindowTitle] (i18n) */ styleWindowTitle: "User Style: {0}", /** api: config[ruleWindowTitle] (i18n) */ ruleWindowTitle: "Style Rule: {0}", /** api: config[stylesFieldsetTitle] (i18n) */ stylesFieldsetTitle: "Styles", /** api: config[rulesFieldsetTitle] (i18n) */ rulesFieldsetTitle: "Rules", /** api: config[errorTitle] (i18n) */ errorTitle: "Error saving style", /** api: config[errorMsg] (i18n) */ errorMsg: "There was an error saving the style back to the server.", //TODO create a StylesStore which can read styles using GetStyles. Create // subclasses for that store with writing capabilities, e.g. // for GeoServer's RESTconfig API. This should replace the current // StyleWriter plugins. /** api: config[layerRecord] * ``GeoExt.data.LayerRecord`` The layer to edit/select styles for. */ /** private: property[layerRecord] * ``GeoExt.data.LayerRecord`` The layer to edit/select styles for. */ layerRecord: null, /** api: config[styleName] * ``String`` A style's name to select in the styles combo box. Optional. * If not provided, the layer's current style will be selected. */ /** api: config[stylesComboOptions] * ``Object`` configuration options to pass to the styles combo of this * dialog. Optional. */ /** api: config[layerDescription] * ``Object`` Array entry of a DescribeLayer response as read by * ``OpenLayers.Format.WMSDescribeLayer``. Optional. If not provided, * a DescribeLayer request will be issued to the WMS. */ /** private: property[layerDescription] * ``Object`` Array entry of a DescribeLayer response as read by * ``OpenLayers.Format.WMSDescribeLayer``. */ layerDescription: null, /** private: property[symbolType] * ``Point`` or ``Line`` or ``Polygon`` - the primary symbol type for the * layer. This is the symbolizer type of the first symbolizer of the * first rule of the current layer style. Only available if the WMS * supports GetStyles. */ symbolType: null, /** api: property[stylesStore] * ``Ext.data.Store`` A store representing the styles returned from * GetCapabilities and GetStyles. It has "name", "title", "abstract", * "legend" and "userStyle" fields. If the WMS supports GetStyles, the * "legend" field will not be available. If it does not, the "userStyle" * field will not be available. */ stylesStore: null, /** api: property[selectedStyle] * ``Ext.data.Record`` The currently selected style from the * ``stylesStore``. */ selectedStyle: null, /** private: property[selectedRule] * ``OpenLayers.Rule`` The currently selected rule, or null if none * selected. */ selectedRule: null, /** api: config[editable] * ``Boolean`` Set to false if styles should not be editable. Default is * true. */ /** api: property[editable] * ``Boolean`` Read-only once the dialog is rendered. True if this * component could gather enough information to allow styles being edited, * false otherwise. This is not supposed to be read before the * ``ready`` event is fired. */ editable: true, /** private: property[modified] * ``Boolean`` Will be true if styles were modified. Initial state is * false. */ modified: false, /** private: config[dialogCls] * ``Ext.Component`` The dialogue class to use. Default is ``Ext.Window``. * If using e.g. ``Ext.Container``, override the ``showDlg`` method to * add the dialogue to a container. */ dialogCls: Ext.Window, /** private: method[initComponent] */ initComponent: function() { this.addEvents( /** api: event[ready] * Fires when this component is ready for user interaction. */ "ready", /** api: event[modified] * Fires on every style modification. * * Listener arguments: * * * :class:`gxp.WMSStylesDialog` this component * * ``String`` the name of the modified style */ "modified", /** api: event[styleselected] * Fires whenever an existing style is selected from this dialog's * Style combo box. * * Listener arguments: * * * :class:`gxp.WMSStylesDialog` this component * * ``String`` the name of the selected style */ "styleselected", /** api: event[beforesaved] * Fires before the styles are saved (using a * :class:`gxp.plugins.StyleWriter` plugin) * * Listener arguments: * * * :class:`gxp.WMSStylesDialog` this component * * ``Object`` options for the ``write`` method of the * :class:`gxp.plugins.StyleWriter` */ "beforesaved", /** api: event[saved] * Fires when a style was successfully saved. Applications should * listen for this event and redraw layers with the currently * selected style. * * Listener arguments: * * * :class:`gxp.WMSStylesDialog` this component * * ``String`` the name of the currently selected style */ "saved" ); var defConfig = { layout: "form", disabled: true, items: [{ xtype: "fieldset", title: this.stylesFieldsetTitle, labelWidth: 85, style: "margin-bottom: 0;" }, { xtype: "toolbar", style: "border-width: 0 1px 1px 1px; margin-bottom: 10px;", items: [ { xtype: "button", iconCls: "add", text: this.addStyleText, tooltip: this.addStyleTip, handler: this.addStyle, scope: this }, { xtype: "button", iconCls: "delete", text: this.deleteStyleText, tooltip: this.deleteStyleTip, handler: function() { this.stylesStore.remove(this.selectedStyle); }, scope: this }, { xtype: "button", iconCls: "edit", text: this.editStyleText, tooltip: this.editStyleTip, handler: function() { this.editStyle(); }, scope: this }, { xtype: "button", iconCls: "duplicate", text: this.duplicateStyleText, tooltip: this.duplicateStyleTip, handler: function() { var prevStyle = this.selectedStyle; var newStyle = prevStyle.get( "userStyle").clone(); newStyle.isDefault = false; newStyle.name = this.newStyleName(); var store = this.stylesStore; store.add(new store.recordType({ "name": newStyle.name, "title": newStyle.title, "abstract": newStyle.description, "userStyle": newStyle })); this.editStyle(prevStyle); }, scope: this } ] }] }; Ext.applyIf(this, defConfig); this.createStylesStore(); this.on({ "beforesaved": function() { this._saving = true; }, "saved": function() { delete this._saving; }, "savefailed": function() { Ext.Msg.show({ title: this.errorTitle, msg: this.errorMsg, icon: Ext.MessageBox.ERROR, buttons: {ok: true} }); delete this._saving; }, "render": function() { gxp.util.dispatch([this.getStyles], function() { this.enable(); }, this); }, scope: this }); gxp.WMSStylesDialog.superclass.initComponent.apply(this, arguments); }, /** api: method[addStyle] * Creates a new style and selects it in the styles combo. */ addStyle: function() { if(!this._ready) { this.on("ready", this.addStyle, this); return; } var prevStyle = this.selectedStyle; var store = this.stylesStore; var newStyle = new OpenLayers.Style(null, { name: this.newStyleName(), rules: [this.createRule()] }); store.add(new store.recordType({ "name": newStyle.name, "userStyle": newStyle })); this.editStyle(prevStyle); }, /** api: method[editStyle] * :arg prevStyle: ``Ext.data.Record`` * * Edit the currently selected style. */ editStyle: function(prevStyle) { var userStyle = this.selectedStyle.get("userStyle"); var buttonCfg = { bbar: ["->", { text: this.cancelText, iconCls: "cancel", handler: function() { styleProperties.propertiesDialog.userStyle = userStyle; styleProperties.destroy(); if (prevStyle) { this._cancelling = true; this.stylesStore.remove(this.selectedStyle); this.changeStyle(prevStyle, { updateCombo: true, markModified: true }); delete this._cancelling; } }, scope: this }, { text: this.saveText, iconCls: "save", handler: function() { styleProperties.destroy(); } }] }; var styleProperties = new this.dialogCls(Ext.apply(buttonCfg, { title: String.format(this.styleWindowTitle, userStyle.title || userStyle.name), shortTitle: userStyle.title || userStyle.name, bodyBorder: false, autoHeight: true, width: 300, modal: true, items: { border: false, items: { xtype: "gxp_stylepropertiesdialog", ref: "../propertiesDialog", userStyle: userStyle.clone(), nameEditable: false, style: "padding: 10px;" } }, listeners: { "beforedestroy": function() { this.selectedStyle.set( "userStyle", styleProperties.propertiesDialog.userStyle); }, scope: this } })); this.showDlg(styleProperties); }, /** api: method[createSLD] * :arg options: ``Object`` * :return: ``String`` The current SLD for the NamedLayer. * * Supported ``options``: * * * userStyles - ``Array(String)`` list of userStyles (by name) that are * to be included in the SLD. By default, all will be included. */ createSLD: function(options) { options = options || {}; var sld = { version: "1.0.0", namedLayers: {} }; var layerName = this.layerRecord.get("name"); sld.namedLayers[layerName] = { name: layerName, userStyles: [] }; this.stylesStore.each(function(r) { if(!options.userStyles || options.userStyles.indexOf(r.get("name")) !== -1) { sld.namedLayers[layerName].userStyles.push(r.get("userStyle")); } }); return new OpenLayers.Format.SLD({ multipleSymbolizers: true, profile: "GeoServer" }).write(sld); }, /** api: method[saveStyles] * :arg options: ``Object`` Options to pass to the * :class:`gxp.plugins.StyleWriter` plugin * * Saves the styles. Without a :class:`gxp.plugins.StyleWriter` plugin * configured for this instance, nothing will happen. */ saveStyles: function(options) { this.modified === true && this.fireEvent("beforesaved", this, options); }, /** private: method[updateStyleRemoveButton] * Enable/disable the "Remove" button to make sure that we don't delete * the last style. */ updateStyleRemoveButton: function() { var userStyle = this.selectedStyle && this.selectedStyle.get("userStyle"); this.items.get(1).items.get(1).setDisabled(!userStyle || this.stylesStore.getCount() <= 1 || userStyle.isDefault === true); }, /** private: method[updateRuleRemoveButton] * Enable/disable the "Remove" button to make sure that we don't delete * the last rule. */ updateRuleRemoveButton: function() { this.items.get(3).items.get(1).setDisabled( !this.selectedRule || this.items.get(2).items.get(0).rules.length < 2 ); }, /** private: method[createRule] */ createRule: function() { return new OpenLayers.Rule({ symbolizers: [new OpenLayers.Symbolizer[this.symbolType]] }); }, /** private: method[addRulesFieldSet] * :return: ``Ext.form.FieldSet`` * * Creates the rules fieldSet and adds it to this container. */ addRulesFieldSet: function() { var rulesFieldSet = new Ext.form.FieldSet({ itemId: "rulesfieldset", title: this.rulesFieldsetTitle, autoScroll: true, style: "margin-bottom: 0;", hideMode: "offsets", hidden: true }); var rulesToolbar = new Ext.Toolbar({ style: "border-width: 0 1px 1px 1px;", hidden: true, items: [ { xtype: "button", iconCls: "add", text: this.addRuleText, tooltip: this.addRuleTip, handler: this.addRule, scope: this }, { xtype: "button", iconCls: "delete", text: this.deleteRuleText, tooltip: this.deleteRuleTip, handler: this.removeRule, scope: this, disabled: true }, { xtype: "button", iconCls: "edit", text: this.editRuleText, toolitp: this.editRuleTip, handler: function() { this.layerDescription ? this.editRule() : this.describeLayer(this.editRule); }, scope: this, disabled: true }, { xtype: "button", iconCls: "duplicate", text: this.duplicateRuleText, tip: this.duplicateRuleTip, handler: this.duplicateRule, scope: this, disabled: true } ] }); this.add(rulesFieldSet, rulesToolbar); this.doLayout(); return rulesFieldSet; }, /** private: method[addRule] */ addRule: function() { var legend = this.items.get(2).items.get(0); this.selectedStyle.get("userStyle").rules.push( this.createRule() ); legend.update(); // mark the style as modified this.selectedStyle.store.afterEdit(this.selectedStyle); this.updateRuleRemoveButton(); }, /** private: method[removeRule] */ removeRule: function() { var selectedRule = this.selectedRule; this.items.get(2).items.get(0).unselect(); this.selectedStyle.get("userStyle").rules.remove(selectedRule); // mark the style as modified this.afterRuleChange(); }, /** private: method[duplicateRule] */ duplicateRule: function() { var legend = this.items.get(2).items.get(0); var newRule = this.selectedRule.clone(); this.selectedStyle.get("userStyle").rules.push( newRule ); legend.update(); // mark the style as modified this.selectedStyle.store.afterEdit(this.selectedStyle); this.updateRuleRemoveButton(); }, /** private: method[editRule] */ editRule: function() { var rule = this.selectedRule; var origRule = rule.clone(); var ruleDlg = new this.dialogCls({ title: String.format(this.ruleWindowTitle, rule.title || rule.name || this.newRuleText), shortTitle: rule.title || rule.name || this.newRuleText, layout: "fit", width: 320, height: 450, modal: true, items: [{ xtype: "gxp_rulepanel", ref: "rulePanel", symbolType: this.symbolType, rule: rule, attributes: new GeoExt.data.AttributeStore({ url: this.layerDescription.owsURL, baseParams: { "SERVICE": this.layerDescription.owsType, "REQUEST": "DescribeFeatureType", "TYPENAME": this.layerDescription.typeName }, method: "GET", disableCaching: false }), autoScroll: true, border: false, defaults: { autoHeight: true, hideMode: "offsets" }, listeners: { "change": this.saveRule, "tabchange": function() { if (ruleDlg instanceof Ext.Window) { ruleDlg.syncShadow(); } }, scope: this } }], bbar: ["->", { text: this.cancelText, iconCls: "cancel", handler: function() { this.saveRule(ruleDlg.rulePanel, origRule); ruleDlg.destroy(); }, scope: this }, { text: this.saveText, iconCls: "save", handler: function() { ruleDlg.destroy(); } }] }); this.showDlg(ruleDlg); }, /** private: method[saveRule] * :arg cmp: * :arg rule: the rule to save back to the userStyle */ saveRule: function(cmp, rule) { var style = this.selectedStyle; var legend = this.items.get(2).items.get(0); var userStyle = style.get("userStyle"); var i = userStyle.rules.indexOf(this.selectedRule); userStyle.rules[i] = rule; this.afterRuleChange(rule); }, /** private: method[afterRuleChange] * :arg rule: the rule to set as selectedRule, can be null * * Performs actions that are required to update the selectedRule and * selectedStyle after a rule was changed. */ afterRuleChange: function(rule) { var legend = this.items.get(2).items.get(0); this.selectedRule = rule; // mark the style as modified this.selectedStyle.store.afterEdit(this.selectedStyle); }, /** private: method[setRulesFieldSetVisible] * :arg visible: ``Boolean`` * * Sets the visibility of the rules fieldset */ setRulesFieldSetVisible: function(visible) { // the toolbar this.items.get(3).setVisible(visible && this.editable); // and the fieldset itself this.items.get(2).setVisible(visible); this.doLayout(); }, /** private: method[parseSLD] * :arg response: ``Object`` * :arg options: ``Object`` * * Success handler for the GetStyles response. Includes a fallback * to GetLegendGraphic if no valid SLD is returned. */ parseSLD: function(response, options) { var data = response.responseXML; if (!data || !data.documentElement) { data = new OpenLayers.Format.XML().read(response.responseText); } var layerParams = this.layerRecord.getLayer().params; var initialStyle = this.initialConfig.styleName || layerParams.STYLES; if (initialStyle) { this.selectedStyle = this.stylesStore.getAt( this.stylesStore.findExact("name", initialStyle)); } var format = new OpenLayers.Format.SLD({profile: "GeoServer", multipleSymbolizers: true}); try { var sld = format.read(data); // add userStyle objects to the stylesStore //TODO this only works if the LAYERS param contains one layer var userStyles = sld.namedLayers[layerParams.LAYERS].userStyles; // add styles from the layer's SLD_BODY *after* the userStyles var inlineStyles; if (layerParams.SLD_BODY) { var sldBody = format.read(layerParams.SLD_BODY); inlineStyles = sldBody.namedLayers[layerParams.LAYERS].userStyles; Array.prototype.push.apply(userStyles, inlineStyles); } // our stylesStore comes from the layerRecord's styles - clear it // and repopulate from GetStyles this.stylesStore.removeAll(); this.selectedStyle = null; var userStyle, record, index, defaultStyle; for (var i=0, len=userStyles.length; i=0; --i) { rec = records[i]; store.suspendEvents(); rec.get("title") || rec.set("title", rec.get("name")); store.resumeEvents(); } } } }); }, /** private: method[getStyles] * :arg callback: ``Function`` function that will be called when the * request result was returned. */ getStyles: function(callback) { var layer = this.layerRecord.getLayer(); if(this.editable === true) { var version = layer.params["VERSION"]; if (parseFloat(version) > 1.1) { //TODO don't force 1.1.1, fall back instead version = "1.1.1"; } Ext.Ajax.request({ url: layer.url, params: { "SERVICE": "WMS", "VERSION": version, "REQUEST": "GetStyles", "LAYERS": [layer.params["LAYERS"]].join(",") }, method: "GET", disableCaching: false, success: this.parseSLD, failure: this.setupNonEditable, callback: callback, scope: this }); } else { this.setupNonEditable(); } }, /** private: method[describeLayer] * :arg callback: ``Function`` function that will be called when the * request result was returned. */ describeLayer: function(callback) { if (this.layerDescription) { // always return before calling callback window.setTimeout(function() { callback.call(this); }, 0); } else { var layer = this.layerRecord.getLayer(); var version = layer.params["VERSION"]; if (parseFloat(version) > 1.1) { //TODO don't force 1.1.1, fall back instead version = "1.1.1"; } Ext.Ajax.request({ url: layer.url, params: { "SERVICE": "WMS", "VERSION": version, "REQUEST": "DescribeLayer", "LAYERS": [layer.params["LAYERS"]].join(",") }, method: "GET", disableCaching: false, success: function(response) { var result = new OpenLayers.Format.WMSDescribeLayer().read( response.responseXML && response.responseXML.documentElement ? response.responseXML : response.responseText); this.layerDescription = result[0]; }, callback: callback, scope: this }); } }, /** private: method[addStylesCombo] * * Adds a combo box with the available style names found for the layer * in the capabilities document to this component's stylesFieldset. */ addStylesCombo: function() { var store = this.stylesStore; var combo = new Ext.form.ComboBox(Ext.apply({ fieldLabel: this.chooseStyleText, store: store, editable: false, displayField: "title", valueField: "name", value: this.selectedStyle ? this.selectedStyle.get("title") : this.layerRecord.getLayer().params.STYLES || "default", disabled: !store.getCount(), mode: "local", typeAhead: true, triggerAction: "all", forceSelection: true, anchor: "100%", listeners: { "select": function(combo, record) { this.changeStyle(record); if (!record.phantom && !this._removing) { this.fireEvent("styleselected", this, record.get("name")); } }, scope: this } }, this.initialConfig.stylesComboOptions)); // add combo to the styles fieldset this.items.get(0).add(combo); this.doLayout(); }, /** private: method[createLegendImage] * :return: ``GeoExt.LegendImage`` or undefined if none available. * * Creates a legend image for the first style of the current layer. This * is used when GetStyles is not available from the layer's WMS. */ createLegendImage: function() { var legend = new GeoExt.WMSLegend({ showTitle: false, layerRecord: this.layerRecord, autoScroll: true, defaults: { listeners: { "render": function(cmp) { cmp.getEl().on({ load: function(evt, img) { if (img.getAttribute("src") != cmp.defaultImgSrc) { this.setRulesFieldSetVisible(true); if (cmp.getEl().getHeight() > 250) { legend.setHeight(250); } } }, "error": function() { this.setRulesFieldSetVisible(false); }, scope: this }); }, scope: this } } }); return legend; }, /** api: method[changeStyle] * :arg value: ``Ext.data.Record`` * :arg options: ``Object`` Additional options for this method. * * Available options: * * updateCombo - ``Boolean`` set to true to update the combo box * * markModified - ``Boolean`` set to true to mark the dialog modified * * Handler for the stylesCombo's ``select`` and the store's ``update`` * event. Updates the layer and the rules fieldset. */ changeStyle: function(record, options) { options = options || {}; var legend = this.items.get(2).items.get(0); this.selectedStyle = record; this.updateStyleRemoveButton(); var styleName = record.get("name"); if (this.editable === true) { var userStyle = record.get("userStyle"); if (userStyle.isDefault === true) { styleName = ""; } var ruleIdx = legend.rules.indexOf(this.selectedRule); // replace the legend legend.ownerCt.remove(legend); this.createLegend(userStyle.rules, {selectedRuleIndex: ruleIdx}); } if (options.updateCombo === true) { // update the combo's value with the new name this.items.get(0).items.get(0).setValue(userStyle.name); options.markModified === true && this.markModified(); } }, /** private: method[addVectorLegend] * :arg rules: ``Array`` * :arg options: ``Object`` * :return: ``GeoExt.VectorLegend`` the legend that was created * * Creates the vector legend for the provided rules and adds it to the * rules fieldset. */ addVectorLegend: function(rules, options) { options = Ext.applyIf(options || {}, {enableDD: true}); this.symbolType = options.symbolType; if (!this.symbolType) { var typeHierarchy = ["Point", "Line", "Polygon"]; // use the highest symbolizer type of the 1st rule highest = 0; var symbolizers = rules[0].symbolizers, symbolType; for (var i=symbolizers.length-1; i>=0; i--) { symbolType = symbolizers[i].CLASS_NAME.split(".").pop(); highest = Math.max(highest, typeHierarchy.indexOf(symbolType)); } this.symbolType = typeHierarchy[highest]; } var legend = this.items.get(2).add({ xtype: "gx_vectorlegend", showTitle: false, height: rules.length > 10 ? 250 : undefined, autoScroll: rules.length > 10, rules: rules, symbolType: this.symbolType, selectOnClick: true, enableDD: options.enableDD, listeners: { "ruleselected": function(cmp, rule) { this.selectedRule = rule; // enable the Remove, Edit and Duplicate buttons var tbItems = this.items.get(3).items; this.updateRuleRemoveButton(); tbItems.get(2).enable(); tbItems.get(3).enable(); }, "ruleunselected": function(cmp, rule) { this.selectedRule = null; // disable the Remove, Edit and Duplicate buttons var tbItems = this.items.get(3).items; tbItems.get(1).disable(); tbItems.get(2).disable(); tbItems.get(3).disable(); }, "rulemoved": function() { this.markModified(); }, "afterlayout": function() { // restore selection //TODO QA: avoid accessing private properties/methods if (this.selectedRule !== null && legend.selectedRule === null && legend.rules.indexOf(this.selectedRule) !== -1) { legend.selectRuleEntry(this.selectedRule); } }, scope: this } }); this.setRulesFieldSetVisible(true); return legend; }, newStyleName: function() { var layerName = this.layerRecord.get("name"); return layerName.split(":").pop() + "_" + gxp.util.md5(layerName + new Date() + Math.random()).substr(0, 8); }, /** private: method[showDlg] * :arg dlg: * * Shows a subdialog */ showDlg: function(dlg) { dlg.show(); } }); /** api: function[createGeoServerStylerConfig] * :arg layerRecord: ``GeoExt.data.LayerRecord`` Layer record to configure the * dialog for. * :arg url: ``String`` Optional. Custaom URL for the GeoServer REST endpoint * for writing styles. * * Creates a configuration object for a :class:`gxp.WMSStylesDialog` with a * :class:`gxp.plugins.GeoServerStyleWriter` plugin and listeners for the * "styleselected", "modified" and "saved" events that take care of saving * styles and keeping the layer view updated. */ gxp.WMSStylesDialog.createGeoServerStylerConfig = function(layerRecord, url) { var layer = layerRecord.getLayer(); if (!url) { url = layerRecord.get("restUrl"); } if (!url) { url = layer.url.split("?").shift().replace(/\/(wms|ows)\/?$/, "/rest"); } return { xtype: "gxp_wmsstylesdialog", layerRecord: layerRecord, plugins: [{ ptype: "gxp_geoserverstylewriter", baseUrl: url }], listeners: { "styleselected": function(cmp, style) { layer.mergeNewParams({ styles: style }); }, "modified": function(cmp, style) { cmp.saveStyles(); }, "saved": function(cmp, style) { layer.mergeNewParams({ _olSalt: Math.random(), styles: style }); }, scope: this } }; }; // set SLD defaults for symbolizer OpenLayers.Renderer.defaultSymbolizer = { fillColor: "#808080", fillOpacity: 1, strokeColor: "#000000", strokeOpacity: 1, strokeWidth: 1, strokeDashstyle: "solid", pointRadius: 3, graphicName: "square", fontColor: "#000000", fontSize: 10, haloColor: "#FFFFFF", haloOpacity: 1, haloRadius: 1, labelAlign: 'cm' }; /** api: xtype = gxp_wmsstylesdialog */ Ext.reg('gxp_wmsstylesdialog', gxp.WMSStylesDialog); /** FILE: widgets/ScaleLimitPanel.js **/ /** * Copyright (c) 2008-2011 The Open Planning Project * * Published under the GPL license. * See https://github.com/opengeo/gxp/raw/master/license.txt for the full text * of the license. */ /** * @include widgets/tips/SliderTip.js */ /** api: (define) * module = gxp * class = ScaleLimitPanel * base_link = `Ext.Panel `_ */ Ext.namespace("gxp"); /** api: constructor * .. class:: ScaleLimitPanel(config) * * A panel for assembling scale constraints in SLD styles. */ gxp.ScaleLimitPanel = Ext.extend(Ext.Panel, { /** api: config[maxScaleDenominatorLimit] * ``Number`` Upper limit for scale denominators. Default is what you get * when you project the world in Spherical Mercator onto a single * 256 x 256 pixel tile and assume OpenLayers.DOTS_PER_INCH (this * corresponds to zoom level 0 in Google Maps). */ maxScaleDenominatorLimit: 40075016.68 * 39.3701 * OpenLayers.DOTS_PER_INCH / 256, /** api: config[limitMaxScaleDenominator] * ``Boolean`` Limit the maximum scale denominator. If false, no upper * limit will be imposed. */ limitMaxScaleDenominator: true, /** api: config[maxScaleDenominator] * ``Number`` The initial maximum scale denominator. If is * true and no minScaleDenominator is provided, will * be used. */ maxScaleDenominator: undefined, /** api: config[minScaleDenominatorLimit] * ``Number`` Lower limit for scale denominators. Default is what you get when * you assume 20 zoom levels starting with the world in Spherical * Mercator on a single 256 x 256 tile at zoom 0 where the zoom factor * is 2. */ minScaleDenominatorLimit: Math.pow(0.5, 19) * 40075016.68 * 39.3701 * OpenLayers.DOTS_PER_INCH / 256, /** api: config[limitMinScaleDenominator] * ``Boolean`` Limit the minimum scale denominator. If false, no lower * limit will be imposed. */ limitMinScaleDenominator: true, /** api: config[minScaleDenominator] * ``Number`` The initial minimum scale denominator. If is * true and no minScaleDenominator is provided, will * be used. */ minScaleDenominator: undefined, /** api: config[scaleLevels] * ``Number`` Number of scale levels to assume. This is only for scaling * values exponentially along the slider. Scale values are not * required to one of the discrete levels. Default is 20. */ scaleLevels: 20, /** api: config[scaleSliderTemplate] * ``String`` Template for the tip displayed by the scale threshold slider. * * Can be customized using the following keywords in curly braces: * * * zoom - the zoom level * * scale - the scale denominator * * type - "Max" or "Min" denominator * * scaleType - "Min" or "Max" scale (sense is opposite of type) * * Default is "{scaleType} Scale 1:{scale}". */ scaleSliderTemplate: "{scaleType} Scale 1:{scale}", /** api: config[modifyScaleTipContext] * ``Function`` Called from the multi-slider tip's getText function. The * function will receive two arguments - a reference to the panel and * a data object. The data object will have scale, zoom, and type * properties already calculated. Other properties added to the data * object are available to the . */ modifyScaleTipContext: Ext.emptyFn, /** private: property[scaleFactor] * ``Number`` Calculated base for determining exponential scaling of values * for the slider. */ scaleFactor: null, /** private: property[changing] * ``Boolean`` The panel is updating itself. */ changing: false, border: false, /** i18n */ maxScaleLimitText: "Max scale limit", minScaleLimitText: "Min scale limit", /** private: method[initComponent] */ initComponent: function() { this.layout = "column"; this.defaults = { border: false, bodyStyle: "margin: 0 5px;" }; this.bodyStyle = { padding: "5px" }; this.scaleSliderTemplate = new Ext.Template(this.scaleSliderTemplate); Ext.applyIf(this, { minScaleDenominator: this.minScaleDenominatorLimit, maxScaleDenominator: this.maxScaleDenominatorLimit }); this.scaleFactor = Math.pow( this.maxScaleDenominatorLimit / this.minScaleDenominatorLimit, 1 / (this.scaleLevels - 1) ); this.scaleSlider = new Ext.Slider({ vertical: true, height: 100, values: [0, 100], listeners: { changecomplete: function(slider, value) { this.updateScaleValues(slider); }, render: function(slider) { slider.thumbs[0].el.setVisible(this.limitMaxScaleDenominator); slider.thumbs[1].el.setVisible(this.limitMinScaleDenominator); slider.setDisabled(!this.limitMinScaleDenominator && !this.limitMaxScaleDenominator); }, scope: this }, plugins: [new gxp.slider.Tip({ getText: (function(thumb) { var index = thumb.slider.thumbs.indexOf(thumb); var value = thumb.value; var scales = this.sliderValuesToScale([thumb.value]); var data = { scale: String(scales[0]), zoom: (thumb.value * (this.scaleLevels / 100)).toFixed(1), type: (index === 0) ? "Max" : "Min", scaleType: (index === 0) ? "Min" : "Max" }; this.modifyScaleTipContext(this, data); return this.scaleSliderTemplate.apply(data); }).createDelegate(this) })] }); this.maxScaleDenominatorInput = new Ext.form.NumberField({ allowNegative: false, width: 100, fieldLabel: "1", value: Math.round(this.maxScaleDenominator), disabled: !this.limitMaxScaleDenominator, validator: (function(value) { return !this.limitMinScaleDenominator || (value > this.minScaleDenominator); }).createDelegate(this), listeners: { valid: function(field) { var value = Number(field.getValue()); var limit = Math.round(this.maxScaleDenominatorLimit); if(value < limit && value > this.minScaleDenominator) { this.maxScaleDenominator = value; this.updateSliderValues(); } }, change: function(field) { var value = Number(field.getValue()); var limit = Math.round(this.maxScaleDenominatorLimit); if(value > limit) { field.setValue(limit); } else if(value < this.minScaleDenominator) { field.setValue(this.minScaleDenominator); } else { this.maxScaleDenominator = value; this.updateSliderValues(); } }, scope: this } }); this.minScaleDenominatorInput = new Ext.form.NumberField({ allowNegative: false, width: 100, fieldLabel: "1", value: Math.round(this.minScaleDenominator), disabled: !this.limitMinScaleDenominator, validator: (function(value) { return !this.limitMaxScaleDenominator || (value < this.maxScaleDenominator); }).createDelegate(this), listeners: { valid: function(field) { var value = Number(field.getValue()); var limit = Math.round(this.minScaleDenominatorLimit); if(value > limit && value < this.maxScaleDenominator) { this.minScaleDenominator = value; this.updateSliderValues(); } }, change: function(field) { var value = Number(field.getValue()); var limit = Math.round(this.minScaleDenominatorLimit); if(value < limit) { field.setValue(limit); } else if(value > this.maxScaleDenominator) { field.setValue(this.maxScaleDenominator); } else { this.minScaleDenominator = value; this.updateSliderValues(); } }, scope: this } }); this.items = [this.scaleSlider, { xtype: "panel", layout: "form", defaults: {border: false}, items: [{ labelWidth: 90, layout: "form", width: 150, items: [{ xtype: "checkbox", checked: !!this.limitMinScaleDenominator, fieldLabel: this.maxScaleLimitText, listeners: { check: function(box, checked) { this.limitMinScaleDenominator = checked; var slider = this.scaleSlider; slider.setValue(1, 100); slider.thumbs[1].el.setVisible(checked); this.minScaleDenominatorInput.setDisabled(!checked); this.updateScaleValues(slider); slider.setDisabled(!this.limitMinScaleDenominator && !this.limitMaxScaleDenominator); }, scope: this } }] }, { labelWidth: 10, layout: "form", items: [this.minScaleDenominatorInput] }, { labelWidth: 90, layout: "form", items: [{ xtype: "checkbox", checked: !!this.limitMaxScaleDenominator, fieldLabel: this.minScaleLimitText, listeners: { check: function(box, checked) { this.limitMaxScaleDenominator = checked; var slider = this.scaleSlider; slider.setValue(0, 0); slider.thumbs[0].el.setVisible(checked); this.maxScaleDenominatorInput.setDisabled(!checked); this.updateScaleValues(slider); slider.setDisabled(!this.limitMinScaleDenominator && !this.limitMaxScaleDenominator); }, scope: this } }] }, { labelWidth: 10, layout: "form", items: [this.maxScaleDenominatorInput] }] }]; this.addEvents( /** * Event: change * Fires before any field blurs if the field value has changed. * * Listener arguments: * symbolizer - {Object} A symbolizer with fill related properties * updated. */ "change" ); gxp.ScaleLimitPanel.superclass.initComponent.call(this); }, /** private: method[updateScaleValues] */ updateScaleValues: function(slider) { if(!this.changing) { var values = slider.getValues(); var resetSlider = false; if(!this.limitMaxScaleDenominator) { if(values[0] > 0) { values[0] = 0; resetSlider = true; } } if(!this.limitMinScaleDenominator) { if(values[1] < 100) { values[1] = 100; resetSlider = true; } } if(resetSlider) { slider.setValue(0, values[0]); slider.setValue(1, values[1]); } else { var scales = this.sliderValuesToScale(values); var max = scales[0]; var min = scales[1]; this.changing = true; this.minScaleDenominatorInput.setValue(min); this.maxScaleDenominatorInput.setValue(max); this.changing = false; this.fireEvent( "change", this, (this.limitMinScaleDenominator) ? min : undefined, (this.limitMaxScaleDenominator) ? max : undefined ); } } }, /** private: method[updateSliderValues] */ updateSliderValues: function() { if(!this.changing) { var min = this.minScaleDenominator; var max = this.maxScaleDenominator; var values = this.scaleToSliderValues([max, min]); this.changing = true; this.scaleSlider.setValue(0, values[0]); this.scaleSlider.setValue(1, values[1]); this.changing = false; this.fireEvent( "change", this, (this.limitMinScaleDenominator) ? min : undefined, (this.limitMaxScaleDenominator) ? max : undefined ); } }, /** private: method[sliderValuesToScale] * :arg values: ``Array`` Values from the scale slider. * :return: ``Array`` A two item array of min and max scale denominators. * * Given two values between 0 and 100, generate the min and max scale * denominators. Assuming exponential scaling with . */ sliderValuesToScale: function(values) { var interval = 100 / (this.scaleLevels - 1); return [Math.round(Math.pow(this.scaleFactor, (100 - values[0]) / interval) * this.minScaleDenominatorLimit), Math.round(Math.pow(this.scaleFactor, (100 - values[1]) / interval) * this.minScaleDenominatorLimit)]; }, /** private: method[scaleToSliderValues] */ scaleToSliderValues: function(scales) { var interval = 100 / (this.scaleLevels - 1); return [100 - (interval * Math.log(scales[0] / this.minScaleDenominatorLimit) / Math.log(this.scaleFactor)), 100 - (interval * Math.log(scales[1] / this.minScaleDenominatorLimit) / Math.log(this.scaleFactor))]; } }); /** api: xtype = gxp_scalelimitpanel */ Ext.reg('gxp_scalelimitpanel', gxp.ScaleLimitPanel); /** FILE: widgets/tips/SliderTip.js **/ /** * Copyright (c) 2008-2011 The Open Planning Project * * Published under the GPL license. * See https://github.com/opengeo/gxp/raw/master/license.txt for the full text * of the license. */ /** api: (define) * module = gxp.slider * class = Tip * base_link = `Ext.slider.Tip `_ */ Ext.namespace("gxp.slider"); /** api: constructor * .. class:: gxp.slider.Tip(config) * * See http://trac.geoext.org/ticket/262 * * This tip matches the Ext.slider.Tip but addes the hover functionality. */ gxp.slider.Tip = Ext.extend(Ext.slider.Tip, { /** api: config[hover] * ``Boolean`` Display the tip when hovering over a thumb. If false, tip * will only be displayed while dragging. Default is true. */ hover: true, /** private: property[dragging] * ``Boolean`` A thumb is currently being dragged. */ dragging: false, /** private: method[init] * :arg slider: ``Object`` */ init: function(slider) { if(this.hover) { slider.on("render", this.registerThumbListeners, this); } this.slider = slider; gxp.slider.Tip.superclass.init.apply(this, arguments); }, /** private: method[registerThumbListeners] */ registerThumbListeners: function() { for(var i=0, len=this.slider.thumbs.length; i`_ */ Ext.namespace("gxp"); /** api: constructor * .. class:: TextSymbolizer(config) * * Form for configuring a text symbolizer. */ gxp.TextSymbolizer = Ext.extend(Ext.Panel, { /** api: config[fonts] * ``Array(String)`` * List of fonts for the font combo. If not set, defaults to the list * provided by the :class:`gxp.FontComboBox`. */ fonts: undefined, /** api: config[symbolizer] * ``Object`` * A symbolizer object that will be used to fill in form values. * This object will be modified when values change. Clone first if * you do not want your symbolizer modified. */ symbolizer: null, /** api: config[defaultSymbolizer] * ``Object`` * Default symbolizer properties to be used where none provided. */ defaultSymbolizer: null, /** api: config[attributes] * :class:`GeoExt.data.AttributeStore` * A configured attributes store for use in the filter property combo. */ attributes: null, /** api: config[colorManager] * ``Function`` * Optional color manager constructor to be used as a plugin for the color * field. */ colorManager: null, /** private: property[haloCache] * ``Object`` * Stores halo properties while fieldset is collapsed. */ haloCache: null, border: false, layout: "form", /** i18n */ labelValuesText: "Label values", haloText: "Halo", sizeText: "Size", priorityText: "Priority", labelOptionsText: "Label options", autoWrapText: "Auto wrap", followLineText: "Follow line", maxDisplacementText: "Maximum displacement", repeatText: "Repeat", forceLeftToRightText: "Force left to right", groupText: "Grouping", spaceAroundText: "Space around", labelAllGroupText: "Label all segments in line group", maxAngleDeltaText: "Maximum angle delta", conflictResolutionText: "Conflict resolution", goodnessOfFitText: "Goodness of fit", polygonAlignText: "Polygon alignment", graphicResizeText: "Graphic resize", graphicMarginText: "Graphic margin", graphicTitle: "Graphic", fontColorTitle: "Font color and opacity", positioningText: "Label positioning", anchorPointText: "Anchor point", displacementXText: "Displacement (X-direction)", displacementYText: "Displacement (Y-direction)", perpendicularOffsetText: "Perpendicular offset", priorityHelp: "The higher the value of the specified field, the sooner the label will be drawn (which makes it win in the conflict resolution game)", autoWrapHelp: "Wrap labels that exceed a certain length in pixels", followLineHelp: "Should the label follow the geometry of the line?", maxDisplacementHelp: "Maximum displacement in pixels if label position is busy", repeatHelp: "Repeat labels after a certain number of pixels", forceLeftToRightHelp: "Labels are usually flipped to make them readable. If the character happens to be a directional arrow then this is not desirable", groupHelp: "Grouping works by collecting all features with the same label text, then choosing a representative geometry for the group. Road data is a classic example to show why grouping is useful. It is usually desirable to display only a single label for all of 'Main Street', not a label for every block of 'Main Street.'", spaceAroundHelp: "Overlapping and Separating Labels. By default GeoServer will not render labels 'on top of each other'. By using the spaceAround option you can either allow labels to overlap, or add extra space around labels. The value supplied for the option is a positive or negative size in pixels. Using the default value of 0, the bounding box of a label cannot overlap the bounding box of another label.", labelAllGroupHelp: "The labelAllGroup option makes sure that all of the segments in a line group are labeled instead of just the longest one.", conflictResolutionHelp: "By default labels are subjected to conflict resolution, meaning the renderer will not allow any label to overlap with a label that has been drawn already. Setting this parameter to false pull the label out of the conflict resolution game, meaning the label will be drawn even if it overlaps with other labels, and other labels drawn after it won’t mind overlapping with it.", goodnessOfFitHelp: "Geoserver will remove labels if they are a particularly bad fit for the geometry they are labeling. For Polygons: the label is sampled approximately at every letter. The distance from these points to the polygon is determined and each sample votes based on how close it is to the polygon. The default value is 0.5.", graphic_resizeHelp: "Specifies a mode for resizing label graphics (such as highway shields) to fit the text of the label. The default mode, ‘none’, never modifies the label graphic. In stretch mode, GeoServer will resize the graphic to exactly surround the label text, possibly modifying the image’s aspect ratio. In proportional mode, GeoServer will expand the image to be large enough to surround the text while preserving its original aspect ratio.", maxAngleDeltaHelp: "Designed to use used in conjuection with followLine, the maxAngleDelta option sets the maximum angle, in degrees, between two subsequent characters in a curved label. Large angles create either visually disconnected words or overlapping characters. It is advised not to use angles larger than 30.", polygonAlignHelp: "GeoServer normally tries to place horizontal labels within a polygon, and give up in case the label position is busy or if the label does not fit enough in the polygon. This options allows GeoServer to try alternate rotations for the labels. Possible options: the default value, only the rotation manually specified in the tag will be used (manual), If the label does not fit horizontally and the polygon is taller than wider the vertical alignement will also be tried (ortho), If the label does not fit horizontally the minimum bounding rectangle will be computed and a label aligned to it will be tried out as well (mbr).", graphic_marginHelp: "Similar to the margin shorthand property in CSS for HTML, its interpretation varies depending on how many margin values are provided: 1 = use that margin length on all sides of the label 2 = use the first for top & bottom margins and the second for left & right margins. 3 = use the first for the top margin, second for left & right margins, third for the bottom margin. 4 = use the first for the top margin, second for the right margin, third for the bottom margin, and fourth for the left margin.", initComponent: function() { if(!this.symbolizer) { this.symbolizer = {}; } Ext.applyIf(this.symbolizer, this.defaultSymbolizer); if (!this.symbolizer.vendorOptions) { this.symbolizer.vendorOptions = {}; } this.haloCache = {}; this.attributes.on('load', this.showHideGeometryOptions, this); this.attributes.load(); var defAttributesComboConfig = { xtype: "combo", fieldLabel: this.labelValuesText, store: this.attributes, mode: 'local', lastQuery: '', editable: false, triggerAction: "all", allowBlank: false, displayField: "name", valueField: "name", value: this.symbolizer.label && this.symbolizer.label.replace(/^\${(.*)}$/, "$1"), listeners: { select: function(combo, record) { this.symbolizer.label = "${" + record.get("name") + "}"; this.fireEvent("change", this.symbolizer); }, scope: this }, width: 120 }; this.attributesComboConfig = this.attributesComboConfig || {}; Ext.applyIf(this.attributesComboConfig, defAttributesComboConfig); this.labelWidth = 80; this.items = [this.attributesComboConfig, { cls: "x-html-editor-tb", style: "background: transparent; border: none; padding: 0 0em 0.5em;", xtype: "toolbar", items: [{ xtype: "gxp_fontcombo", fonts: this.fonts || undefined, width: 110, value: this.symbolizer.fontFamily, listeners: { select: function(combo, record) { this.symbolizer.fontFamily = record.get("field1"); this.fireEvent("change", this.symbolizer); }, scope: this } }, { xtype: "tbtext", text: this.sizeText + ": " }, { xtype: "numberfield", allowNegative: false, emptyText: OpenLayers.Renderer.defaultSymbolizer.fontSize, value: this.symbolizer.fontSize, width: 30, listeners: { change: function(field, value) { value = parseFloat(value); if (isNaN(value)) { delete this.symbolizer.fontSize; } else { this.symbolizer.fontSize = value; } this.fireEvent("change", this.symbolizer); }, scope: this } }, { enableToggle: true, cls: "x-btn-icon", iconCls: "x-edit-bold", pressed: this.symbolizer.fontWeight === "bold", listeners: { toggle: function(button, pressed) { this.symbolizer.fontWeight = pressed ? "bold" : "normal"; this.fireEvent("change", this.symbolizer); }, scope: this } }, { enableToggle: true, cls: "x-btn-icon", iconCls: "x-edit-italic", pressed: this.symbolizer.fontStyle === "italic", listeners: { toggle: function(button, pressed) { this.symbolizer.fontStyle = pressed ? "italic" : "normal"; this.fireEvent("change", this.symbolizer); }, scope: this } }] }, { xtype: "gxp_fillsymbolizer", fillText: this.fontColorTitle, symbolizer: this.symbolizer, colorProperty: "fontColor", opacityProperty: "fontOpacity", checkboxToggle: false, autoHeight: true, width: 213, labelWidth: 70, plugins: this.colorManager && [new this.colorManager()], listeners: { change: function(symbolizer) { this.fireEvent("change", this.symbolizer); }, scope: this } }, { xtype: "fieldset", title: this.graphicTitle, checkboxToggle: true, hideMode: 'offsets', collapsed: !(this.symbolizer.fillColor || this.symbolizer.fillOpacity || this.symbolizer.vendorOptions["graphic-resize"] || this.symbolizer.vendorOptions["graphic-margin"]), labelWidth: 70, items: [{ xtype: "gxp_pointsymbolizer", symbolizer: this.symbolizer, listeners: { "change": function(symbolizer) { symbolizer.graphic = !!symbolizer.graphicName || !!symbolizer.externalGraphic; this.fireEvent("change", this.symbolizer); }, scope: this }, border: false, labelWidth: 70 }, this.createVendorSpecificField({ name: "graphic-resize", xtype: "combo", store: ["none", "stretch", "proportional"], mode: 'local', listeners: { "select": function(combo, record) { if (combo.getValue() === "none") { this.graphicMargin.hide(); } else { if (Ext.isEmpty(this.graphicMargin.getValue())) { this.graphicMargin.setValue(0); this.symbolizer.vendorOptions["graphic-margin"] = 0; } this.graphicMargin.show(); } }, scope: this }, width: 100, triggerAction: 'all', fieldLabel: this.graphicResizeText }), this.createVendorSpecificField({ name: "graphic-margin", ref: "../graphicMargin", hidden: (this.symbolizer.vendorOptions["graphic-resize"] !== "stretch" && this.symbolizer.vendorOptions["graphic-resize"] !== "proportional"), width: 100, fieldLabel: this.graphicMarginText, xtype: "textfield" })], listeners: { collapse: function() { this.graphicCache = { externalGraphic: this.symbolizer.externalGraphic, fillColor: this.symbolizer.fillColor, fillOpacity: this.symbolizer.fillOpacity, graphicName: this.symbolizer.graphicName, pointRadius: this.symbolizer.pointRadius, rotation: this.symbolizer.rotation, strokeColor: this.symbolizer.strokeColor, strokeWidth: this.symbolizer.strokeWidth, strokeDashStyle: this.symbolizer.strokeDashStyle }; delete this.symbolizer.externalGraphic; delete this.symbolizer.fillColor; delete this.symbolizer.fillOpacity; delete this.symbolizer.graphicName; delete this.symbolizer.pointRadius; delete this.symbolizer.rotation; delete this.symbolizer.strokeColor; delete this.symbolizer.strokeWidth; delete this.symbolizer.strokeDashStyle; this.fireEvent("change", this.symbolizer) }, expand: function() { Ext.apply(this.symbolizer, this.graphicCache); /** * Start workaround for * http://projects.opengeo.org/suite/ticket/676 */ this.doLayout(); /** * End workaround for * http://projects.opengeo.org/suite/ticket/676 */ this.fireEvent("change", this.symbolizer); }, scope: this } }, { xtype: "fieldset", title: this.haloText, checkboxToggle: true, collapsed: !(this.symbolizer.haloRadius || this.symbolizer.haloColor || this.symbolizer.haloOpacity), autoHeight: true, labelWidth: 50, items: [{ xtype: "numberfield", fieldLabel: this.sizeText, anchor: "89%", allowNegative: false, emptyText: OpenLayers.Renderer.defaultSymbolizer.haloRadius, value: this.symbolizer.haloRadius, listeners: { change: function(field, value) { value = parseFloat(value); if (isNaN(value)) { delete this.symbolizer.haloRadius; } else { this.symbolizer.haloRadius = value; } this.fireEvent("change", this.symbolizer); }, scope: this } }, { xtype: "gxp_fillsymbolizer", symbolizer: { fillColor: ("haloColor" in this.symbolizer) ? this.symbolizer.haloColor : OpenLayers.Renderer.defaultSymbolizer.haloColor, fillOpacity: ("haloOpacity" in this.symbolizer) ? this.symbolizer.haloOpacity : OpenLayers.Renderer.defaultSymbolizer.haloOpacity*100 }, defaultColor: OpenLayers.Renderer.defaultSymbolizer.haloColor, checkboxToggle: false, width: 190, labelWidth: 60, plugins: this.colorManager && [new this.colorManager()], listeners: { change: function(symbolizer) { this.symbolizer.haloColor = symbolizer.fillColor; this.symbolizer.haloOpacity = symbolizer.fillOpacity; this.fireEvent("change", this.symbolizer); }, scope: this } }], listeners: { collapse: function() { this.haloCache = { haloRadius: this.symbolizer.haloRadius, haloColor: this.symbolizer.haloColor, haloOpacity: this.symbolizer.haloOpacity }; delete this.symbolizer.haloRadius; delete this.symbolizer.haloColor; delete this.symbolizer.haloOpacity; this.fireEvent("change", this.symbolizer) }, expand: function() { Ext.apply(this.symbolizer, this.haloCache); /** * Start workaround for * http://projects.opengeo.org/suite/ticket/676 */ this.doLayout(); /** * End workaround for * http://projects.opengeo.org/suite/ticket/676 */ this.fireEvent("change", this.symbolizer); }, scope: this } }, { xtype: "fieldset", collapsed: !(this.symbolizer.labelAlign || this.symbolizer.vendorOptions['polygonAlign'] || this.symbolizer.labelXOffset || this.symbolizer.labelYOffset || this.symbolizer.labelPerpendicularOffset), title: this.positioningText, checkboxToggle: true, autoHeight: true, labelWidth: 75, defaults: { width: 100 }, items: [this.createField(Ext.applyIf({ fieldLabel: this.anchorPointText, geometryTypes: ["POINT"], value: this.symbolizer.labelAlign || "lb", store: [ ['lt', 'Left-top'], ['ct', 'Center-top'], ['rt', 'Right-top'], ['lm', 'Left-center'], ['cm', 'Center'], ['rm', 'Right-center'], ['lb', 'Left-bottom'], ['cb', 'Center-bottom'], ['rb', 'Right-bottom'] ], listeners: { select: function(combo, record) { this.symbolizer.labelAlign = combo.getValue(); delete this.symbolizer.labelAnchorPointX; delete this.symbolizer.labelAnchorPointY; this.fireEvent("change", this.symbolizer); }, scope: this } }, this.attributesComboConfig)), this.createField({ xtype: "numberfield", geometryTypes: ["POINT"], fieldLabel: this.displacementXText, value: this.symbolizer.labelXOffset, listeners: { change: function(field, value) { this.symbolizer.labelXOffset = value; this.fireEvent("change", this.symbolizer); }, scope: this } }), this.createField({ xtype: "numberfield", geometryTypes: ["POINT"], fieldLabel: this.displacementYText, value: this.symbolizer.labelYOffset, listeners: { change: function(field, value) { this.symbolizer.labelYOffset = value; this.fireEvent("change", this.symbolizer); }, scope: this } }), this.createField({ xtype: "numberfield", geometryTypes: ["LINE"], fieldLabel: this.perpendicularOffsetText, value: this.symbolizer.labelPerpendicularOffset, listeners: { change: function(field, value) { if (Ext.isEmpty(value)) { delete this.symbolizer.labelPerpendicularOffset; } else { this.symbolizer.labelPerpendicularOffset = value; } this.fireEvent("change", this.symbolizer); }, scope: this } }), this.createVendorSpecificField({ name: 'polygonAlign', geometryTypes: ['POLYGON'], xtype: "combo", mode: 'local', value: this.symbolizer.vendorOptions['polygonAlign'] || 'manual', triggerAction: 'all', store: ["manual", "ortho", "mbr"], fieldLabel: this.polygonAlignText })] }, { xtype: "fieldset", title: this.priorityText, checkboxToggle: true, collapsed: !(this.symbolizer.priority), autoHeight: true, labelWidth: 50, items: [Ext.applyIf({ fieldLabel: this.priorityText, value: this.symbolizer.priority && this.symbolizer.priority.replace(/^\${(.*)}$/, "$1"), allowBlank: true, name: 'priority', plugins: [{ ptype: 'gxp_formfieldhelp', dismissDelay: 20000, helpText: this.priorityHelp }], listeners: { select: function(combo, record) { this.symbolizer[combo.name] = "${" + record.get("name") + "}"; this.fireEvent("change", this.symbolizer); }, scope: this } }, this.attributesComboConfig)] }, { xtype: "fieldset", title: this.labelOptionsText, checkboxToggle: true, collapsed: !(this.symbolizer.vendorOptions['autoWrap'] || this.symbolizer.vendorOptions['followLine'] || this.symbolizer.vendorOptions['maxAngleDelta'] || this.symbolizer.vendorOptions['maxDisplacement'] || this.symbolizer.vendorOptions['repeat'] || this.symbolizer.vendorOptions['forceLeftToRight'] || this.symbolizer.vendorOptions['group'] || this.symbolizer.vendorOptions['spaceAround'] || this.symbolizer.vendorOptions['labelAllGroup'] || this.symbolizer.vendorOptions['conflictResolution'] || this.symbolizer.vendorOptions['goodnessOfFit'] || this.symbolizer.vendorOptions['polygonAlign']), autoHeight: true, labelWidth: 80, defaults: { width: 100 }, items: [ this.createVendorSpecificField({ name: 'autoWrap', allowBlank: false, fieldLabel: this.autoWrapText }), this.createVendorSpecificField({ name: 'followLine', geometryTypes: ["LINE"], xtype: 'checkbox', listeners: { 'check': function(cb, checked) { if (!checked) { this.maxAngleDelta.hide(); } else { this.maxAngleDelta.show(); } }, scope: this }, fieldLabel: this.followLineText }), this.createVendorSpecificField({ name: 'maxAngleDelta', ref: "../maxAngleDelta", hidden: (this.symbolizer.vendorOptions["followLine"] == null), geometryTypes: ["LINE"], fieldLabel: this.maxAngleDeltaText }), this.createVendorSpecificField({ name: 'maxDisplacement', fieldLabel: this.maxDisplacementText }), this.createVendorSpecificField({ name: 'repeat', geometryTypes: ["LINE"], fieldLabel: this.repeatText }), this.createVendorSpecificField({ name: 'forceLeftToRight', xtype: "checkbox", geometryTypes: ["LINE"], fieldLabel: this.forceLeftToRightText }), this.createVendorSpecificField({ name: 'group', listeners: { 'check': function(cb, value) { if (this.geometryType === 'LINE') { if (value === false) { this.labelAllGroup.hide(); } else { this.labelAllGroup.show(); } } }, scope: this }, xtype: 'checkbox', yesno: true, fieldLabel: this.groupText }), this.createVendorSpecificField({ name: 'labelAllGroup', ref: "../labelAllGroup", geometryTypes: ["LINE"], hidden: (this.symbolizer.vendorOptions['group'] !== 'yes'), xtype: "checkbox", fieldLabel: this.labelAllGroupText }), this.createVendorSpecificField({ name: 'conflictResolution', xtype: "checkbox", listeners: { 'check': function(cb, checked) { if (!checked) { this.spaceAround.hide(); } else { this.spaceAround.show(); } }, scope: this }, fieldLabel: this.conflictResolutionText }), this.createVendorSpecificField({ name: 'spaceAround', hidden: (this.symbolizer.vendorOptions['conflictResolution'] !== true), allowNegative: true, ref: "../spaceAround", fieldLabel: this.spaceAroundText }), this.createVendorSpecificField({ name: 'goodnessOfFit', geometryTypes: ['POLYGON'], fieldLabel: this.goodnessOfFitText }) ] }]; this.addEvents( /** * Event: change * Fires before any field blurs if the field value has changed. * * Listener arguments: * symbolizer - {Object} A symbolizer with text related properties * updated. */ "change" ); gxp.TextSymbolizer.superclass.initComponent.call(this); }, createField: function(config) { var field = Ext.ComponentMgr.create(config); if (config.geometryTypes) { this.on('geometrytype', function(type) { if (config.geometryTypes.indexOf(type) === -1) { field.hide(); } }); } return field; }, /** * private: method[createVendorSpecificField] * :arg config: ``Object`` config object for the field to create * * Create a form field that will generate a VendorSpecific tag. */ createVendorSpecificField: function(config) { var listener = function(field, value) { // empty VendorOption tags can cause null pointer exceptions in GeoServer if (Ext.isEmpty(value)) { delete this.symbolizer.vendorOptions[config.name]; } else { if (config.yesno === true) { this.symbolizer.vendorOptions[config.name] = (value == true) ? 'yes': 'no'; } else { this.symbolizer.vendorOptions[config.name] = value; } } this.fireEvent("change", this.symbolizer); }; var field = Ext.ComponentMgr.create(Ext.applyIf(config, { xtype: "numberfield", allowNegative: false, value: config.value || this.symbolizer.vendorOptions[config.name], checked: (config.yesno === true) ? (this.symbolizer.vendorOptions[config.name] === 'yes') : this.symbolizer.vendorOptions[config.name], plugins: [{ ptype: 'gxp_formfieldhelp', dismissDelay: 20000, helpText: this[config.name.replace(/-/g, '_') + 'Help'] }] })); field.on("change", listener, this); field.on("check", listener, this); if (config.geometryTypes) { this.on('geometrytype', function(type) { if (config.geometryTypes.indexOf(type) === -1) { field.hide(); } }); } return field; }, showHideGeometryOptions: function() { var geomRegex = /gml:((Multi)?(Point|Line|Polygon|Curve|Surface|Geometry)).*/; var polygonRegex = /gml:((Multi)?(Polygon|Surface)).*/; var pointRegex = /gml:((Multi)?(Point)).*/; var lineRegex = /gml:((Multi)?(Line|Curve|Surface)).*/; var geomType = null; this.attributes.each(function(r) { var type = r.get("type"); var match = geomRegex.exec(type); if (match) { if (polygonRegex.exec(type)) { geomType = "POLYGON"; } else if (pointRegex.exec(type)) { geomType = "POINT"; } else if (lineRegex.exec(type)) { geomType = "LINE"; } } }, this); if (geomType !== null) { this.geometryType = geomType; this.fireEvent('geometrytype', geomType); } } }); /** api: xtype = gxp_textsymbolizer */ Ext.reg('gxp_textsymbolizer', gxp.TextSymbolizer); /** FILE: widgets/FillSymbolizer.js **/ /** * Copyright (c) 2008-2011 The Open Planning Project * * Published under the GPL license. * See https://github.com/opengeo/gxp/raw/master/license.txt for the full text * of the license. */ /** * @include widgets/form/ColorField.js */ /** api: (define) * module = gxp * class = FillSymbolizer * base_link = `Ext.FormPanel `_ */ Ext.namespace("gxp"); /** api: constructor * .. class:: FillSymbolizer(config) * * Form for configuring a symbolizer fill. */ gxp.FillSymbolizer = Ext.extend(Ext.FormPanel, { /** api: config[symbolizer] * ``Object`` * A symbolizer object that will be used to fill in form values. * This object will be modified when values change. Clone first if * you do not want your symbolizer modified. */ symbolizer: null, /** api: config[colorProperty] * ``String`` The property that should be set on the symbolizer to * represent the fill color. Defaults to fillColor. But can also be * set to fontColor for labels. */ colorProperty: "fillColor", /** api: config[opacityProperty] * ``String`` The property that should be set on the symbolizer to * represent the fill opacity. Defaults to fillOpacity. But can also be * set to fontOpacity for labels. */ opacityProperty: "fillOpacity", /** api: config[colorManager] * ``Function`` * Optional color manager constructor to be used as a plugin for the color * field. */ colorManager: null, /** api: config[checkboxToggle] * ``Boolean`` Set to false if the "Fill" fieldset should not be * toggleable. Default is true. */ checkboxToggle: true, /** api: config[defaultColor] * ``String`` Default background color for the Color field. This * color will be displayed when no fillColor value for the symbolizer * is available. Defaults to the ``fillColor`` property of * ``OpenLayers.Renderer.defaultSymbolizer``. */ defaultColor: null, border: false, /** i18n */ fillText: "Fill", colorText: "Color", opacityText: "Opacity", initComponent: function() { if(!this.symbolizer) { this.symbolizer = {}; } var colorFieldPlugins; if (this.colorManager) { colorFieldPlugins = [new this.colorManager()]; } var sliderValue = 100; if (this.opacityProperty in this.symbolizer) { sliderValue = this.symbolizer[this.opacityProperty]*100; } else if (OpenLayers.Renderer.defaultSymbolizer[this.opacityProperty]) { sliderValue = OpenLayers.Renderer.defaultSymbolizer[this.opacityProperty]*100; } this.items = [{ xtype: "fieldset", title: this.fillText, autoHeight: true, checkboxToggle: this.checkboxToggle, collapsed: this.checkboxToggle === true && this.symbolizer.fill === false, hideMode: "offsets", defaults: { width: 100 // TODO: move to css }, items: [{ xtype: "gxp_colorfield", fieldLabel: this.colorText, name: "color", emptyText: OpenLayers.Renderer.defaultSymbolizer[this.colorProperty], value: this.symbolizer[this.colorProperty], defaultBackground: this.defaultColor || OpenLayers.Renderer.defaultSymbolizer[this.colorProperty], plugins: colorFieldPlugins, listeners: { valid: function(field) { var newValue = field.getValue(); var modified = this.symbolizer[this.colorProperty] != newValue; this.symbolizer[this.colorProperty] = newValue; modified && this.fireEvent("change", this.symbolizer); }, scope: this } }, { xtype: "slider", fieldLabel: this.opacityText, name: "opacity", values: [sliderValue], isFormField: true, listeners: { changecomplete: function(slider, value) { this.symbolizer[this.opacityProperty] = value / 100; this.fireEvent("change", this.symbolizer); }, scope: this }, plugins: [ new GeoExt.SliderTip({ getText: function(thumb) { return thumb.value + "%"; } }) ] }], listeners: { "collapse": function() { if (this.symbolizer.fill !== false) { this.symbolizer.fill = false; this.fireEvent("change", this.symbolizer); } }, "expand": function() { this.symbolizer.fill = true; this.fireEvent("change", this.symbolizer); }, scope: this } }]; this.addEvents( /** * Event: change * Fires before any field blurs if the field value has changed. * * Listener arguments: * symbolizer - {Object} A symbolizer with fill related properties * updated. */ "change" ); gxp.FillSymbolizer.superclass.initComponent.call(this); } }); /** api: xtype = gxp_fillsymbolizer */ Ext.reg('gxp_fillsymbolizer', gxp.FillSymbolizer); /** FILE: widgets/form/ColorField.js **/ /** * Copyright (c) 2008-2011 The Open Planning Project * * Published under the GPL license. * See https://github.com/opengeo/gxp/raw/master/license.txt for the full text * of the license. */ /** api: (define) * module = gxp.form * class = ColorField * base_link = `Ext.form.TextField `_ */ Ext.namespace("gxp.form"); /** api: constructor * .. class:: ColorField(config) * * A text field that colors its own background based on the input value. * The value may be any one of the 16 W3C supported CSS color names * (http://www.w3.org/TR/css3-color/). The value can also be an arbitrary * RGB hex value prefixed by a '#' (e.g. '#FFCC66'). */ gxp.form.ColorField = Ext.extend(Ext.form.TextField, { /** api: property[cssColors] * ``Object`` * Properties are supported CSS color names. Values are RGB hex strings * (prefixed with '#'). */ cssColors: { aqua: "#00FFFF", black: "#000000", blue: "#0000FF", fuchsia: "#FF00FF", gray: "#808080", green: "#008000", lime: "#00FF00", maroon: "#800000", navy: "#000080", olive: "#808000", purple: "#800080", red: "#FF0000", silver: "#C0C0C0", teal: "#008080", white: "#FFFFFF", yellow: "#FFFF00" }, /** api: config[defaultBackground] * The default background color if the symbolizer has no fillColor set. * Defaults to #ffffff. */ defaultBackground: "#ffffff", /** private: method[initComponent] * Override */ initComponent: function() { if (this.value) { this.value = this.hexToColor(this.value); } gxp.form.ColorField.superclass.initComponent.call(this); // Add the colorField listener to color the field. this.on({ render: this.colorField, valid: this.colorField, scope: this }); }, /** private: method[isDark] * :arg hex: ``String`` A RGB hex color string (prefixed by '#'). * :returns: ``Boolean`` The color is dark. * * Determine if a color is dark by avaluating brightness according to the * W3C suggested algorithm for calculating brightness of screen colors. * http://www.w3.org/WAI/ER/WD-AERT/#color-contrast */ isDark: function(hex) { var dark = false; if(hex) { // convert hex color values to decimal var r = parseInt(hex.substring(1, 3), 16) / 255; var g = parseInt(hex.substring(3, 5), 16) / 255; var b = parseInt(hex.substring(5, 7), 16) / 255; // use w3C brightness measure var brightness = (r * 0.299) + (g * 0.587) + (b * 0.144); dark = brightness < 0.5; } return dark; }, /** private: method[colorField] * Set the background and font color for the field. */ colorField: function() { var color = this.colorToHex(this.getValue()) || this.defaultBackground; this.getEl().setStyle({ "background": color, "color": this.isDark(color) ? "#ffffff" : "#000000" }); }, /** api: method[getHexValue] * :returns: ``String`` The RGB hex string for the field's value (prefixed * with '#'). * * As a compliment to the field's ``getValue`` method, this method always * returns the RGB hex string representation of the current value * in the field (given a named color or a hex string). */ getHexValue: function() { return this.colorToHex( gxp.form.ColorField.superclass.getValue.apply(this, arguments)); }, /** api: method[getValue] * :returns: ``String`` The RGB hex string for the field's value (prefixed * with '#'). * * This method always returns the RGB hex string representation of the * current value in the field (given a named color or a hex string), * except for the case when the value has not been changed. */ getValue: function() { var v = this.getHexValue(); var o = this.initialConfig.value; if (v === this.hexToColor(o)) { v = o; } return v; }, /** api: method[setValue] * :arg value: ``Object`` * * Sets the value of the field. If the value matches one of the well known * colors in ``cssColors``, a human readable value will be displayed * instead of the hex code. */ setValue: function(value) { gxp.form.ColorField.superclass.setValue.apply(this, [this.hexToColor(value)]); }, /** private: method[colorToHex] * :returns: ``String`` A RGB hex color string or null if none found. * * Return the RGB hex representation of a color string. If a CSS supported * named color is supplied, the hex representation will be returned. * If a non-CSS supported named color is supplied, null will be * returned. If a RGB hex string is supplied, the same will be returned. */ colorToHex: function(color) { if (!color) { return color; } var hex; if (color.match(/^#[0-9a-f]{6}$/i)) { hex = color; } else { hex = this.cssColors[color.toLowerCase()] || null; } return hex; }, /** private: method[hexToColor] */ hexToColor: function(hex) { if (!hex) { return hex; } var color = hex; for (var c in this.cssColors) { if (this.cssColors[c] == color.toUpperCase()) { color = c; break; } } return color; } }); Ext.reg("gxp_colorfield", gxp.form.ColorField); /** FILE: widgets/form/FontComboBox.js **/ /** * Copyright (c) 2008-2011 The Open Planning Project * * Published under the GPL license. * See https://github.com/opengeo/gxp/raw/master/license.txt for the full text * of the license. */ /** api: (define) * module = gxp.form * class = FontComboBox * base_link = `Ext.form.ComboBox `_ */ Ext.namespace("gxp.form"); /** api: constructor * .. class:: FontComboBox(config) * * A combo box for selecting a font. */ gxp.form.FontComboBox = Ext.extend(Ext.form.ComboBox, { /** api: property[fonts] * ``Array`` * List of font families to choose from. Default is ["Arial", * "Courier New", "Tahoma", "Times New Roman", "Verdana"]. */ fonts: [ "Serif", "SansSerif", "Arial", "Courier New", "Tahoma", "Times New Roman", "Verdana" ], /** api: property[defaultFont] * ``String`` * The ``fonts`` item to select by default. */ defaultFont: "Serif", allowBlank: false, mode: "local", triggerAction: "all", editable: false, initComponent: function() { var fonts = this.fonts || gxp.form.FontComboBox.prototype.fonts; var defaultFont = this.defaultFont; if (fonts.indexOf(this.defaultFont) === -1) { defaultFont = fonts[0]; } var defConfig = { displayField: "field1", valueField: "field1", store: fonts, value: defaultFont, tpl: new Ext.XTemplate( '' + '
    ' + '{field1}' + '
    ' ) }; Ext.applyIf(this, defConfig); gxp.form.FontComboBox.superclass.initComponent.call(this); } }); /** api: xtype = gxp_fontcombo */ Ext.reg("gxp_fontcombo", gxp.form.FontComboBox); /** FILE: widgets/PolygonSymbolizer.js **/ /** * Copyright (c) 2008-2011 The Open Planning Project * * Published under the GPL license. * See https://github.com/opengeo/gxp/raw/master/license.txt for the full text * of the license. */ /** * @include widgets/FillSymbolizer.js * @include widgets/StrokeSymbolizer.js */ /** api: (define) * module = gxp * class = PolygonSymbolizer * base_link = `Ext.Panel `_ */ Ext.namespace("gxp"); /** api: constructor * .. class:: PolygonSymbolizer(config) * * Form for configuring a polygon symbolizer. */ gxp.PolygonSymbolizer = Ext.extend(Ext.Panel, { /** api: config[symbolizer] * ``Object`` * A symbolizer object that will be used to fill in form values. * This object will be modified when values change. Clone first if * you do not want your symbolizer modified. */ symbolizer: null, initComponent: function() { this.items = [{ xtype: "gxp_fillsymbolizer", symbolizer: this.symbolizer, listeners: { change: function(symbolizer) { this.fireEvent("change", this.symbolizer); }, scope: this } }, { xtype: "gxp_strokesymbolizer", symbolizer: this.symbolizer, listeners: { change: function(symbolizer) { this.fireEvent("change", this.symbolizer); }, scope: this } }]; this.addEvents( /** * Event: change * Fires before any field blurs if the field value has changed. * * Listener arguments: * symbolizer - {Object} A symbolizer with stroke related properties * updated. */ "change" ); gxp.PolygonSymbolizer.superclass.initComponent.call(this); } }); /** api: xtype = gxp_linesymbolizer */ Ext.reg('gxp_polygonsymbolizer', gxp.PolygonSymbolizer); /** FILE: widgets/StrokeSymbolizer.js **/ /** * Copyright (c) 2008-2011 The Open Planning Project * * Published under the GPL license. * See https://github.com/opengeo/gxp/raw/master/license.txt for the full text * of the license. */ /** * @include widgets/form/ColorField.js */ /** api: (define) * module = gxp * class = StrokeSymbolizer * base_link = `Ext.FormPanel `_ */ Ext.namespace("gxp"); /** api: constructor * .. class:: StrokeSymbolizer(config) * * Form for configuring a symbolizer stroke. */ gxp.StrokeSymbolizer = Ext.extend(Ext.FormPanel, { /* i18n */ solidStrokeName: "solid", dashStrokeName: "dash", dotStrokeName: "dot", titleText: "Stroke", styleText: "Style", colorText: "Color", widthText: "Width", opacityText: "Opacity", /* ~i18n */ /** api: config[symbolizer] * ``Object`` * A symbolizer object that will be used to fill in form values. * This object will be modified when values change. Clone first if * you do not want your symbolizer modified. */ symbolizer: null, /** api: config[colorManager] * ``Function`` * Optional color manager constructor to be used as a plugin for the color * field. */ colorManager: null, /** api: config[checkboxToggle] * ``Boolean`` Set to false if the "Fill" fieldset should not be * toggleable. Default is true. */ checkboxToggle: true, /** api: config[defaultColor] * ``String`` Default background color for the Color field. This * color will be displayed when no strokeColor value for the symbolizer * is available. Defaults to the ``strokeColor`` property of * ``OpenLayers.Renderer.defaultSymbolizer``. */ defaultColor: null, /** api: config[dashStyles] * ``Array(Array)`` * A list of [value, name] pairs for stroke dash styles. * The first item in each list is the value and the second is the * display name. Default is [["solid", "solid"], ["2 4", "dash"], * ["1 4", "dot"]]. */ dashStyles: null, border: false, initComponent: function() { this.dashStyles = this.dashStyles || [["solid", this.solidStrokeName], ["4 4", this.dashStrokeName], ["2 4", this.dotStrokeName]]; if(!this.symbolizer) { this.symbolizer = {}; } var colorFieldPlugins; if (this.colorManager) { colorFieldPlugins = [new this.colorManager]; } this.items = [{ xtype: "fieldset", title: this.titleText, autoHeight: true, checkboxToggle: this.checkboxToggle, collapsed: this.checkboxToggle === true && this.symbolizer.stroke === false, hideMode: "offsets", defaults: { width: 100 // TODO: move to css }, items: [{ xtype: "combo", name: "style", fieldLabel: this.styleText, store: new Ext.data.SimpleStore({ data: this.dashStyles, fields: ["value", "display"] }), displayField: "display", valueField: "value", value: this.getDashArray(this.symbolizer.strokeDashstyle) || OpenLayers.Renderer.defaultSymbolizer.strokeDashstyle, mode: "local", allowBlank: true, triggerAction: "all", editable: false, listeners: { select: function(combo, record) { this.symbolizer.strokeDashstyle = record.get("value"); this.fireEvent("change", this.symbolizer); }, scope: this } }, { xtype: "gxp_colorfield", name: "color", fieldLabel: this.colorText, emptyText: OpenLayers.Renderer.defaultSymbolizer.strokeColor, value: this.symbolizer.strokeColor, defaultBackground: this.defaultColor || OpenLayers.Renderer.defaultSymbolizer.strokeColor, plugins: colorFieldPlugins, listeners: { valid: function(field) { var newValue = field.getValue(); var modified = this.symbolizer.strokeColor != newValue; this.symbolizer.strokeColor = newValue; modified && this.fireEvent("change", this.symbolizer); }, scope: this } }, { xtype: "numberfield", name: "width", fieldLabel: this.widthText, allowNegative: false, emptyText: OpenLayers.Renderer.defaultSymbolizer.strokeWidth, value: this.symbolizer.strokeWidth, listeners: { change: function(field, value) { value = parseFloat(value); if (isNaN(value)) { delete this.symbolizer.strokeWidth; } else { this.symbolizer.strokeWidth = value; } this.fireEvent("change", this.symbolizer); }, scope: this } }, { xtype: "slider", name: "opacity", fieldLabel: this.opacityText, values: [(("strokeOpacity" in this.symbolizer) ? this.symbolizer.strokeOpacity : OpenLayers.Renderer.defaultSymbolizer.strokeOpacity) * 100], isFormField: true, listeners: { changecomplete: function(slider, value) { this.symbolizer.strokeOpacity = value / 100; this.fireEvent("change", this.symbolizer); }, scope: this }, plugins: [ new GeoExt.SliderTip({ getText: function(thumb) { return thumb.value + "%"; } }) ] }], listeners: { "collapse": function() { if (this.symbolizer.stroke !== false) { this.symbolizer.stroke = false; this.fireEvent("change", this.symbolizer); } }, "expand": function() { this.symbolizer.stroke = true; this.fireEvent("change", this.symbolizer); }, scope: this } }]; this.addEvents( /** * Event: change * Fires before any field blurs if the field value has changed. * * Listener arguments: * symbolizer - {Object} A symbolizer with stroke related properties * updated. */ "change" ); gxp.StrokeSymbolizer.superclass.initComponent.call(this); }, getDashArray: function(style) { var array; if (style) { var parts = style.split(/\s+/); var ratio = parts[0] / parts[1]; if (!isNaN(ratio)) { array = ratio >= 1 ? "4 4" : "2 4"; } } return array; } }); /** api: xtype = gxp_strokesymbolizer */ Ext.reg('gxp_strokesymbolizer', gxp.StrokeSymbolizer); /** FILE: widgets/LineSymbolizer.js **/ /** * Copyright (c) 2008-2011 The Open Planning Project * * Published under the GPL license. * See https://github.com/opengeo/gxp/raw/master/license.txt for the full text * of the license. */ /** * @include widgets/StrokeSymbolizer.js */ /** api: (define) * module = gxp * class = LineSymbolizer * base_link = `Ext.Panel `_ */ Ext.namespace("gxp"); /** api: constructor * .. class:: LineSymbolizer(config) * * Form for configuring a line symbolizer. */ gxp.LineSymbolizer = Ext.extend(Ext.Panel, { /** api: config[symbolizer] * ``Object`` * A symbolizer object that will be used to fill in form values. * This object will be modified when values change. Clone first if * you do not want your symbolizer modified. */ symbolizer: null, initComponent: function() { this.items = [{ xtype: "gxp_strokesymbolizer", symbolizer: this.symbolizer, listeners: { change: function(symbolizer) { this.fireEvent("change", this.symbolizer); }, scope: this } }]; this.addEvents( /** * Event: change * Fires before any field blurs if the field value has changed. * * Listener arguments: * symbolizer - {Object} A symbolizer with stroke related properties * updated. */ "change" ); gxp.LineSymbolizer.superclass.initComponent.call(this); } }); /** api: xtype = gxp_linesymbolizer */ Ext.reg('gxp_linesymbolizer', gxp.LineSymbolizer); /** FILE: widgets/PointSymbolizer.js **/ /** * Copyright (c) 2008-2011 The Open Planning Project * * Published under the GPL license. * See https://github.com/opengeo/gxp/raw/master/license.txt for the full text * of the license. */ /** * @include widgets/FillSymbolizer.js * @include widgets/StrokeSymbolizer.js */ /** api: (define) * module = gxp * class = PointSymbolizer * base_link = `Ext.Panel `_ */ Ext.namespace("gxp"); /** api: constructor * .. class:: PointSymbolizer(config) * * Form for configuring a point symbolizer. */ gxp.PointSymbolizer = Ext.extend(Ext.Panel, { /** api: config[symbolizer] * ``Object`` * A symbolizer object that will be used to fill in form values. * This object will be modified when values change. Clone first if * you do not want your symbolizer modified. */ symbolizer: null, /** i18n */ graphicCircleText: "circle", graphicSquareText: "square", graphicTriangleText: "triangle", graphicStarText: "star", graphicCrossText: "cross", graphicXText: "x", graphicExternalText: "external", urlText: "URL", opacityText: "opacity", symbolText: "Symbol", sizeText: "Size", rotationText: "Rotation", /** api: config[pointGraphics] * ``Array`` * A list of objects to be used as the root of the data for a * JsonStore. These will become records used in the selection of * a point graphic. If an object in the list has no "value" property, * the user will be presented with an input to provide their own URL * for an external graphic. By default, names of well-known marks are * provided. In addition, the default list will produce a record with * display of "external" that create an input for an external graphic * URL. * * Fields: * * * display - ``String`` The name to be displayed to the user. * * preview - ``String`` URL to a graphic for preview. * * value - ``String`` Value to be sent to the server. * * mark - ``Boolean`` The value is a well-known name for a mark. If * ``false``, the value will be assumed to be a url for an external graphic. */ pointGraphics: null, /** api: config[colorManager] * ``Function`` * Optional color manager constructor to be used as a plugin for the color * field. */ colorManager: null, /** private: property[external] * ``Boolean`` * Currently using an external graphic. */ external: null, /** private: config[layout] * ``String`` */ layout: "form", initComponent: function() { if(!this.symbolizer) { this.symbolizer = {}; } if (!this.pointGraphics) { this.pointGraphics = [ {display: this.graphicCircleText, value: "circle", mark: true}, {display: this.graphicSquareText, value: "square", mark: true}, {display: this.graphicTriangleText, value: "triangle", mark: true}, {display: this.graphicStarText, value: "star", mark: true}, {display: this.graphicCrossText, value: "cross", mark: true}, {display: this.graphicXText, value: "x", mark: true}, {display: this.graphicExternalText} ]; } this.external = !!this.symbolizer["externalGraphic"]; this.markPanel = new Ext.Panel({ border: false, collapsed: this.external, layout: "form", items: [{ xtype: "gxp_fillsymbolizer", symbolizer: this.symbolizer, labelWidth: this.labelWidth, labelAlign: this.labelAlign, colorManager: this.colorManager, listeners: { change: function(symbolizer) { this.fireEvent("change", this.symbolizer); }, scope: this } }, { xtype: "gxp_strokesymbolizer", symbolizer: this.symbolizer, labelWidth: this.labelWidth, labelAlign: this.labelAlign, colorManager: this.colorManager, listeners: { change: function(symbolizer) { this.fireEvent("change", this.symbolizer); }, scope: this } }] }); this.urlField = new Ext.form.TextField({ name: "url", fieldLabel: this.urlText, value: this.symbolizer["externalGraphic"], hidden: !this.external, listeners: { change: function(field, value) { this.symbolizer["externalGraphic"] = value; this.fireEvent("change", this.symbolizer); }, scope: this }, width: 100 // TODO: push this to css }); this.graphicPanel = new Ext.Panel({ border: false, collapsed: !this.external, layout: "form", items: [this.urlField, { xtype: "slider", name: "opacity", fieldLabel: this.opacityText, value: [(this.symbolizer["graphicOpacity"] == null) ? 100 : this.symbolizer["graphicOpacity"] * 100], isFormField: true, listeners: { changecomplete: function(slider, value) { this.symbolizer["graphicOpacity"] = value / 100; this.fireEvent("change", this.symbolizer); }, scope: this }, plugins: [ new GeoExt.SliderTip({ getText: function(thumb) { return thumb.value + "%"; } }) ], width: 100 // TODO: push this to css }] }); this.items = [{ xtype: "combo", name: "mark", fieldLabel: this.symbolText, store: new Ext.data.JsonStore({ data: {root: this.pointGraphics}, root: "root", fields: ["value", "display", "preview", {name: "mark", type: "boolean"}] }), value: this.external ? 0 : this.symbolizer["graphicName"], displayField: "display", valueField: "value", tpl: new Ext.XTemplate( '' + '
    ' + '' + '{display}' + '' + '{display}' + '
    ' ), mode: "local", allowBlank: false, triggerAction: "all", editable: false, listeners: { select: function(combo, record) { var mark = record.get("mark"); var value = record.get("value"); if(!mark) { if(value) { this.urlField.hide(); this.symbolizer["externalGraphic"] = value; } else { this.urlField.show(); } if(!this.external) { this.external = true; var urlValue = this.urlField.getValue(); if (!Ext.isEmpty(urlValue)) { this.symbolizer["externalGraphic"] = urlValue; } delete this.symbolizer["graphicName"]; this.updateGraphicDisplay(); } } else { if(this.external) { this.external = false; delete this.symbolizer["externalGraphic"]; this.updateGraphicDisplay(); } this.symbolizer["graphicName"] = value; } this.fireEvent("change", this.symbolizer); }, scope: this }, width: 100 // TODO: push this to css }, { xtype: "textfield", name: "size", fieldLabel: this.sizeText, value: this.symbolizer["pointRadius"] && this.symbolizer["pointRadius"] * 2, listeners: { change: function(field, value) { this.symbolizer["pointRadius"] = value / 2; this.fireEvent("change", this.symbolizer); }, scope: this }, width: 100 // TODO: push this to css }, { xtype: "textfield", name: "rotation", fieldLabel: this.rotationText, value: this.symbolizer["rotation"], listeners: { change: function(field, value) { this.symbolizer["rotation"] = value; this.fireEvent("change", this.symbolizer); }, scope: this }, width: 100 // TODO: push this to css }, this.markPanel, this.graphicPanel ]; this.addEvents( /** * Event: change * Fires before any field blurs if the field value has changed. * * Listener arguments: * symbolizer - {Object} A symbolizer with stroke related properties * updated. */ "change" ); gxp.PointSymbolizer.superclass.initComponent.call(this); }, updateGraphicDisplay: function() { if(this.external) { this.markPanel.collapse(); this.graphicPanel.expand(); } else { this.graphicPanel.collapse(); this.markPanel.expand(); } // TODO: window shadow fails to sync } }); /** api: xtype = gxp_pointsymbolizer */ Ext.reg('gxp_pointsymbolizer', gxp.PointSymbolizer); /** FILE: 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: * - * * Is a very close copy of: * - */ 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" }); /** FILE: widgets/ScaleOverlay.js **/ /** * Copyright (c) 2008-2011 The Open Planning Project * * Published under the GPL license. * See https://github.com/opengeo/gxp/raw/master/license.txt for the full text * of the license. */ /** * @requires OpenLayers/Control/ScaleLine.js * @requires GeoExt/data/ScaleStore.js */ /** api: (define) * module = gxp * class = ScaleOverlay * base_link = `Ext.Panel `_ */ Ext.namespace("gxp"); /** api: constructor * .. class:: ScaleOverlay(config) * * Create a panel for showing a ScaleLine control and a combobox for * selecting the map scale. */ gxp.ScaleOverlay = Ext.extend(Ext.Panel, { /** api: config[map] * ``OpenLayers.Map`` or :class:`GeoExt.MapPanel` * The map for which to show the scale info. */ map: null, /** i18n */ zoomLevelText: "Zoom level", /** private: method[initComponent] * Initialize the component. */ initComponent: function() { gxp.ScaleOverlay.superclass.initComponent.call(this); this.cls = 'map-overlay'; if(this.map) { if(this.map instanceof GeoExt.MapPanel) { this.map = this.map.map; } this.bind(this.map); } this.on("beforedestroy", this.unbind, this); }, /** private: method[addToMapPanel] * :param panel: :class:`GeoExt.MapPanel` * * Called by a MapPanel if this component is one of the items in the panel. */ addToMapPanel: function(panel) { this.on({ afterrender: function() { this.bind(panel.map); }, scope: this }); }, /** private: method[stopMouseEvents] * :param e: ``Object`` */ stopMouseEvents: function(e) { e.stopEvent(); }, /** private: method[removeFromMapPanel] * :param panel: :class:`GeoExt.MapPanel` * * Called by a MapPanel if this component is one of the items in the panel. */ removeFromMapPanel: function(panel) { var el = this.getEl(); el.un("mousedown", this.stopMouseEvents, this); el.un("click", this.stopMouseEvents, this); this.unbind(); }, /** private: method[addScaleLine] * * Create the scale line control and add it to the panel. */ addScaleLine: function() { var scaleLinePanel = new Ext.BoxComponent({ autoEl: { tag: "div", cls: "olControlScaleLine overlay-element overlay-scaleline" } }); this.on("afterlayout", function(){ scaleLinePanel.getEl().dom.style.position = 'relative'; scaleLinePanel.getEl().dom.style.display = 'inline'; this.getEl().on("click", this.stopMouseEvents, this); this.getEl().on("mousedown", this.stopMouseEvents, this); }, this); scaleLinePanel.on('render', function(){ var scaleLine = new OpenLayers.Control.ScaleLine({ geodesic: true, div: scaleLinePanel.getEl().dom }); this.map.addControl(scaleLine); scaleLine.activate(); }, this); this.add(scaleLinePanel); }, /** private: method[handleZoomEnd] * * Set the correct value in the scale combo box. */ handleZoomEnd: function() { var scale = this.zoomStore.queryBy(function(record) { return this.map.getZoom() == record.data.level; }, this); if (scale.length > 0) { scale = scale.items[0]; this.zoomSelector.setValue("1 : " + parseInt(scale.data.scale, 10)); } else { if (!this.zoomSelector.rendered) { return; } this.zoomSelector.clearValue(); } }, /** private: method[addScaleCombo] * * Create the scale combo and add it to the panel. */ addScaleCombo: function() { this.zoomStore = new GeoExt.data.ScaleStore({ map: this.map }); this.zoomSelector = new Ext.form.ComboBox({ emptyText: this.zoomLevelText, tpl: '
    1 : {[parseInt(values.scale)]}
    ', editable: false, triggerAction: 'all', mode: 'local', store: this.zoomStore, width: 110 }); this.zoomSelector.on({ click: this.stopMouseEvents, mousedown: this.stopMouseEvents, select: function(combo, record, index) { this.map.zoomTo(record.data.level); }, scope: this }); this.map.events.register('zoomend', this, this.handleZoomEnd); var zoomSelectorWrapper = new Ext.Panel({ items: [this.zoomSelector], cls: 'overlay-element overlay-scalechooser', border: false }); this.add(zoomSelectorWrapper); }, /** private: method[bind] * :param map: ``OpenLayers.Map`` */ bind: function(map) { this.map = map; this.addScaleLine(); this.addScaleCombo(); this.doLayout(); }, /** private: method[unbind] */ unbind: function() { if(this.map && this.map.events) { this.map.events.unregister('zoomend', this, this.handleZoomEnd); } this.zoomStore = null; this.zoomSelector = null; } }); /** api: xtype = gxp_scaleoverlay */ Ext.reg('gxp_scaleoverlay', gxp.ScaleOverlay); /** FILE: plugins/Tool.js **/ /** * Copyright (c) 2008-2011 The Open Planning Project * * Published under the GPL license. * See https://github.com/opengeo/gxp/raw/master/license.txt for the full text * of the license. */ /** * @requires GeoExt/widgets/Action.js */ /** api: (define) * module = gxp.plugins * class = Tool * base_link = `Ext.util.Observable `_ */ Ext.namespace("gxp.plugins"); /** api: constructor * .. class:: Tool(config) * * Base class for plugins that add tool functionality to * :class:`gxp.Viewer`. These plugins are used by adding configuration * objects for them to the ``tools`` array of the viewer's config object, * using their ``ptype``. */ gxp.plugins.Tool = Ext.extend(Ext.util.Observable, { /** api: ptype = gxp_tool */ ptype: "gxp_tool", /** api: config[autoActivate] * ``Boolean`` Set to false if the tool should be initialized without * activating it. Default is true. */ autoActivate: true, /** api: property[active] * ``Boolean`` Is the tool currently active? */ /** api: config[actions] * ``Array`` Custom actions for tools that do not provide their own. Array * elements are expected to be valid Ext config objects or strings * referencing a valid Ext component. Actions provided here may have * additional ``menuText`` and ``buttonText`` properties. The former * will be used as text when the action is used in a menu. The latter will * be conditionally used on buttons, only if ``showButtonText`` is set to * true. The native ``text`` property will unconditionally be used for * buttons. Optional, only needed to create custom actions. */ /** api: config[outputAction] * ``Number`` The ``actions`` array index of the action that should * trigger this tool's output. Only valid if ``actions`` is configured. * Leave this unconfigured if none of the ``actions`` should trigger this * tool's output. */ /** api: config[actionTarget] * ``Object`` or ``String`` or ``Array`` Where to place the tool's actions * (e.g. buttons or menus)? * * In case of a string, this can be any string that references an * ``Ext.Container`` property on the portal, or a unique id configured on a * component. * * In case of an object, the object has a "target" and an "index" property, * so that the tool can be inserted at a specified index in the target. * * actionTarget can also be an array of strings or objects, if the action is * to be put in more than one place (e.g. a button and a context menu item). * * To reference one of the toolbars of an ``Ext.Panel``, ".tbar", ".bbar" or * ".fbar" has to be appended. The default is "map.tbar". The viewer's main * MapPanel can always be accessed with "map" as actionTarget. Set to null if * no actions should be created. * * Some tools provide a context menu. To reference this context menu as * actionTarget for other tools, configure an id in the tool's * outputConfig, and use the id with ".contextMenu" appended. In the * snippet below, a layer tree is created, with a "Remove layer" action * as button on the tree's top toolbar, and as menu item in its context * menu: * * .. code-block:: javascript * * { * xtype: "gxp_layertree", * outputConfig: { * id: "tree", * tbar: [] * } * }, { * xtype: "gxp_removelayer", * actionTarget: ["tree.tbar", "tree.contextMenu"] * } * * If a tool has both actions and output, and you want to force it to * immediately output to a container, set actionTarget to null. If you * want to hide the actions, set actionTarget to false. In this case, you * should configure a defaultAction to make sure that an action is active. */ actionTarget: "map.tbar", /** api: config[showButtonText] * Show the ``buttonText`` an action is configured with, if used as a * button. Default is false. */ showButtonText: false, /** api: config[toggleGroup] * ``String`` If this tool should be radio-button style toggled with other * tools, this string is to identify the toggle group. */ /** api: config[defaultAction] * ``Number`` Optional index of an action that should be active by * default. Only works for actions that are a ``GeoExt.Action`` instance. */ /** api: config[outputTarget] * ``String`` Where to add the tool's output container? This can be any * string that references an ``Ext.Container`` property on the portal, or * "map" to access the viewer's main map. If not provided, a window will * be created. To reference one of the toolbars of an ``Ext.Panel``, * ".tbar", ".bbar" or ".fbar" has to be appended. */ /** api: config[outputConfig] * ``Object`` Optional configuration for the output container. This may * be useful to override the xtype (e.g. "window" instead of "gx_popup"), * or to provide layout configurations when rendering to an * ``outputTarget``. */ /** api: config[controlOptions] * ``Object`` If this tool is associated with an ``OpenLayers.Control`` * then this is an optional object to pass to the constructor of the * associated ``OpenLayers.Control``. */ /** private: property[target] * ``Object`` * The :class:`gxp.Viewer` that this plugin is plugged into. */ /** private: property[actions] * ``Array`` The actions this tool has added to viewer components. */ /** private: property[output] * ``Array`` output added by this container */ output: null, /** private: method[constructor] */ constructor: function(config) { this.initialConfig = config || {}; this.active = false; Ext.apply(this, config); if (!this.id) { this.id = Ext.id(); } this.output = []; this.addEvents( /** api: event[activate] * Fired when the tool is activated. * * Listener arguments: * * tool - :class:`gxp.plugins.Tool` the activated tool */ "activate", /** api: event[deactivate] * Fired when the tool is deactivated. * * Listener arguments: * * tool - :class:`gxp.plugins.Tool` the deactivated tool */ "deactivate" ); gxp.plugins.Tool.superclass.constructor.apply(this, arguments); }, /** private: method[init] * :arg target: ``Object`` The object initializing this plugin. */ init: function(target) { target.tools[this.id] = this; this.target = target; this.autoActivate && this.activate(); this.target.on("portalready", this.addActions, this); }, /** api: method[activate] * :returns: ``Boolean`` true when this tool was activated * * Activates this tool. */ activate: function() { if (this.active === false) { this.active = true; this.fireEvent("activate", this); return true; } }, /** api: method[deactivate] * :returns: ``Boolean`` true when this tool was deactivated * * Deactivates this tool. */ deactivate: function() { if (this.active === true) { this.active = false; this.fireEvent("deactivate", this); return true; } }, /** private: method[getContainer] * :arg target: ``String`` A reference as described for :obj:`actionTarget` * and :obj:`outputTarget` * :returns: ``Ext.Component`` The container reference matching the target. */ getContainer: function(target) { var ct, item, meth, parts = target.split("."), ref = parts[0]; if (ref) { if (ref == "map") { ct = this.target.mapPanel; } else { ct = Ext.getCmp(ref) || this.target.portal[ref]; if (!ct) { throw new Error("Can't find component with id: " + ref); } } } else { ct = this.target.portal; } item = parts.length > 1 && parts[1]; if (item) { meth = { "tbar": "getTopToolbar", "bbar": "getBottomToolbar", "fbar": "getFooterToolbar" }[item]; if (meth) { ct = ct[meth](); } else { ct = ct[item]; } } return ct; }, /** api: method[addActions] * :arg actions: ``Array`` Optional actions to add. If not provided, * this.actions will be added. * :returns: ``Array`` The actions added. */ addActions: function(actions) { actions = actions || this.actions; if (!actions || this.actionTarget === null) { // add output immediately if we have no actions to trigger it this.removeOutput(); this.addOutput(this.outputConfig); return; } var actionTargets = this.actionTarget instanceof Array ? this.actionTarget : [this.actionTarget]; var a = actions instanceof Array ? actions : [actions]; var action, actionTarget, cmp, i, j, jj, ct, index = null; for (i=actionTargets.length-1; i>=0; --i) { actionTarget = actionTargets[i]; if (actionTarget) { if (actionTarget instanceof Object) { index = actionTarget.index; actionTarget = actionTarget.target; } ct = this.getContainer(actionTarget); } for (j=0, jj=a.length; j