Strabon
changeset 542:4f97fb6389a5
added support for viewing a KML in Strabon endpoint when in localhost (using geoxml3)
Fixes Ticket #8: http://bug.strabon.di.uoa.gr/ticket/8
Fixes Ticket #8: http://bug.strabon.di.uoa.gr/ticket/8
author | Babis Nikolaou <charnik@di.uoa.gr> |
---|---|
date | Fri Sep 14 13:03:24 2012 +0300 (2012-09-14) |
parents | f263e457fb35 |
children | f7e99907a8f1 4657ea5fa82d |
files | endpoint/WebContent/js/ProjectedOverlay.js endpoint/WebContent/js/ZipFile.complete.js endpoint/WebContent/js/geoxml3-kmz.js endpoint/WebContent/query.jsp endpoint/pom.xml endpoint/src/main/java/eu/earthobservatory/org/StrabonEndpoint/QueryBean.java |
line diff
1.1 --- /dev/null Thu Jan 01 00:00:00 1970 +0000 1.2 +++ b/endpoint/WebContent/js/ProjectedOverlay.js Fri Sep 14 13:03:24 2012 +0300 1.3 @@ -0,0 +1,138 @@ 1.4 +// Create an overlay on the map from a projected image - Maps v3... 1.5 +// Author. John D. Coryat 05/2009 1.6 +// USNaviguide LLC - http://www.usnaviguide.com 1.7 +// Thanks go to Mile Williams EInsert: http://econym.googlepages.com/einsert.js, Google's GOverlay Example and Bratliff's suggestion... 1.8 +// Opacity code from TPhoto: http://gmaps.tommangan.us/addtphoto.html 1.9 +// This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program; if not, write to the Free Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA. 1.10 +// 1.11 +// Parameters: 1.12 +// map: This Map 1.13 +// imageUrl: URL of the image (Mandatory) 1.14 +// bounds: Bounds object of image destination (Mandatory) 1.15 +// Options: 1.16 +// addZoom: Added Zoom factor as a parameter to the imageUrl (include complete parameter, including separater like '?zoom=' 1.17 +// percentOpacity: Default 50, percent opacity to use when the image is loaded 0-100. 1.18 +// id: Default imageUrl, ID of the div 1.19 +// 1.20 + 1.21 +function ProjectedOverlay(map, imageUrl, bounds, opts) 1.22 +{ 1.23 + google.maps.OverlayView.call(this); 1.24 + 1.25 + this.map_ = map; 1.26 + this.url_ = imageUrl ; 1.27 + this.bounds_ = bounds ; 1.28 + this.addZ_ = opts.addZoom || '' ; // Add the zoom to the image as a parameter 1.29 + this.id_ = opts.id || this.url_ ; // Added to allow for multiple images 1.30 + this.percentOpacity_ = opts.percentOpacity || 50 ; 1.31 + 1.32 + this.setMap(map); 1.33 +} 1.34 + 1.35 +ProjectedOverlay.prototype = new google.maps.OverlayView(); 1.36 + 1.37 +ProjectedOverlay.prototype.createElement = function() 1.38 +{ 1.39 + var panes = this.getPanes() ; 1.40 + var div = this.div_ ; 1.41 + 1.42 + if (!div) 1.43 + { 1.44 + div = this.div_ = document.createElement("div"); 1.45 + div.style.position = "absolute" ; 1.46 + div.setAttribute('id',this.id_) ; 1.47 + this.div_ = div ; 1.48 + this.lastZoom_ = -1 ; 1.49 + if( this.percentOpacity_ ) 1.50 + { 1.51 + this.setOpacity(this.percentOpacity_) ; 1.52 + } 1.53 + panes.overlayLayer.appendChild(div); 1.54 + } 1.55 +} 1.56 + 1.57 +// Remove the main DIV from the map pane 1.58 + 1.59 +ProjectedOverlay.prototype.remove = function() 1.60 +{ 1.61 + if (this.div_) 1.62 + { 1.63 + this.div_.parentNode.removeChild(this.div_); 1.64 + this.div_ = null; 1.65 + } 1.66 +} 1.67 + 1.68 +// Redraw based on the current projection and zoom level... 1.69 + 1.70 +ProjectedOverlay.prototype.draw = function(firstTime) 1.71 +{ 1.72 + // Creates the element if it doesn't exist already. 1.73 + 1.74 + this.createElement(); 1.75 + 1.76 + if (!this.div_) 1.77 + { 1.78 + return ; 1.79 + } 1.80 + 1.81 + var c1 = this.get('projection').fromLatLngToDivPixel(this.bounds_.getSouthWest()); 1.82 + var c2 = this.get('projection').fromLatLngToDivPixel(this.bounds_.getNorthEast()); 1.83 + 1.84 + if (!c1 || !c2) return; 1.85 + 1.86 + // Now position our DIV based on the DIV coordinates of our bounds 1.87 + 1.88 + this.div_.style.width = Math.abs(c2.x - c1.x) + "px"; 1.89 + this.div_.style.height = Math.abs(c2.y - c1.y) + "px"; 1.90 + this.div_.style.left = Math.min(c2.x, c1.x) + "px"; 1.91 + this.div_.style.top = Math.min(c2.y, c1.y) + "px"; 1.92 + 1.93 + // Do the rest only if the zoom has changed... 1.94 + 1.95 + if ( this.lastZoom_ == this.map_.getZoom() ) 1.96 + { 1.97 + return ; 1.98 + } 1.99 + 1.100 + this.lastZoom_ = this.map_.getZoom() ; 1.101 + 1.102 + var url = this.url_ ; 1.103 + 1.104 + if ( this.addZ_ ) 1.105 + { 1.106 + url += this.addZ_ + this.map_.getZoom() ; 1.107 + } 1.108 + 1.109 + this.div_.innerHTML = '<img src="' + url + '" width=' + this.div_.style.width + ' height=' + this.div_.style.height + ' >' ; 1.110 +} 1.111 + 1.112 +ProjectedOverlay.prototype.setOpacity=function(opacity) 1.113 +{ 1.114 + if (opacity < 0) 1.115 + { 1.116 + opacity = 0 ; 1.117 + } 1.118 + if(opacity > 100) 1.119 + { 1.120 + opacity = 100 ; 1.121 + } 1.122 + var c = opacity/100 ; 1.123 + 1.124 + if (typeof(this.div_.style.filter) =='string') 1.125 + { 1.126 + this.div_.style.filter = 'alpha(opacity:' + opacity + ')' ; 1.127 + } 1.128 + if (typeof(this.div_.style.KHTMLOpacity) == 'string' ) 1.129 + { 1.130 + this.div_.style.KHTMLOpacity = c ; 1.131 + } 1.132 + if (typeof(this.div_.style.MozOpacity) == 'string') 1.133 + { 1.134 + this.div_.style.MozOpacity = c ; 1.135 + } 1.136 + if (typeof(this.div_.style.opacity) == 'string') 1.137 + { 1.138 + this.div_.style.opacity = c ; 1.139 + } 1.140 +} 1.141 +
2.1 --- /dev/null Thu Jan 01 00:00:00 1970 +0000 2.2 +++ b/endpoint/WebContent/js/ZipFile.complete.js Fri Sep 14 13:03:24 2012 +0300 2.3 @@ -0,0 +1,2172 @@ 2.4 +// ZipFile.complete.js 2.5 +// 2.6 +// 2/17/2012 2.7 +// 2.8 +// ======================================================= 2.9 +// 2.10 + 2.11 +// JSIO.core.js 2.12 +// ------------------------------------------------------------------ 2.13 +// 2.14 +// core methods for Javascript IO. 2.15 +// 2.16 +// ======================================================= 2.17 +// 2.18 +// Copyleft (c) 2010, Dino Chiesa via MS-PL 2.19 +// Copyleft (c) 2012, Brendan Byrd via GPL 2.20 +// 2.21 +// This work is licensed under the GPLv3. 2.22 + 2.23 + 2.24 +(function(){ 2.25 + if (typeof JSIO == "object"){ 2.26 + var e1 = new Error("JSIO is already defined"); 2.27 + e1.source = "JSIO.core.js"; 2.28 + throw e1; 2.29 + } 2.30 + 2.31 + JSIO = {}; 2.32 + 2.33 + JSIO.version = "2.0 2012Feb"; 2.34 + 2.35 + JSIO.throwError = function(msg, source, sub) { 2.36 + var error = new Error("Error: " + msg); 2.37 + error.source = (source || this._typename || 'JSIO') + (sub ? '.'+sub : ''); 2.38 + throw error; 2.39 + } 2.40 + 2.41 + // Format a number as hex. Quantities over 7ffffff will be displayed properly. 2.42 + JSIO.decimalToHexString = function(number, digits) { 2.43 + if (number < 0) { 2.44 + number = 0xFFFFFFFF + number + 1; 2.45 + } 2.46 + var r1 = number.toString(16).toUpperCase(); 2.47 + if (digits) { 2.48 + r1 = "00000000" + r1; 2.49 + r1 = r1.substring(r1.length - digits); 2.50 + } 2.51 + return r1; 2.52 + }; 2.53 + 2.54 + JSIO.FileType = { 2.55 + Text : 0, 2.56 + Binary : 1, 2.57 + XML : 2, 2.58 + Unknown : 3 2.59 + }; 2.60 + 2.61 + 2.62 + JSIO.guessFileType = function(name) { 2.63 + if (name == "makefile") { return JSIO.FileType.Text; } 2.64 + 2.65 + var lastDot = name.lastIndexOf("."); 2.66 + if (lastDot <= 0) { return JSIO.FileType.Unknown; } 2.67 + 2.68 + var ext= name.substring(lastDot); 2.69 + if (ext == ".zip") { return JSIO.FileType.Binary; } 2.70 + if (ext == ".xlsx") { return JSIO.FileType.Binary; } 2.71 + if (ext == ".docx") { return JSIO.FileType.Binary; } 2.72 + if (ext == ".dll") { return JSIO.FileType.Binary; } 2.73 + if (ext == ".obj") { return JSIO.FileType.Binary; } 2.74 + if (ext == ".pdb") { return JSIO.FileType.Binary; } 2.75 + if (ext == ".exe") { return JSIO.FileType.Binary; } 2.76 + if (ext == ".kmz") { return JSIO.FileType.Binary; } 2.77 + 2.78 + if (ext == ".xml") { return JSIO.FileType.XML; } 2.79 + if (ext == ".xsl") { return JSIO.FileType.XML; } 2.80 + if (ext == ".kml") { return JSIO.FileType.XML; } 2.81 + if (ext == ".csproj") { return JSIO.FileType.XML; } 2.82 + if (ext == ".vbproj") { return JSIO.FileType.XML; } 2.83 + if (ext == ".shfbproj") { return JSIO.FileType.XML; } 2.84 + if (ext == ".resx") { return JSIO.FileType.XML; } 2.85 + if (ext == ".xslt") { return JSIO.FileType.XML; } 2.86 + 2.87 + if (ext == ".sln") { return JSIO.FileType.Text; } 2.88 + if (ext == ".htm") { return JSIO.FileType.Text; } 2.89 + if (ext == ".html") { return JSIO.FileType.Text; } 2.90 + if (ext == ".js") { return JSIO.FileType.Text; } 2.91 + if (ext == ".vb") { return JSIO.FileType.Text; } 2.92 + if (ext == ".txt") { return JSIO.FileType.Text; } 2.93 + if (ext == ".rels") { return JSIO.FileType.Text; } 2.94 + if (ext == ".css") { return JSIO.FileType.Text; } 2.95 + if (ext == ".cs") { return JSIO.FileType.Text; } 2.96 + if (ext == ".asp") { return JSIO.FileType.Text; } 2.97 + 2.98 + return JSIO.FileType.Unknown; 2.99 + }; 2.100 + 2.101 + JSIO.stringOfLength = function (charCode, length) { 2.102 + var s3 = ""; 2.103 + for (var i = 0; i < length; i++) { 2.104 + s3 += String.fromCharCode(charCode); 2.105 + } 2.106 + return s3; 2.107 + }; 2.108 + 2.109 + JSIO.formatByteArray = function(b) { 2.110 + var s1 = "0000 "; 2.111 + var s2 = ""; 2.112 + for (var i = 0; i < b.length; i++) { 2.113 + if (i !== 0 && i % 16 === 0) { 2.114 + s1 += " " + s2 +"\n" + JSIO.decimalToHexString(i, 4) + " "; 2.115 + s2 = ""; 2.116 + } 2.117 + s1 += JSIO.decimalToHexString(b[i], 2) + " "; 2.118 + if (b[i] >=32 && b[i] <= 126) { 2.119 + s2 += String.fromCharCode(b[i]); 2.120 + } else { 2.121 + s2 += "."; 2.122 + } 2.123 + } 2.124 + if (s2.length > 0) { 2.125 + s1 += JSIO.stringOfLength(32, ((i%16>0)? ((16 - i%16) * 3) : 0) + 4) + s2; 2.126 + } 2.127 + return s1; 2.128 + }; 2.129 + 2.130 + JSIO.htmlEscape = function(str) { 2.131 + return str 2.132 + .replace(new RegExp( "&", "g" ), "&") 2.133 + .replace(new RegExp( "<", "g" ), "<") 2.134 + .replace(new RegExp( ">", "g" ), ">") 2.135 + .replace(new RegExp( "\x13", "g" ), "<br/>") 2.136 + .replace(new RegExp( "\x10", "g" ), "<br/>"); 2.137 + }; 2.138 + 2.139 + JSIO.massApply = function(func, funcThis, arr, needReturn) { 2.140 + var arrayLimit = 99999; // Chrome has an apply/array limit of 99999; Firefox = 491519 2.141 + if (arr.length < arrayLimit) return func.apply(funcThis, arr); 2.142 + else { 2.143 + var newThis = funcThis; 2.144 + var offset = 0; 2.145 + var end = 99999; 2.146 + 2.147 + while (offset < arr.length) { 2.148 + var arrSlice; 2.149 + if (arr.subarray) arrSlice = arr.subarray(offset, end); 2.150 + else if (arr.slice) arrSlice = arr.slice(offset, end); 2.151 + 2.152 + if (needReturn) newThis += func.apply(newThis, arrSlice); 2.153 + else func.apply(funcThis, arrSlice); 2.154 + 2.155 + offset += arrayLimit; 2.156 + end += arrayLimit; 2.157 + end = Math.min(arr.length, end); 2.158 + } 2.159 + return newThis; 2.160 + } 2.161 + } 2.162 + 2.163 +})(); 2.164 + 2.165 +/// JSIO.core.js ends 2.166 + 2.167 + 2.168 +// JSIO.BasicByteReaders.js 2.169 +// ------------------------------------------------------------------ 2.170 +// 2.171 +// Part of the JSIO library. Adds a couple basic ByteReaders to JSIO. 2.172 +// ByteReaders are forward-only byte-wise readers. They read one byte at 2.173 +// a time from a source. 2.174 +// 2.175 +// ======================================================= 2.176 +// 2.177 +// A ByteReader exposes an interface with these functions: 2.178 +// 2.179 +// readByte() 2.180 +// must return null when EOF is reached. 2.181 +// 2.182 +// readToEnd() 2.183 +// returns an array of all bytes read, to EOF 2.184 +// 2.185 +// beginReadToEnd(callback) 2.186 +// async version of the above 2.187 +// 2.188 +// readBytes(n) 2.189 +// returns an array of the next n bytes from the source 2.190 +// 2.191 +// beginReadBytes(n, callback) 2.192 +// async version of the above 2.193 +// 2.194 +// ======================================================= 2.195 +// 2.196 +// Copyleft (c) 2010, Dino Chiesa via MS-PL 2.197 +// Copyleft (c) 2012, Brendan Byrd via GPL 2.198 +// 2.199 +// This work is licensed under the GPLv3. 2.200 + 2.201 + 2.202 +(function(){ 2.203 + var version = "2.0 2012Feb"; 2.204 + 2.205 + if (typeof JSIO !== "object") { JSIO = {}; } 2.206 + if ((typeof JSIO.version !== "string")) { 2.207 + JSIO.version = version; 2.208 + } 2.209 + else if ((JSIO.version.length < 3) || 2.210 + (JSIO.version.substring(0,3) !== "2.0")) { 2.211 + JSIO.version += " " + version; 2.212 + } 2.213 + 2.214 + // ======================================================= 2.215 + // the base object, used as the prototype of all ByteReader objects. 2.216 + var _byteReaderBase = function () { 2.217 + this.position = 0; 2.218 + // position must be incremented in .readByte() for all derived classes 2.219 + }; 2.220 + 2.221 + _byteReaderBase.prototype._throwError = JSIO.throwError; 2.222 + 2.223 + _byteReaderBase.prototype._limitCheck = function(len, startPos) { 2.224 + var LOE = { 2.225 + len: len, 2.226 + pos: startPos, 2.227 + end: startPos+len 2.228 + }; 2.229 + 2.230 + if (len === 0) return {len:0, pos:0, end:0}; 2.231 + if (len < 0) this._throwError("Invalid read length"); 2.232 + if (!this.length) return {len:len, pos:this.position, end:len+this.position}; 2.233 + if (!startPos >= 0) LOE.pos = this.position; 2.234 + if (this.length <= LOE.pos) this._throwError("EOF reached"); 2.235 + 2.236 + LOE.end = LOE.pos+len; 2.237 + if (this.length < LOE.end) LOE.end = LOE.pos+(LOE.len = this.length-this.position); 2.238 + return LOE; 2.239 + } 2.240 + 2.241 + JSIO.SeekOrigin = { 2.242 + Begin : 0, 2.243 + Current : 1, 2.244 + End : 2, 2.245 + SEEK_SET : 0, 2.246 + SEEK_CUR : 1, 2.247 + SEEK_END : 2 2.248 + }; 2.249 + 2.250 + _byteReaderBase.prototype.seek = function(offset, origin) { 2.251 + switch (origin) { 2.252 + case JSIO.SeekOrigin.Begin: 2.253 + if (offset == this.position) return this.position; 2.254 + if (!this.length) { 2.255 + if (offset < this.position) this._throwError('Uni-directional stream cannot seek backwards', null, 'seek'); 2.256 + else if (offset > this.position) return this.read(offset - this.position); // read will limit check 2.257 + } 2.258 + else { 2.259 + if (this.length < offset) this._throwError('Cannot seek past reader length', null, 'seek'); 2.260 + this.position = offset; 2.261 + } 2.262 + break; 2.263 + case JSIO.SeekOrigin.Current: 2.264 + return this.seek(this.position + offset, JSIO.SeekOrigin.Begin); 2.265 + break; 2.266 + case JSIO.SeekOrigin.End: 2.267 + if (!this.length) this._throwError('Uni-directional stream has no known end length for seek', null, 'seek'); 2.268 + return this.seek(this.length - 1 + offset, JSIO.SeekOrigin.Begin); 2.269 + break; 2.270 + default: 2.271 + this._throwError('Invalid seek method', null, 'seek'); 2.272 + break; 2.273 + } 2.274 + 2.275 + return this.position; 2.276 + }; 2.277 + 2.278 + _byteReaderBase.prototype.read = function(len, startPos) { 2.279 + var LOE = this._limitCheck(len, startPos); 2.280 + if (LOE.len === 0) return []; 2.281 + if (LOE.pos != this.position) this.seek(LOE.pos, JSIO.SeekOrigin.Begin); 2.282 + 2.283 + var bytesRead = []; 2.284 + 2.285 + // Faster methods with an array or stream 2.286 + if (this.array && this.array.subarray) bytesRead = this.array.subarray(LOE.pos, LOE.end); 2.287 + else if (this.array && this.array.slice) bytesRead = this.array.slice(LOE.pos, LOE.end); 2.288 + else if (this.stream) bytesRead = this.stream.read(LOE.len, LOE.pos); 2.289 + else if (this.length) { // Random-access stream 2.290 + for(var i=LOE.pos; i<LOE.end; i++) { bytesRead.push(this.readByteAt(i)); } 2.291 + } 2.292 + else { // Uni-directional stream 2.293 + for(var i=LOE.pos; i<LOE.end; i++) { 2.294 + var b = this.readByte(); 2.295 + if (b === null || b === undefined) break; 2.296 + bytesRead.push(b); 2.297 + } 2.298 + } 2.299 + this.position = LOE.end; 2.300 + return bytesRead; 2.301 + }; 2.302 + 2.303 + _byteReaderBase.prototype.beginRead = function(len, startPos, callback) { 2.304 + var LOE = this._limitCheck(len, startPos); 2.305 + if (LOE.len === 0) return setTimeout(function() { callback([]); }, 1); 2.306 + if (LOE.pos != this.position) this.seek(LOE.pos, JSIO.SeekOrigin.Begin); 2.307 + 2.308 + var bytesRead = []; 2.309 + var thisReader = this; 2.310 + var leftToRead = LOE.len; 2.311 + 2.312 + var readBatchAsync = function() { 2.313 + var c = 0; 2.314 + var pos = thisReader.position; 2.315 + 2.316 + // read a 32k batch 2.317 + var l = (leftToRead >= 32768) ? 32768 : leftToRead; 2.318 + var newBytes = thisReader.read(l); 2.319 + JSIO.massApply(bytesRead.push, bytesRead, newBytes); 2.320 + c += l; 2.321 + leftToRead -= l; 2.322 + if (newBytes.length < l) leftToRead = 0; 2.323 + 2.324 + if (leftToRead>0) setTimeout(readBatchAsync, 1); 2.325 + else callback(bytesRead); 2.326 + }; 2.327 + 2.328 + // kickoff 2.329 + setTimeout(readBatchAsync, 1); // always async, in ALL situations 2.330 + return null; 2.331 + }; 2.332 + 2.333 + _byteReaderBase.prototype.readToEnd = function() { 2.334 + if (this.array && this.array.subarray) return this.array.subarray(this.position); 2.335 + else if (this.array && this.array.slice) return this.array.slice(this.position); 2.336 + else if (this.length) return this.read(this.length - this.position); 2.337 + else return this.read(9000 * 9000); // over 9000 2.338 + }; 2.339 + 2.340 + _byteReaderBase.prototype.beginReadToEnd = function(callback) { 2.341 + if (this.array && this.array.subarray) setTimeout(function() { callback( this.array.subarray(this.position) ); }, 1); 2.342 + else if (this.array && this.array.slice) setTimeout(function() { callback( this.array.slice(this.position) ); }, 1); 2.343 + else if (this.length) return this.beginRead(this.length - this.position, this.position, callback); 2.344 + else return this.beginRead(9000 * 9000, this.position, callback); 2.345 + }; 2.346 + 2.347 + // Generic routines; one of these two MUST be overloaded (preferrably both) 2.348 + _byteReaderBase.prototype.readByte = function(){ 2.349 + if (this.length && this.position >= this.length) return null; // EOF 2.350 + 2.351 + var byte; 2.352 + if (this.array) byte = this.array[this.position++]; 2.353 + else if (this.length) byte = this.readByteAt(this.position++); 2.354 + else if (this.stream) byte = this.stream.read(1)[0]; 2.355 + else byte = this.read(1)[0]; 2.356 + return (byte === null || byte === undefined) ? null : byte; 2.357 + }; 2.358 + _byteReaderBase.prototype.readByteAt = function(i) { 2.359 + var pos = this.position; // no position changes on this one 2.360 + if (i === null || i === undefined) i = this.position; 2.361 + 2.362 + var byte; 2.363 + if (this.array) byte = this.array[i]; 2.364 + else if (i === pos) byte = this.readByte(); 2.365 + else if (this.stream) byte = this.stream.read(1, i)[0]; 2.366 + else byte = this.read(1, i)[0]; 2.367 + 2.368 + this.position = pos; 2.369 + return (byte === null || byte === undefined) ? null : byte; 2.370 + } 2.371 + 2.372 + _byteReaderBase.prototype.readBytes = _byteReaderBase.prototype.read; 2.373 + _byteReaderBase.prototype.beginReadBytes = function(len, callback) { return this.beginRead(len, this.position, callback); }; 2.374 + 2.375 + _byteReaderBase.prototype.readNumber = function(len, startPos){ 2.376 + var LOE = this._limitCheck(len, startPos); 2.377 + if (LOE.len === 0) LOE.len = 1; 2.378 + if (LOE.pos != this.position) this.seek(LOE.pos, JSIO.SeekOrigin.Begin); 2.379 + 2.380 + var result = 0; 2.381 + var bytes = this.read(LOE.len, LOE.pos); 2.382 + for (var i=bytes.length-1; i>=0; i--) { 2.383 + // IE only supports 32-bit integer shifting 2.384 + //result = result << 8 | bytes[i]; 2.385 + result = result*256 + bytes[i]; 2.386 + } 2.387 + return result; 2.388 + }; 2.389 + 2.390 + _byteReaderBase.prototype.readString = function(len, startPos){ 2.391 + var LOE = this._limitCheck(len, startPos); 2.392 + if (LOE.len === 0) return ''; 2.393 + if (LOE.pos != this.position) this.seek(LOE.pos, JSIO.SeekOrigin.Begin); 2.394 + 2.395 + var result = ''; 2.396 + var bytes = this.read(LOE.len, LOE.pos); 2.397 + for(var i=0; i<bytes.length; i++){ 2.398 + result += String.fromCharCode(bytes[i]); 2.399 + } 2.400 + return result; 2.401 + }; 2.402 + 2.403 + _byteReaderBase.prototype.readNullTerminatedString = function(startPos){ 2.404 + var pos = startPos || this.position; 2.405 + if (this.length && this.length < pos) this._throwError('EOF reached', null, 'readNullTerminatedString'); 2.406 + if (pos != this.position) this.seek(pos, JSIO.SeekOrigin.Begin); 2.407 + 2.408 + var slarge = ""; 2.409 + var s = ""; 2.410 + var c = 0; 2.411 + 2.412 + // Faster method with an array 2.413 + if (this.array && this.array.indexOf) { 2.414 + var len = pos - this.array.indexOf(0, pos); 2.415 + if (len > 0) return this.readString(len, pos); 2.416 + } 2.417 + 2.418 + var ch; 2.419 + while(1) { 2.420 + ch = String.fromCharCode(this.readByteAt(pos+c)); 2.421 + if (ch === null) break; 2.422 + 2.423 + s += ch; 2.424 + c++; 2.425 + if(c >= 32768) { 2.426 + slarge += s; 2.427 + s = ""; 2.428 + pos += c; 2.429 + this.position += c; 2.430 + c = 0; 2.431 + } 2.432 + }; 2.433 + this.position = pos + c; 2.434 + return slarge + s; 2.435 + }; 2.436 + 2.437 + _byteReaderBase.prototype.beginReadNullTerminatedString = function(callback, startPos){ 2.438 + var pos = startPos || this.position; 2.439 + if (this.length && this.length < pos) this._throwError('EOF reached', null, 'beginReadNullTerminatedString'); 2.440 + 2.441 + var slarge = ""; 2.442 + var s = ""; 2.443 + var thisBinStream = this; 2.444 + 2.445 + var readBatchAsync = function() { 2.446 + var c = 0; 2.447 + 2.448 + var ch; 2.449 + while(1) { 2.450 + ch = String.fromCharCode(this.readByteAt(pos+c)); 2.451 + if (ch === null) break; 2.452 + 2.453 + s += ch; 2.454 + c++; 2.455 + if(c >= 32768) { 2.456 + slarge += s; 2.457 + s = ""; 2.458 + pos += c; 2.459 + this.position += c; 2.460 + c = 0; 2.461 + } 2.462 + }; 2.463 + 2.464 + thisBinStream.position = pos + c; 2.465 + if (ch!==null) setTimeout(readBatchAsync, 1); 2.466 + else callback(slarge+s); 2.467 + }; 2.468 + 2.469 + // Faster method with an array 2.470 + if (this.array && this.array.indexOf) { 2.471 + var len = pos - this.array.indexOf(0, pos); 2.472 + if (len > 0) readBatchASync = function() { callback(thisBinStream.readString(len, pos)); }; 2.473 + } 2.474 + 2.475 + // kickoff 2.476 + setTimeout(readBatchAsync, 1); // always async, in ALL situations 2.477 + return null; 2.478 + }; 2.479 + 2.480 + 2.481 + 2.482 + JSIO._ByteReaderBase = _byteReaderBase; 2.483 + // ======================================================= 2.484 + 2.485 + 2.486 + 2.487 + 2.488 + // ======================================================= 2.489 + // reads from an array of bytes. 2.490 + // This basically wraps a readByte() fn onto array access. 2.491 + var _arrayReader = function(array) { 2.492 + if (! (this instanceof arguments.callee) ) this._throwError('You must use new to instantiate this class', 'JSIO.ArrayReader', 'ctor'); 2.493 + this.position = 0; 2.494 + this.array = array; 2.495 + this.length = array.length; 2.496 + this._typename = "JSIO.ArrayReader"; 2.497 + this._version = version; 2.498 + return this; 2.499 + }; 2.500 + 2.501 + _arrayReader.prototype = new JSIO._ByteReaderBase(); 2.502 + 2.503 + _arrayReader.prototype.readByte = function() { 2.504 + if (this.position >= this.array.length) return null; // EOF 2.505 + return this.array[this.position++]; 2.506 + }; 2.507 + _arrayReader.prototype.readByteAt = function(i) { 2.508 + return this.array[i]; 2.509 + }; 2.510 + 2.511 + // ======================================================= 2.512 + 2.513 + 2.514 + // ======================================================= 2.515 + // reads bytes at a time from a defined segment of a stream. 2.516 + var _streamSegmentReader = function(stream, offset, len) { 2.517 + if (! (this instanceof arguments.callee) ) this._throwError('You must use new to instantiate this class', 'JSIO.StreamSegmentReader', 'ctor'); 2.518 + if (!stream) this._throwError('You must pass a non-null stream', 'JSIO.StreamSegmentReader', 'ctor'); 2.519 + 2.520 + if (!(offset >= 1)) offset = 0; 2.521 + if (!(len >= 1)) len = 0; 2.522 + 2.523 + this.stream = stream; 2.524 + this.array = null; 2.525 + if (stream.array) { 2.526 + var end = len ? offset + len : null; 2.527 + if (stream.array.subarray) this.array = stream.array.subarray(offset, end); 2.528 + else if (stream.array.slice) this.array = stream.array.slice(offset, end); 2.529 + } 2.530 + this.length = this.array ? this.array.length : (stream.length ? stream.length - offset : null); 2.531 + this.offset = offset; 2.532 + this.limit = len; 2.533 + this.position = 0; 2.534 + this._typeName = 'JSIO.StreamSegmentReader'; 2.535 + this._version = version; 2.536 + 2.537 + if (this.array) { 2.538 + this.readByte = _arrayReader.prototype.readByte; 2.539 + this.readByteAt = _arrayReader.prototype.readByteAt; 2.540 + } 2.541 + return this; 2.542 + }; 2.543 + 2.544 + _streamSegmentReader.prototype = new JSIO._ByteReaderBase(); 2.545 + 2.546 + _streamSegmentReader.prototype.readByte = function() { 2.547 + if (this.limit && this.position >= this.limit) return null; // EOF 2.548 + this.position++; 2.549 + return this.stream.readByteAt(this.offset + this.position - 1); 2.550 + }; 2.551 + _streamSegmentReader.prototype.readByteAt = function(i) { 2.552 + if (this.limit && i >= this.limit) return null; // EOF 2.553 + return this.stream.readByteAt(this.offset + i); 2.554 + }; 2.555 + 2.556 + // ======================================================= 2.557 + 2.558 + JSIO.ArrayReader = _arrayReader; 2.559 + JSIO.StreamReader = _streamSegmentReader; 2.560 + JSIO.StreamSegmentReader = _streamSegmentReader; 2.561 + 2.562 +})(); 2.563 + 2.564 + 2.565 +/// JSIO.BasicByteReaders.js ends 2.566 + 2.567 +// JSIO.BinaryUrlStream.js 2.568 +// ------------------------------------------------------------------ 2.569 +// 2.570 +// a class that acts as a stream wrapper around binary files obtained from URLs. 2.571 +// 2.572 +// ======================================================= 2.573 +// 2.574 +// Copyleft (c) 2008, Andy G.P. Na <nagoon97@naver.com> via an MIT-style license 2.575 +// Copyleft (c) 2012, Brendan Byrd via GPL 2.576 +// 2.577 +// This work is licensed under the GPLv3. 2.578 + 2.579 +(function(){ 2.580 + var version = "2.0 2012Feb"; 2.581 + var typename = "JSIO.BinaryUrlStream"; 2.582 + 2.583 + if ((typeof JSIO !== "object") || 2.584 + (typeof JSIO.version !== "string") || 2.585 + (JSIO.version.length < 3) || 2.586 + (JSIO.version.substring(0,3) !== "2.0")) 2.587 + JSIO.throwError('This extension requires JSIO.core.js v2.0', typename); 2.588 + 2.589 + if (typeof JSIO._ByteReaderBase !== "function") 2.590 + JSIO.throwError('This extension requires JSIO.BasicByteReaders.js', typename); 2.591 + 2.592 + if (/msie/i.test(navigator.userAgent) && !/opera/i.test(navigator.userAgent)) { 2.593 + var IEBinaryToArray_ByteStr_Script = 2.594 + "<!-- IEBinaryToArray_ByteStr -->\r\n"+ 2.595 + "<script type='text/vbscript'>\r\n"+ 2.596 + "Function IEBinaryToArray_ByteStr(Binary)\r\n"+ 2.597 + " IEBinaryToArray_ByteStr = CStr(Binary)\r\n"+ 2.598 + "End Function\r\n"+ 2.599 + "Function IEBinaryToArray_ByteAsc_Last(Binary)\r\n"+ 2.600 + " Dim lastIndex\r\n"+ 2.601 + " lastIndex = LenB(Binary)\r\n"+ 2.602 + " if lastIndex mod 2 Then\r\n"+ 2.603 + " IEBinaryToArray_ByteAsc_Last = AscB( MidB( Binary, lastIndex, 1 ) )\r\n"+ 2.604 + " Else\r\n"+ 2.605 + " IEBinaryToArray_ByteAsc_Last = -1\r\n"+ 2.606 + " End If\r\n"+ 2.607 + "End Function\r\n"+ 2.608 + "</script>\r\n"; 2.609 + 2.610 + // inject VBScript 2.611 + document.write(IEBinaryToArray_ByteStr_Script); 2.612 + } 2.613 + 2.614 + JSIO.IEByteMapping = null; 2.615 + 2.616 + var bus = function(url, callback) { 2.617 + if (! (this instanceof arguments.callee) ) this._throwError('You must use new to instantiate this class', 'JSIO.BinaryUrlStream', 'ctor'); 2.618 + 2.619 + this.callback = callback; 2.620 + this.position = 0; 2.621 + this.length = null; 2.622 + this.readByte = JSIO.ArrayReader.prototype.readByte; 2.623 + this.readByteAt = JSIO.ArrayReader.prototype.readByteAt; 2.624 + this.req = null; 2.625 + this._typename = typename; 2.626 + this._version = version; 2.627 + this.status = "-none-"; 2.628 + 2.629 + var _IeGetBinResource = function(fileURL){ 2.630 + var binStream = this; 2.631 + // http://stackoverflow.com/questions/1919972/how-do-i-access-xhr-responsebody-for-binary-data-from-javascript-in-ie 2.632 + 2.633 + // my helper to convert from responseBody to a byte array 2.634 + var convertResponseBodyToArray = function (binary) { 2.635 + var byteArray = new Array; 2.636 + 2.637 + try { 2.638 + // very fast; very little work involved 2.639 + byteArray = new VBArray(binary).toArray(); 2.640 + } 2.641 + catch(err) { 2.642 + // use the BinaryToArray VBScript 2.643 + if (!JSIO.IEByteMapping) { 2.644 + JSIO.IEByteMapping = {}; 2.645 + for ( var i = 0; i < 256; i++ ) { 2.646 + for ( var j = 0; j < 256; j++ ) { 2.647 + JSIO.IEByteMapping[ String.fromCharCode( i + j * 256 ) ] = [ i, j ]; 2.648 + } 2.649 + } 2.650 + } 2.651 + var rawBytes = IEBinaryToArray_ByteStr(binary); 2.652 + var lastAsc = IEBinaryToArray_ByteAsc_Last(binary); 2.653 + 2.654 + for ( var i = 0; i < rawBytes.length; i++ ) { 2.655 + byteArray.push.apply(byteArray, JSIO.IEByteMapping[ rawBytes.substr(i,1) ]); 2.656 + } 2.657 + if (lastAsc >= 0) byteArray.push(lastAsc); 2.658 + } 2.659 + 2.660 + return byteArray; 2.661 + }; 2.662 + 2.663 + this.req = (function() { 2.664 + if (window.XMLHttpRequest) return new window.XMLHttpRequest(); 2.665 + else if (window.ActiveXObject) { 2.666 + // the many versions of IE's XML fetchers 2.667 + var AXOs = [ 2.668 + 'MSXML2.XMLHTTP.6.0', 2.669 + 'MSXML2.XMLHTTP.5.0', 2.670 + 'MSXML2.XMLHTTP.4.0', 2.671 + 'MSXML2.XMLHTTP.3.0', 2.672 + 'MSXML2.XMLHTTP', 2.673 + 'Microsoft.XMLHTTP', 2.674 + 'MSXML.XMLHTTP' 2.675 + ]; 2.676 + for (var i = 0; i < AXOs.length; i++) { 2.677 + try { return new ActiveXObject(AXOs[i]); } 2.678 + catch(e) { continue; } 2.679 + } 2.680 + } 2.681 + return null; 2.682 + })(); 2.683 + this.req.open("GET", fileURL, true); 2.684 + this.req.setRequestHeader("Accept-Charset", "x-user-defined"); 2.685 + this.req.onreadystatechange = function(event){ 2.686 + if (binStream.req.readyState == 4) { 2.687 + binStream.status = "Status: " + binStream.req.status + ' ' + binStream.req.statusText; 2.688 + if (binStream.req.status == 200) { 2.689 + binStream.array = convertResponseBodyToArray(binStream.req.responseBody); 2.690 + binStream.length = binStream.array.length; 2.691 + if (binStream.length < 0) this._throwError('Failed to load "'+ fileURL + '" after converting'); 2.692 + 2.693 + if (typeof binStream.callback == "function") binStream.callback(binStream); 2.694 + } 2.695 + else { 2.696 + binStream._throwError('Failed to load "'+ fileURL + '": HTTP ' + binStream.status); 2.697 + } 2.698 + } 2.699 + }; 2.700 + this.req.send(); 2.701 + }; 2.702 + 2.703 + var _NormalGetBinResource = function(fileURL){ 2.704 + var binStream= this; 2.705 + this.req = new XMLHttpRequest(); 2.706 + this.req.open('GET', fileURL, true); 2.707 + this.req.onreadystatechange = function(aEvt) { 2.708 + if (binStream.req.readyState == 4) { 2.709 + binStream.status = "Status: " + binStream.req.status + ' ' + binStream.req.statusText; 2.710 + if(binStream.req.status == 200){ 2.711 + var fileContents = binStream.req.responseText; 2.712 + binStream.length = fileContents.byteLength; 2.713 + binStream.array = fileContents.split(''); 2.714 + for ( var i = 0; i < binStream.array.length; i++ ) { 2.715 + binStream.array[i] = binStream.array[i].charCodeAt(0) & 0xff; 2.716 + } 2.717 + 2.718 + if (typeof binStream.callback == "function") binStream.callback(binStream); 2.719 + } 2.720 + else { 2.721 + binStream._throwError('Failed to load "'+ fileURL + '": HTTP ' + binStream.status); 2.722 + } 2.723 + } 2.724 + }; 2.725 + //XHR binary charset opt by Marcus Granado 2006 [http://mgran.blogspot.com] 2.726 + this.req.overrideMimeType('text/plain; charset=x-user-defined'); 2.727 + this.req.send(null); 2.728 + }; 2.729 + 2.730 + // http://stackoverflow.com/questions/327685/is-there-a-way-to-read-binary-data-into-javascript 2.731 + var _ArrayBufferGetBinResource = function(fileURL){ 2.732 + var binStream= this; 2.733 + this.req = new XMLHttpRequest(); 2.734 + this.req.open('GET', fileURL, true); 2.735 + this.req.onreadystatechange = function(aEvt) { 2.736 + if (binStream.req.readyState == 4) { 2.737 + binStream.status = "Status: " + binStream.req.status + ' ' + binStream.req.statusText; 2.738 + if(binStream.req.status == 200){ 2.739 + var fileContents = binStream.req.response; 2.740 + binStream.length = fileContents.byteLength; 2.741 + binStream.array = new Uint8Array(fileContents); 2.742 + if (typeof binStream.callback == "function") binStream.callback(binStream); 2.743 + } 2.744 + else { 2.745 + binStream._throwError('Failed to load "'+ fileURL + '": HTTP ' + binStream.status); 2.746 + } 2.747 + } 2.748 + }; 2.749 + this.req.responseType = 'arraybuffer'; 2.750 + // http://stackoverflow.com/questions/11284728/how-do-i-access-8-bit-binary-data-from-javascript-in-opera 2.751 + this.req.overrideMimeType('application/octet-stream; charset=x-user-defined'); 2.752 + this.req.send(null); 2.753 + }; 2.754 + 2.755 + 2.756 + if (typeof ArrayBuffer !== 'undefined') _ArrayBufferGetBinResource.apply(this, [url]); 2.757 + else if (/msie/i.test(navigator.userAgent) && !/opera/i.test(navigator.userAgent)) _IeGetBinResource.apply(this, [url]); 2.758 + else _NormalGetBinResource.apply(this, [url]); 2.759 + }; 2.760 + 2.761 + bus.prototype = new JSIO._ByteReaderBase(); 2.762 + 2.763 + bus.prototype.readByte = function(){ 2.764 + var byte = this.readByteAt(this.position++); 2.765 + return (byte === null || byte === undefined) ? null : byte; 2.766 + }; 2.767 + 2.768 + JSIO.BinaryUrlStream = bus; 2.769 + 2.770 +})(); 2.771 + 2.772 +/// JSIO.BinaryUrlStream.js ends 2.773 + 2.774 +// JSIO.TextDecoder.js 2.775 +// ------------------------------------------------------------------ 2.776 +// 2.777 +// Part of the JSIO library. Adds text decoders, for UTF-8 and UTF-16, 2.778 +// and plain text. 2.779 +// 2.780 +// ======================================================= 2.781 +// 2.782 +// Derived in part from work by notmasteryet. 2.783 +// http://www.codeproject.com/KB/scripting/Javascript_binaryenc.aspx 2.784 + 2.785 +// Copyleft (c) 2008, notmasteryet via an MIT-style license 2.786 +// Copyleft (c) 2010, Dino Chiesa via MS-PL 2.787 +// Copyleft (c) 2012, Brendan Byrd via GPL 2.788 +// 2.789 +// This work is licensed under the GPLv3. 2.790 + 2.791 +(function(){ 2.792 + var version = "2.0 2012Feb"; 2.793 + var typename = "JSIO.TextDecoder"; 2.794 + 2.795 + if ((typeof JSIO !== "object") || 2.796 + (typeof JSIO.version !== "string") || 2.797 + (JSIO.version.length < 3) || 2.798 + (JSIO.version.substring(0,3) !== "2.0")) 2.799 + JSIO.throwError('This extension requires JSIO.core.js v2.0', typename); 2.800 + 2.801 + if (typeof JSIO._ByteReaderBase !== "function") 2.802 + JSIO.throwError('This extension requires JSIO.BasicByteReaders.js', typename); 2.803 + 2.804 + var _ansi = function(reader) { 2.805 + if (! (this instanceof arguments.callee) ) this._throwError('You must use new to instantiate this class', 'JSIO.TextDecoder.ANSI', 'ctor'); 2.806 + this.byteReader = reader; 2.807 + this.charWidth = 1; 2.808 + this._version = version; 2.809 + this._typename = typename + ".ANSI"; 2.810 + return this; 2.811 + }; 2.812 + 2.813 + _ansi.prototype.readChar = function() { 2.814 + var code = this.byteReader.readByte(); 2.815 + return (code < 0) ? null : String.fromCharCode(code); 2.816 + }; 2.817 + 2.818 + _ansi.prototype.parseChar = function(code) { 2.819 + return (code < 0) ? null : String.fromCharCode(code); 2.820 + }; 2.821 + 2.822 + var _utf16 = function(reader) { 2.823 + if (! (this instanceof arguments.callee) ) this._throwError('You must use new to instantiate this class', 'JSIO.TextDecoder.UTF16', 'ctor'); 2.824 + this.byteReader = reader; 2.825 + this.charWidth = 2; 2.826 + this.bomState = 0; 2.827 + this._version = version; 2.828 + this._typename = typename + ".UTF16"; 2.829 + return this; 2.830 + }; 2.831 + 2.832 + _utf16.prototype.readChar = function() { 2.833 + var b1 = this.byteReader.readByte(); 2.834 + if (b1 < 0) return null; 2.835 + var b2 = this.byteReader.readByte(); 2.836 + if (b2 < 0) this._throwError('Incomplete UTF16 character', null, 'readChar'); 2.837 + 2.838 + if ((this.bomState === 0) && ((b1 + b2) == 509)) { 2.839 + this.bomState = (b2 == 254) ? 1 : 2; 2.840 + 2.841 + b1 = this.byteReader.readByte(); 2.842 + if (b1 < 0) return null; 2.843 + b2 = this.byteReader.readByte(); 2.844 + if (b2 < 0) this._throwError('Incomplete UTF16 character', null, 'readChar'); 2.845 + } 2.846 + else { 2.847 + this.bomState = 1; 2.848 + } 2.849 + return this.parseChar(b1, b2); 2.850 + }; 2.851 + 2.852 + _utf16.prototype.parseChar = function(b1, b2) { 2.853 + return String.fromCharCode( this.bomState == 1 ? (b2 << 8 | b1) : (b1 << 8 | b2) ); 2.854 + } 2.855 + 2.856 + /* RFC 3629 */ 2.857 + var _utf8 = function(reader) { 2.858 + if (! (this instanceof arguments.callee) ) this._throwError('You must use new to instantiate this class', 'JSIO.TextDecoder.UTF8', 'ctor'); 2.859 + this.byteReader = reader; 2.860 + this.charWidth = null; 2.861 + this.waitBom = true; 2.862 + this.strict = false; 2.863 + this.pendingChar = null; 2.864 + this._version = version; 2.865 + this._typename = typename + ".UTF8"; 2.866 + return this; 2.867 + }; 2.868 + 2.869 + _utf8.prototype.readChar = function() { 2.870 + var ch = null; 2.871 + do { 2.872 + if (this.pendingChar !== null) { 2.873 + ch = this.pendingChar; 2.874 + this.pendingChar = null; 2.875 + } 2.876 + else { 2.877 + var b1 = this.byteReader.readByte(); 2.878 + if (b1 === null) return null; 2.879 + 2.880 + if ((b1 & 0x80) === 0) ch = String.fromCharCode(b1); 2.881 + else { 2.882 + var currentPrefix = 0xC0; 2.883 + var ttlBytes = 0; 2.884 + do { 2.885 + var mask = currentPrefix >> 1 | 0x80; 2.886 + if((b1 & mask) == currentPrefix) break; 2.887 + currentPrefix = currentPrefix >> 1 | 0x80; 2.888 + } while(++ttlBytes < 5); 2.889 + 2.890 + if (ttlBytes > 0) { 2.891 + var code; 2.892 + if (ttlBytes === 1) code = (b1 & 0x1F) << 6 | (this.byteReader.readByte() & 0x3F); 2.893 + else { 2.894 + code = code << 6*ttlBytes 2.895 + var bytes = this.byteReader.read(ttlBytes); 2.896 + for (var i = 0; i > ttlBytes; i++) { 2.897 + var bi = bytes[i]; 2.898 + if ((bi & 0xC0) != 0x80) this._throwError('Invalid sequence character', null, 'readChar'); 2.899 + code = (code << 6) | (bi & 0x3F); 2.900 + } 2.901 + } 2.902 + 2.903 + if (code <= 0xFFFF) { 2.904 + ch = (code == 0xFEFF && this.waitBom) ? null : String.fromCharCode(code); 2.905 + } 2.906 + else { 2.907 + var v = code - 0x10000; 2.908 + var w1 = 0xD800 | ((v >> 10) & 0x3FF); 2.909 + var w2 = 0xDC00 | (v & 0x3FF); 2.910 + this.pendingChar = String.fromCharCode(w2); 2.911 + ch = String.fromCharCode(w1); 2.912 + } 2.913 + } 2.914 + else { 2.915 + // a byte higher than 0x80. 2.916 + if (this.strict) this._throwError('Invalid character', null, 'readChar'); 2.917 + // fall back to "super ascii" (eg IBM-437) 2.918 + else ch = String.fromCharCode(b1); 2.919 + } 2.920 + } 2.921 + } 2.922 + this.waitBom = false; 2.923 + } while(ch === null); 2.924 + return ch; 2.925 + }; 2.926 + 2.927 + JSIO.TextDecoder = { 2.928 + Default : _ansi, 2.929 + ANSI : _ansi, 2.930 + UTF16 : _utf16, 2.931 + UTF8 : _utf8 2.932 + }; 2.933 + 2.934 +})(); 2.935 + 2.936 + 2.937 +/// JSIO.TextDecoder.js ends 2.938 + 2.939 +// JSIO.TextReader.js 2.940 +// ------------------------------------------------------------------ 2.941 +// 2.942 +// A reader class that decodes text as it reads. 2.943 +// 2.944 +// ======================================================= 2.945 +// 2.946 +// Methods: 2.947 +// readChar() = read 1 char 2.948 +// read(n) = read n chars 2.949 +// readLine() = read one line of data (to \n) 2.950 +// unreadChar(ch) = unread one char 2.951 +// readToEnd() = read all data in the reader; 2.952 +// return a string. 2.953 +// beginReadToEnd(cb) = asynchronously read all data. 2.954 +// 2.955 +// ======================================================= 2.956 +// 2.957 +// Derived in part from work by notmasteryet. 2.958 +// http://www.codeproject.com/KB/scripting/Javascript_binaryenc.aspx 2.959 +// 2.960 +// Copyleft (c) 2008, notmasteryet via an MIT-style license 2.961 +// Copyleft (c) 2010, Dino Chiesa via MS-PL 2.962 +// Copyleft (c) 2012, Brendan Byrd via GPL 2.963 +// 2.964 +// This work is licensed under the GPLv3. 2.965 + 2.966 + 2.967 +(function(){ 2.968 + var version = "2.0 2012Feb"; 2.969 + var typename = "JSIO.TextReader"; 2.970 + 2.971 + if (typeof JSIO._ByteReaderBase !== "function") 2.972 + JSIO.throwError('This extension requires JSIO.BasicByteReaders.js', typename); 2.973 + 2.974 + var tr = function(textDecoder) { 2.975 + if (! (this instanceof arguments.callee) ) this._throwError('You must use new to instantiate this class', typename, 'ctor'); 2.976 + this.decoder = textDecoder; 2.977 + this._version = version; 2.978 + this._typename = typename; 2.979 + this.unreads = []; 2.980 + }; 2.981 + 2.982 + // read one char 2.983 + tr.prototype.readChar = function() { 2.984 + return (this.unreads.length > 0) ? this.unreads.pop() : this.decoder.readChar(); 2.985 + }; 2.986 + 2.987 + // read a length of data 2.988 + tr.prototype.read = function(n) { 2.989 + // ANSI makes this easy 2.990 + if (this.decoder.charWidth === 1) return JSIO.massApply(String.fromCharCode, new String, this.decoder.byteReader.read(n), true); 2.991 + 2.992 + var s = ""; 2.993 + for (vari=0; i<n; i++) { 2.994 + var ch = this.readChar(); 2.995 + if (ch !== null) s+= ch; 2.996 + else break; 2.997 + } 2.998 + return s; 2.999 + }; 2.1000 + 2.1001 + tr.prototype.unreadChar = function(ch) { 2.1002 + this.unreads.push(ch); 2.1003 + }; 2.1004 + 2.1005 + tr.prototype.readToEnd = function() { 2.1006 + // ANSI makes this easy 2.1007 + if (this.decoder.charWidth === 1) return JSIO.massApply(String.fromCharCode, new String, this.decoder.byteReader.readToEnd(n), true); 2.1008 + 2.1009 + var slarge = ""; 2.1010 + var s = ""; 2.1011 + var c = 0; 2.1012 + var ch = this.readChar(); 2.1013 + while(ch !== null) { 2.1014 + s += ch; 2.1015 + c++; 2.1016 + if(c >= 32768) { 2.1017 + slarge += s; 2.1018 + s = ""; 2.1019 + c = 0; 2.1020 + } 2.1021 + ch = this.readChar(); 2.1022 + } 2.1023 + return slarge + s; 2.1024 + }; 2.1025 + 2.1026 + tr.prototype.beginReadToEnd = function(callback) { 2.1027 + // ANSI makes this easy 2.1028 + if (this.decoder.charWidth === 1) { 2.1029 + this.decoder.byteReader.beginReadToEnd(function (bytes) { 2.1030 + callback( JSIO.massApply(String.fromCharCode, new String, bytes, true) ); 2.1031 + }); 2.1032 + return null; 2.1033 + } 2.1034 + 2.1035 + var slarge = ""; 2.1036 + var s = ""; 2.1037 + var txtrdr = this; 2.1038 + 2.1039 + var readBatchAsync = function() { 2.1040 + var c = 0; 2.1041 + var ch = txtrdr.readChar(); 2.1042 + while(ch !== null) { 2.1043 + s += ch;c++; 2.1044 + if(c >= 32768) { 2.1045 + slarge += s; 2.1046 + s = ""; 2.1047 + break; 2.1048 + } 2.1049 + ch = txtrdr.readChar(); 2.1050 + } 2.1051 + if (ch!==null){ 2.1052 + setTimeout(readBatchAsync, 1); 2.1053 + } 2.1054 + else { 2.1055 + callback(slarge+s); 2.1056 + } 2.1057 + }; 2.1058 + 2.1059 + // kickoff 2.1060 + setTimeout(readBatchAsync, 1); // always async, in ALL situations 2.1061 + return null; 2.1062 + }; 2.1063 + 2.1064 + tr.prototype.readLine = function() { 2.1065 + var s = ""; 2.1066 + var ch = this.readChar(); 2.1067 + if (ch === null) return null; 2.1068 + 2.1069 + while(ch != "\r" && ch != "\n") { 2.1070 + s += ch; 2.1071 + ch = this.readChar(); 2.1072 + if (ch === null) return s; 2.1073 + } 2.1074 + if(ch == "\r") { 2.1075 + ch = this.readChar(); 2.1076 + if(ch !== null && ch != "\n"){ 2.1077 + this.unreadChar(ch); 2.1078 + } 2.1079 + } 2.1080 + return s; 2.1081 + }; 2.1082 + 2.1083 + JSIO.TextReader = tr; 2.1084 + 2.1085 +})(); 2.1086 + 2.1087 + 2.1088 +/// JSIO.TextReader.js ends 2.1089 + 2.1090 +// JSIO.Crc32.js 2.1091 +// 2.1092 +// Part of the JSIO library. This adds an CRC32-calculating 2.1093 +// ByteReader to JSIO. 2.1094 +// 2.1095 +// ======================================================= 2.1096 +// 2.1097 +// A ByteReader exposes an interface with these functions: 2.1098 +// 2.1099 +// readByte() 2.1100 +// must return null when EOF is reached. 2.1101 +// 2.1102 +// readToEnd() 2.1103 +// returns an array of all bytes read, to EOF 2.1104 +// 2.1105 +// beginReadToEnd(callback) 2.1106 +// async version of the above 2.1107 +// 2.1108 +// readBytes(n) 2.1109 +// returns an array of all n bytes read from the source 2.1110 +// 2.1111 +// beginReadBytes(n, callback) 2.1112 +// async version of the above 2.1113 +// 2.1114 +// ======================================================= 2.1115 +// 2.1116 +// Copyleft (c) 2010, Dino Chiesa via MS-PL 2.1117 +// Copyleft (c) 2012, Brendan Byrd via GPL 2.1118 +// 2.1119 +// This work is licensed under the GPLv3. 2.1120 + 2.1121 +(function(){ 2.1122 + var version = "2.0 2012Feb"; 2.1123 + var typename = "JSIO.Crc32"; 2.1124 + 2.1125 + if (typeof JSIO._ByteReaderBase !== "function") 2.1126 + JSIO.throwError('This extension requires JSIO.BasicByteReaders.js', typename); 2.1127 + 2.1128 + JSIO.crc32Table = null; 2.1129 + JSIO.crc32Polynomial = 0xEDB88320; 2.1130 + 2.1131 + var crc32TableCalc = function () { 2.1132 + // do this once only, for all instances 2.1133 + if (JSIO.crc32Table) return; 2.1134 + JSIO.crc32Table = new Array(256); 2.1135 + for (var i = 0; i < 256; i++) { 2.1136 + var c=i; 2.1137 + for (var k = 0; k < 8; k++) { 2.1138 + if ((c & 1) == 1) c = JSIO.crc32Polynomial ^ (c>>>1); 2.1139 + else c >>>= 1; 2.1140 + } 2.1141 + JSIO.crc32Table[i] = c; 2.1142 + } 2.1143 + }; 2.1144 + 2.1145 + JSIO.computeCrc32 = function(str) { 2.1146 + crc32TableCalc(); // once 2.1147 + var c = 0xFFFFFFFF; 2.1148 + var sL = str.length; 2.1149 + if (typeof str == "object") { 2.1150 + for (var n1=0; n1<sL; n1++) { 2.1151 + c = JSIO.crc32Table[(c&0xff) ^ str[n1]] ^ (c>>>8); 2.1152 + } 2.1153 + } else { 2.1154 + for (var n2=0; n2<sL; n2++) { 2.1155 + c = JSIO.crc32Table[(c&0xff) ^ str.charCodeAt(n2)] ^ (c>>>8); 2.1156 + } 2.1157 + } 2.1158 + c ^= 0xFFFFFFFF; 2.1159 + if (c < 0) c += 0xFFFFFFFF+1; 2.1160 + return c; 2.1161 + }; 2.1162 + 2.1163 + // ======================================================= 2.1164 + var _crc32 = function() { 2.1165 + if (! (this instanceof arguments.callee) ) this._throwError('You must use new to instantiate this class', typename, 'ctor'); 2.1166 + crc32TableCalc(); // once 2.1167 + this._typename = typename; 2.1168 + this._version = version; 2.1169 + this._runningCrc32 = 0xFFFFFFFF; 2.1170 + }; 2.1171 + 2.1172 + _crc32.prototype.slurpByte = function(b) { 2.1173 + var r = this._runningCrc32; 2.1174 + this._runningCrc32 = r>>>8 ^ JSIO.crc32Table[b ^ (r & 0x000000FF)]; 2.1175 + }; 2.1176 + 2.1177 + _crc32.prototype.result = function() { 2.1178 + var c = this._runningCrc32 ^ 0xFFFFFFFF; 2.1179 + if (c < 0) c += 0xFFFFFFFF+1; 2.1180 + return c; 2.1181 + }; 2.1182 + // ======================================================= 2.1183 + 2.1184 + 2.1185 + 2.1186 + var _crc32CalculatingReader = function(reader) { 2.1187 + if (! (this instanceof arguments.callee) ) this._throwError('You must use new to instantiate this class', 'JSIO.Crc32Reader', 'ctor'); 2.1188 + this._byteReader = reader; 2.1189 + this._typename = "JSIO.Crc32Reader"; 2.1190 + this._version = version; 2.1191 + this._crc32 = new JSIO.Crc32(); 2.1192 + }; 2.1193 + 2.1194 + _crc32CalculatingReader.prototype = new JSIO._ByteReaderBase(); 2.1195 + 2.1196 + _crc32CalculatingReader.prototype.readByte = function() { 2.1197 + var b = this._byteReader.readByte(); 2.1198 + if (b !== null) this._crc32.slurpByte(b); 2.1199 + this.position++; 2.1200 + return b; 2.1201 + }; 2.1202 + 2.1203 + _crc32CalculatingReader.prototype.read = function(len) { 2.1204 + if (len === 0) return []; 2.1205 + var bytes = this._byteReader.read(len); 2.1206 + len = bytes.length; 2.1207 + 2.1208 + var tbl = JSIO.crc32Table; 2.1209 + var r = this._crc32._runningCrc32; 2.1210 + var t; 2.1211 + for (var i = 0; i < len; i++) { 2.1212 + t = tbl[ bytes[i] ^ (r & 0x000000FF) ]; 2.1213 + r = (r >>> 8) ^ t; 2.1214 + } 2.1215 + this._crc32._runningCrc32 = r; 2.1216 + 2.1217 + this.position += len; 2.1218 + return bytes; 2.1219 + } 2.1220 + 2.1221 + _crc32CalculatingReader.prototype.crc32 = function() { 2.1222 + return this._crc32.result(); 2.1223 + }; 2.1224 + 2.1225 + JSIO.Crc32 = _crc32; 2.1226 + JSIO.Crc32Reader = _crc32CalculatingReader; 2.1227 + 2.1228 +})(); 2.1229 + 2.1230 +/// JSIO.CRC32.js ends 2.1231 +// JSIO.InflatingReader.js 2.1232 +// ------------------------------------------------------------------ 2.1233 +// 2.1234 +// Part of the JSIO library. This adds an Inflating ByteReader to 2.1235 +// JSIO. 2.1236 +// 2.1237 +// ======================================================= 2.1238 +// 2.1239 +// A ByteReader exposes an interface with these functions: 2.1240 +// 2.1241 +// readByte() 2.1242 +// must return null when EOF is reached. 2.1243 +// 2.1244 +// readToEnd() 2.1245 +// returns an array of all bytes read, to EOF 2.1246 +// 2.1247 +// beginReadToEnd(callback) 2.1248 +// async version of the above 2.1249 +// 2.1250 +// readBytes(n) 2.1251 +// returns an array of all n bytes read from the source 2.1252 +// 2.1253 +// beginReadBytes(n, callback) 2.1254 +// async version of the above 2.1255 +// 2.1256 +// ======================================================= 2.1257 +// 2.1258 +// Derived in part from work by notmasteryet. 2.1259 +// http://www.codeproject.com/KB/scripting/Javascript_binaryenc.aspx 2.1260 +// 2.1261 +// Copyleft (c) 2008, notmasteryet via an MIT-style license 2.1262 +// Copyleft (c) 2010, Dino Chiesa via MS-PL 2.1263 +// Copyleft (c) 2012, Brendan Byrd via GPL 2.1264 +// 2.1265 +// This work is licensed under the GPLv3. 2.1266 + 2.1267 + 2.1268 +(function(){ 2.1269 + var version = "2.0 2012Feb"; 2.1270 + var typename = "JSIO.InflatingReader"; 2.1271 + 2.1272 + if (typeof JSIO._ByteReaderBase !== "function") 2.1273 + JSIO.throwError('This extension requires JSIO.BasicByteReaders.js', typename); 2.1274 + 2.1275 + // ======================================================= 2.1276 + // _InternalBitReader is used internally in the InflatingReader class. 2.1277 + // 2.1278 + 2.1279 + JSIO.bitShiftTable = null; 2.1280 + 2.1281 + var bitShiftTableCalc = function () { 2.1282 + // do this once only, for all instances 2.1283 + if (JSIO.bitShiftTable) return; 2.1284 + 2.1285 + var bits = 8; 2.1286 + JSIO.bitShiftTable = { 2.1287 + LSB: new Array(bits), 2.1288 + MSB: new Array(bits) 2.1289 + } 2.1290 + for (var bp = 0; bp < bits; bp++) { 2.1291 + var lim = bits - bp; 2.1292 + JSIO.bitShiftTable.LSB[bp] = new Array(lim); 2.1293 + JSIO.bitShiftTable.MSB[bp] = new Array(lim); 2.1294 + 2.1295 + var maskLSB = 1 << bp; 2.1296 + var maskMSB = 1 << lim-1; 2.1297 + for (var len = 1; len <= lim; len++) { 2.1298 + JSIO.bitShiftTable.LSB[bp][len-1] = maskLSB; 2.1299 + JSIO.bitShiftTable.MSB[bp][len-1] = maskMSB; 2.1300 + maskLSB |= 1 << bp+len; 2.1301 + maskMSB |= 1 << lim-len-1; 2.1302 + } 2.1303 + } 2.1304 + }; 2.1305 + 2.1306 + var _InternalBitReader = function(reader) { 2.1307 + if (! (this instanceof arguments.callee) ) this._throwError('You must use new to instantiate this class', typename + '._InternalBitReader.ctor'); 2.1308 + this.bitsLength = 0; 2.1309 + this.bits = 0; 2.1310 + this.byteReader = reader; 2.1311 + this._typeName = typename + "._InternalBitReader"; 2.1312 + this._version = version; 2.1313 + bitShiftTableCalc(); 2.1314 + }; 2.1315 + 2.1316 + _InternalBitReader.prototype._throwError = JSIO.throwError; 2.1317 + 2.1318 + _InternalBitReader.prototype.readBit = function() { 2.1319 + if (this.bitsLength === 0) { 2.1320 + var nextByte = this.byteReader.readByte(); 2.1321 + if (nextByte === null) this._throwError('Unexpected end of stream', null, 'readBit'); 2.1322 + this.bits = nextByte; 2.1323 + this.bitsLength = 8; 2.1324 + } 2.1325 + 2.1326 + var bit = (this.bits & 1 << 8-this.bitsLength) !== 0; 2.1327 + this.bitsLength--; 2.1328 + return bit; 2.1329 + }; 2.1330 + 2.1331 + _InternalBitReader.prototype.align = function() { this.bitsLength = 0; }; 2.1332 + 2.1333 + _InternalBitReader.prototype.readBits = function(len, type) { 2.1334 + var data = 0; 2.1335 + type = type || 'LSB'; 2.1336 + var tbl = JSIO.bitShiftTable[type]; 2.1337 + var dl = 0; 2.1338 + while (len > 0) { 2.1339 + if (this.bitsLength === 0) { 2.1340 + var nextByte = this.byteReader.readByte(); 2.1341 + if (nextByte === null) this._throwError('Unexpected end of stream', null, 'read'+type); 2.1342 + this.bits = nextByte; 2.1343 + this.bitsLength = 8; 2.1344 + } 2.1345 + 2.1346 + var maskLen = (this.bitsLength >= len) ? len : this.bitsLength; 2.1347 + var bitsPos = 8-this.bitsLength; 2.1348 + data |= (this.bits & tbl[bitsPos][maskLen-1]) >>> bitsPos << dl; 2.1349 + 2.1350 + dl += maskLen; 2.1351 + len -= maskLen; 2.1352 + this.bitsLength -= maskLen; 2.1353 + }; 2.1354 + return data; 2.1355 + }; 2.1356 + 2.1357 + _InternalBitReader.prototype.readLSB = function(len) { return this.readBits(len, 'LSB'); } 2.1358 + _InternalBitReader.prototype.readMSB = function(len) { return this.readBits(len, 'MSB'); } 2.1359 + 2.1360 + // 2.1361 + // ======================================================= 2.1362 + 2.1363 + 2.1364 + /* inflating ByteReader - RFC 1951 */ 2.1365 + var _inflatingReader = function(reader) { 2.1366 + if (! (this instanceof arguments.callee) ) this._throwError('You must use new to instantiate this class', typename, 'ctor'); 2.1367 + this._byteReader = reader; 2.1368 + this._bitReader = new _InternalBitReader(reader); 2.1369 + this._buffer = []; 2.1370 + this._bufferPosition = 0; 2.1371 + this._state = 0; 2.1372 + this._blockFinal = false; 2.1373 + this._typeName = typename; 2.1374 + this._version = version; 2.1375 + return this; 2.1376 + }; 2.1377 + 2.1378 + 2.1379 + // shared fns and variables 2.1380 + 2.1381 + var staticCodes = null; 2.1382 + var staticDistances = null; 2.1383 + 2.1384 + var clenMap = [16, 17, 18, 0, 8, 7, 9, 6, 10, 5, 11, 4, 12, 3, 13, 2, 14, 1, 15]; 2.1385 + 2.1386 + var buildCodes = function(lengths){ 2.1387 + var i=0; 2.1388 + var codes = new Array(lengths.length); 2.1389 + var maxBits = lengths[0]; 2.1390 + for (i=1; i<lengths.length; i++) { 2.1391 + if (maxBits < lengths[i]) maxBits = lengths[i]; 2.1392 + } 2.1393 + 2.1394 + var bitLengthsCount = new Array(maxBits + 1); 2.1395 + for (i=0; i<=maxBits; i++) bitLengthsCount[i]=0; 2.1396 + 2.1397 + for (i=0; i<lengths.length; i++) { 2.1398 + ++bitLengthsCount[lengths[i]]; 2.1399 + } 2.1400 + 2.1401 + var nextCode = new Array(maxBits + 1); 2.1402 + var code = 0; 2.1403 + bitLengthsCount[0] = 0; 2.1404 + for (var bits=1; bits<=maxBits; bits++) { 2.1405 + code = (code + bitLengthsCount[bits - 1]) << 1; 2.1406 + nextCode[bits] = code; 2.1407 + } 2.1408 + 2.1409 + for (i=0; i<codes.length; i++) { 2.1410 + var len = lengths[i]; 2.1411 + if (len !== 0) { 2.1412 + codes[i] = nextCode[len]; 2.1413 + nextCode[len]++; 2.1414 + } 2.1415 + } 2.1416 + return codes; 2.1417 + }; 2.1418 + 2.1419 + var buildTree = function(codes, lengths){ 2.1420 + var nonEmptyCodes = []; 2.1421 + for(var i=0; i<codes.length; ++i) { 2.1422 + if(lengths[i] > 0) { 2.1423 + var code = {}; 2.1424 + code.bits = codes[i]; 2.1425 + code.length = lengths[i]; 2.1426 + code.index = i; 2.1427 + nonEmptyCodes.push(code); 2.1428 + } 2.1429 + } 2.1430 + return buildTreeBranch(nonEmptyCodes, 0, 0); 2.1431 + }; 2.1432 + 2.1433 + 2.1434 + var buildTreeBranch = function(codes, prefix, prefixLength){ 2.1435 + if (codes.length === 0) return null; 2.1436 + 2.1437 + var zeros = []; 2.1438 + var ones = []; 2.1439 + var branch = {}; 2.1440 + branch.isLeaf = false; 2.1441 + for(var i=0; i<codes.length; ++i) { 2.1442 + if (codes[i].length == prefixLength && codes[i].bits == prefix) { 2.1443 + branch.isLeaf = true; 2.1444 + branch.index = codes[i].index; 2.1445 + break; 2.1446 + } 2.1447 + else { 2.1448 + var nextBit = ((codes[i].bits >> (codes[i].length - prefixLength - 1)) & 1) > 0; 2.1449 + if (nextBit) ones.push(codes[i]); 2.1450 + else zeros.push(codes[i]); 2.1451 + } 2.1452 + } 2.1453 + if(!branch.isLeaf) { 2.1454 + branch.zero = buildTreeBranch(zeros, (prefix << 1), prefixLength + 1); 2.1455 + branch.one = buildTreeBranch(ones, (prefix << 1) | 1, prefixLength + 1); 2.1456 + } 2.1457 + return branch; 2.1458 + }; 2.1459 + 2.1460 + 2.1461 + var encodedLengthStart = [3,4,5,6,7,8,9,10, 2.1462 + 11,13,15,17,19,23,27,31,35,43,51,59,67,83,99, 2.1463 + 115,131,163,195,227,258]; 2.1464 + 2.1465 + var encodedLengthAdditionalBits = [0,0,0,0,0,0,0,0, 2.1466 + 1,1,1,1,2,2,2,2,3,3,3,3,4,4,4,4,5,5,5,5,0]; 2.1467 + 2.1468 + var encodedDistanceStart = [1,2,3,4, 5,7,9, 13,17,25, 33,49,65, 2.1469 + 97,129,193,257,385,513,769,1025,1537,2049, 2.1470 + 3073,4097,6145,8193,12289,16385,24577]; 2.1471 + 2.1472 + var encodedDistanceAdditionalBits = [0,0,0,0,1,1,2,2,3,3,4,4, 2.1473 + 5,5,6,6,7,7,8,8,9,9,10,10,11,11,12,12,13,13]; 2.1474 + 2.1475 + 2.1476 + var readDynamicTrees = function(bitReader){ 2.1477 + var hlit = bitReader.readLSB(5) + 257; 2.1478 + var hdist = bitReader.readLSB(5) + 1; 2.1479 + var hclen = bitReader.readLSB(4) + 4; 2.1480 + var clen = new Array(19); 2.1481 + var i; 2.1482 + for (i=0; i<clen.length; i++) { clen[i] = 0; } 2.1483 + for (i=0; i<hclen; i++) { clen[clenMap[i]] = bitReader.readLSB(3); } 2.1484 + 2.1485 + var clenCodes = buildCodes(clen); 2.1486 + var clenTree = buildTree(clenCodes, clen); 2.1487 + 2.1488 + var lengthsSequence = []; 2.1489 + while(lengthsSequence.length < hlit + hdist) { 2.1490 + var p = clenTree; 2.1491 + while(!p.isLeaf) { p = bitReader.readBit() ? p.one : p.zero; } 2.1492 + 2.1493 + var code = p.index; 2.1494 + if (code <= 15) lengthsSequence.push(code); 2.1495 + else if (code == 16) { 2.1496 + // TODO: replace with faster repeat algorythm 2.1497 + var repeat = bitReader.readLSB(2) + 3; 2.1498 + for(var q=0; q<repeat; ++q){ 2.1499 + lengthsSequence.push(lengthsSequence[lengthsSequence.length - 1]); 2.1500 + } 2.1501 + } 2.1502 + else if (code == 17) { 2.1503 + var repeat1 = bitReader.readLSB(3) + 3; 2.1504 + for(var q1=0; q1<repeat1; ++q1) { 2.1505 + lengthsSequence.push(0); 2.1506 + } 2.1507 + } 2.1508 + else if (code == 18) { 2.1509 + var repeat2 = bitReader.readLSB(7) + 11; 2.1510 + for(var q2=0; q2<repeat2; ++q2){ 2.1511 + lengthsSequence.push(0); 2.1512 + } 2.1513 + } 2.1514 + } 2.1515 + 2.1516 + var codesLengths = lengthsSequence.slice(0, hlit); 2.1517 + var codes = buildCodes(codesLengths); 2.1518 + var distancesLengths = lengthsSequence.slice(hlit, hlit + hdist); 2.1519 + var distances = buildCodes(distancesLengths); 2.1520 + 2.1521 + return { 2.1522 + codesTree : buildTree(codes, codesLengths), 2.1523 + distancesTree : buildTree(distances, distancesLengths) 2.1524 + }; 2.1525 + }; 2.1526 + 2.1527 + 2.1528 + _inflatingReader.prototype = new JSIO._ByteReaderBase(); 2.1529 + 2.1530 + 2.1531 + // internal instance fns 2.1532 + _inflatingReader.prototype._decodeItem = function() { 2.1533 + if (this._state == 2) return null; // end-of-blocks 2.1534 + 2.1535 + var item; 2.1536 + if(this._state === 0) { 2.1537 + this._blockFinal = this._bitReader.readBit(); 2.1538 + var blockType = this._bitReader.readLSB(2); 2.1539 + switch(blockType) { 2.1540 + case 0: 2.1541 + this._bitReader.align(); 2.1542 + var len = this._bitReader.readLSB(16); // low-byte first, as opposed to readNumber's HBF 2.1543 + var nlen = this._bitReader.readLSB(16); 2.1544 + if ((len & ~nlen) != len) this._throwError('Invalid block type 0 length', null, '_decodeItem'); 2.1545 + 2.1546 + item = {}; 2.1547 + item.itemType = 0; 2.1548 + item.array = this._bitReader.byteReader.read(len); 2.1549 + if (item.array.length < len) this._throwError('Incomplete block', null, '_decodeItem'); 2.1550 + if (this._blockFinal) this._state = 2; 2.1551 + return item; 2.1552 + case 1: 2.1553 + this._codesTree = staticCodes; 2.1554 + this._distancesTree = staticDistances; 2.1555 + this._state = 1; 2.1556 + break; 2.1557 + case 2: 2.1558 + var dTrees = readDynamicTrees(this._bitReader); 2.1559 + this._codesTree = dTrees.codesTree; 2.1560 + this._distancesTree = dTrees.distancesTree; 2.1561 + this._state = 1; 2.1562 + break; 2.1563 + default: 2.1564 + this._throwError('Invalid block type ('+ blockType +')', null, '_decodeItem'); 2.1565 + } 2.1566 + } 2.1567 + 2.1568 + item = {}; 2.1569 + 2.1570 + var p = this._codesTree; 2.1571 + while (!p.isLeaf) { p = this._bitReader.readBit() ? p.one : p.zero; } 2.1572 + if(p.index < 256) { 2.1573 + item.itemType = 2; 2.1574 + item.symbol = p.index; 2.1575 + } else if(p.index > 256) { 2.1576 + var lengthCode = p.index; 2.1577 + if(lengthCode > 285) this._throwError('Invalid length code', null, '_decodeItem'); 2.1578 + 2.1579 + var length = encodedLengthStart[lengthCode - 257]; 2.1580 + if(encodedLengthAdditionalBits[lengthCode - 257] > 0) { 2.1581 + length += this._bitReader.readLSB(encodedLengthAdditionalBits[lengthCode - 257]); 2.1582 + } 2.1583 + 2.1584 + p = this._distancesTree; 2.1585 + while (!p.isLeaf) { p = this._bitReader.readBit() ? p.one : p.zero; } 2.1586 + 2.1587 + var distanceCode = p.index; 2.1588 + var distance = encodedDistanceStart[distanceCode]; 2.1589 + if (encodedDistanceAdditionalBits[distanceCode] > 0) 2.1590 + distance += this._bitReader.readLSB(encodedDistanceAdditionalBits[distanceCode]); 2.1591 + 2.1592 + item.itemType = 3; 2.1593 + item.distance = distance; 2.1594 + item.length = length; 2.1595 + } else { 2.1596 + item.itemType = 1; 2.1597 + this._state = this._blockFinal ? 2 : 0; // EOB 2.1598 + } 2.1599 + return item; 2.1600 + }; 2.1601 + 2.1602 + 2.1603 + 2.1604 + // public instance functions 2.1605 + 2.1606 + _inflatingReader.prototype.readByte = function() { 2.1607 + var byte = this.read(1)[0]; 2.1608 + return (byte === null || byte === undefined) ? null : byte; 2.1609 + }; 2.1610 + 2.1611 + _inflatingReader.prototype.read = function(len) { 2.1612 + var b = this._buffer; // (since we use this so much...) 2.1613 + 2.1614 + // Keep reading until we get to the right length 2.1615 + while (this._bufferPosition+len > b.length) { 2.1616 + var item = this._decodeItem(); 2.1617 + if (item === null) { // EOF 2.1618 + len = b.length - this._bufferPosition; 2.1619 + break; 2.1620 + } 2.1621 + switch(item.itemType) { 2.1622 + case 0: 2.1623 + JSIO.massApply(b.push, b, item.array); 2.1624 + break; 2.1625 + case 2: 2.1626 + b.push(item.symbol); 2.1627 + break; 2.1628 + case 3: 2.1629 + var j = b.length - item.distance; 2.1630 + if (item.distance >= item.length) 2.1631 + JSIO.massApply(b.push, b, b.slice(j, j+item.length)); 2.1632 + // sometimes DEFLATE tries some trickery with "look-ahead" compression 2.1633 + else { 2.1634 + // this is basically just a repetition of the same string, plus some possible cutoff 2.1635 + var count = parseInt(item.length / item.distance); 2.1636 + var repArr = b.slice(j); 2.1637 + // http://stackoverflow.com/questions/202605/repeat-string-javascript/5450113#5450113 2.1638 + while (count > 0) { 2.1639 + if (count & 1) JSIO.massApply( b.push, b, repArr); 2.1640 + if (count >>= 1) JSIO.massApply(repArr.push, repArr, repArr); 2.1641 + } 2.1642 + // add any remaining cutoff 2.1643 + var r; 2.1644 + if (r = item.length % item.distance) 2.1645 + JSIO.massApply(b.push, b, b.slice(j, j+r)); 2.1646 + } 2.1647 + break; 2.1648 + } 2.1649 + } 2.1650 + var bytes = b.slice(this._bufferPosition, this._bufferPosition+len); 2.1651 + this._bufferPosition += len; 2.1652 + this.position += len; 2.1653 + 2.1654 + if (this._bufferPosition > 0xC000) { 2.1655 + var shift = b.length - 0x8000; 2.1656 + if (shift > this._bufferPosition) shift = this._bufferPosition; 2.1657 + b.splice(0, shift); 2.1658 + this._bufferPosition -= shift; 2.1659 + } 2.1660 + 2.1661 + return bytes; 2.1662 + }; 2.1663 + 2.1664 + // initialization routine - once per type 2.1665 + (function(){ 2.1666 + 2.1667 + var codes = new Array(288); 2.1668 + var codesLengths = new Array(288); 2.1669 + var i=0; 2.1670 + for ( i = 0; i <= 143; i++) { 2.1671 + codes[i] = 0x0030 + i; 2.1672 + codesLengths[i] = 8; 2.1673 + } 2.1674 + for ( i = 144; i <= 255; i++) { 2.1675 + codes[i] = 0x0190 + i - 144; 2.1676 + codesLengths[i] = 9; 2.1677 + } 2.1678 + for ( i = 256; i <= 279; i++) { 2.1679 + codes[i] = 0x0000 + i - 256; 2.1680 + codesLengths[i] = 7; 2.1681 + } 2.1682 + for ( i = 280; i <= 287; i++) { 2.1683 + codes[i] = 0x00C0 + i - 280; 2.1684 + codesLengths[i] = 8; 2.1685 + } 2.1686 + staticCodes = buildTree(codes, codesLengths); 2.1687 + 2.1688 + var distances = new Array(32); 2.1689 + var distancesLengths = new Array(32); 2.1690 + for ( i = 0; i <= 31; i++) { 2.1691 + distances[i] = i; 2.1692 + distancesLengths[i] = 5; 2.1693 + } 2.1694 + staticDistances = buildTree(distances, distancesLengths); 2.1695 + })(); 2.1696 + 2.1697 + 2.1698 + JSIO.InflatingReader = _inflatingReader; 2.1699 + 2.1700 +})(); 2.1701 + 2.1702 + 2.1703 +/// JSIO.InflatingReader.js ends 2.1704 + 2.1705 +// Zipfile.js 2.1706 +// ------------------------------------------------------------------ 2.1707 +// 2.1708 +// A class that reads Zip files. 2.1709 +// Depends on the JSIO library functions. 2.1710 +// 2.1711 +// ======================================================= 2.1712 +// 2.1713 +// Copyleft (c) 2010, Dino Chiesa via MS-PL 2.1714 +// Copyleft (c) 2012, Brendan Byrd via GPL 2.1715 +// 2.1716 +// This work is licensed under the GPLv3. 2.1717 + 2.1718 +(function(){ 2.1719 + var version = "2.0 2012Feb"; 2.1720 + var typename = "Zipfile"; 2.1721 + 2.1722 + if (typeof JSIO.BinaryUrlStream !== "function") JSIO.throwError('This extension requires JSIO.BinaryUrlStream.js v2.0', typename); 2.1723 + if (typeof JSIO.TextDecoder !== "object") JSIO.throwError('This extension requires JSIO.TextDecoder.js v2.0', typename); 2.1724 + if (typeof JSIO.TextReader !== "function") JSIO.throwError('This extension requires JSIO.TextReader.js v2.0', typename); 2.1725 + if (typeof JSIO.Crc32 !== "function") JSIO.throwError('This extension requires JSIO.Crc32.js v2.0', typename); 2.1726 + if (typeof JSIO.InflatingReader !== "function") JSIO.throwError('This extension requires JSIO.InflatingReader.js v2.0', typename); 2.1727 + 2.1728 + // ======================================================= 2.1729 + function ZipEntry(zip) { 2.1730 + this.zipfile = zip; 2.1731 + this._typename = "ZipEntry"; 2.1732 + this._version = version; 2.1733 + this._crcCalculator = null; 2.1734 + } 2.1735 + 2.1736 + ZipEntry.prototype._throwError = JSIO.throwError; 2.1737 + 2.1738 + // return byte array or string 2.1739 + ZipEntry.prototype.extract = function(callback, asString) { 2.1740 + this.contentType = JSIO.guessFileType(this.name); 2.1741 + asString = asString || ( this.contentType == JSIO.FileType.Text || 2.1742 + this.contentType == JSIO.FileType.XML); 2.1743 + var thisEntry = this; 2.1744 + 2.1745 + if (this.compressionMethod !== 0 && this.compressionMethod != 8) 2.1746 + this._throwError('Unsupported compression method: ' + this.compressionMethod, null, 'extract'); 2.1747 + 2.1748 + var reader = (asString) ? this.openTextReader(thisEntry.utf8 ? JSIO.TextDecoder.UTF8 : JSIO.TextDecoder.ANSI) : this.openBinaryReader(); 2.1749 + 2.1750 + // diagnostic purpose only; tag the reader with the entry name 2.1751 + reader.zipEntryName = thisEntry.name; 2.1752 + 2.1753 + if (typeof callback != "function") { 2.1754 + // synchronous 2.1755 + var result = reader.readToEnd(); 2.1756 + this.verifyCrc32(); 2.1757 + return result; 2.1758 + } 2.1759 + 2.1760 + // asynchronous 2.1761 + reader.beginReadToEnd(function(result){ 2.1762 + try { 2.1763 + thisEntry.verifyCrc32(); 2.1764 + callback(thisEntry, result); 2.1765 + } 2.1766 + catch (exc1) { 2.1767 + callback(thisEntry, exc1); 2.1768 + } 2.1769 + }); 2.1770 + return null; 2.1771 + }; 2.1772 + 2.1773 + 2.1774 + // open a ByteReader on the entry, which will read binary 2.1775 + // content from the compressed stream. 2.1776 + ZipEntry.prototype.openBinaryReader = function() { 2.1777 + var reader = 2.1778 + new JSIO.StreamSegmentReader(this.zipfile.binaryStream, 2.1779 + this.offset + this.lengthOfHeader, 2.1780 + this.compressedSize); 2.1781 + if (this.compressionMethod === 0) { 2.1782 + this._crcCalculator = new JSIO.Crc32Reader(reader); 2.1783 + } 2.1784 + else { 2.1785 + var inflator = new JSIO.InflatingReader(reader); 2.1786 + this._crcCalculator = new JSIO.Crc32Reader(inflator); 2.1787 + } 2.1788 + // Whether compressed or not, the source ByteReader in each case 2.1789 + // is wrapped in a second ByteReader object that calculates CRC 2.1790 + // as it reads. That way, after all reading is complete, the 2.1791 + // caller can check the calcuated CRC against the expected CRC. 2.1792 + return this._crcCalculator; 2.1793 + }; 2.1794 + 2.1795 + // open a TextReader on the entry, to read text from the 2.1796 + // compressed stream. 2.1797 + ZipEntry.prototype.openTextReader = function(decoderKind) { 2.1798 + var reader = this.openBinaryReader(); 2.1799 + decoderKind = decoderKind || JSIO.TextDecoder.UTF8; 2.1800 + var d = new decoderKind(reader); 2.1801 + var textReader = new JSIO.TextReader(d); 2.1802 + d._parent = textReader; // store a reference, for diagnostic purposes only 2.1803 + return textReader; 2.1804 + }; 2.1805 + 2.1806 + // verify the CRC on the entry. 2.1807 + // call this after all bytes have been read. 2.1808 + ZipEntry.prototype.verifyCrc32 = function() { 2.1809 + var computedCrc = this._crcCalculator.crc32(); 2.1810 + var rc = false; // CRC FAIL 2.1811 + if (this.crc32 != computedCrc) { 2.1812 + var msg = "WARNING: CRC check failed: " + 2.1813 + "entry(" + this.name + ") " + 2.1814 + "computed(" + JSIO.decimalToHexString(computedCrc,8) + ") " + 2.1815 + "expected(" + JSIO.decimalToHexString(this.crc32,8) + ") "; 2.1816 + this.zipfile.status.push(msg); 2.1817 + } else { 2.1818 + rc = true; // OK 2.1819 + if (this.zipfile.verbose>2) { 2.1820 + this.zipfile.status.push("INFO: CRC check ok: 0x" + 2.1821 + JSIO.decimalToHexString(this.crc32,8)); 2.1822 + } 2.1823 + } 2.1824 + return rc; 2.1825 + }; 2.1826 + 2.1827 + 2.1828 + // ctor 2.1829 + ZipFile = function(fileUrl, callback, verbosity) { 2.1830 + if (! (this instanceof arguments.callee) ) JSIO.throwError('You must use new to instantiate this class', typename, 'ctor'); 2.1831 + 2.1832 + this.verbose = verbosity || 0; 2.1833 + this.entries = []; 2.1834 + this.entryNames = []; 2.1835 + this.status = []; 2.1836 + this._version = version; 2.1837 + this._typename = "ZipFile"; 2.1838 + this._throwError = JSIO.throwError; 2.1839 + 2.1840 + var thisZipFile = this; 2.1841 + 2.1842 + // Could use a back-tracking reader for the central directory, but 2.1843 + // there's no point, since all the zip data is held in memory anyway. 2.1844 + 2.1845 + /* function ReadCentralDirectory(){ 2.1846 + var posn = thisZipFile.binaryStream.length - 64; 2.1847 + var maxSeekback = Math.Max(s.Length - 0x4000, 10); 2.1848 + var success = false; 2.1849 + var nTries = 0; 2.1850 + do 2.1851 + { 2.1852 + thisZipFile.binaryStream.Seek(posn, SeekOrigin.Begin); 2.1853 + var bytesRead = thisZipFile.binaryStream.findSignature(thisZipFile.Signatures.EndOfCentralDirectory); 2.1854 + if (bytesRead != -1) 2.1855 + success = true; 2.1856 + else 2.1857 + { 2.1858 + nTries++; 2.1859 + // increasingly larger 2.1860 + posn -= (32 * (nTries + 1) * nTries); 2.1861 + if (posn < 0) posn = 0; // BOF 2.1862 + } 2.1863 + } 2.1864 + while (!success && posn > maxSeekback); 2.1865 + if (!success) { 2.1866 + thisZipFile.status.push("cannot find End of Central Directory"); 2.1867 + return; 2.1868 + } 2.1869 + } */ 2.1870 + 2.1871 + 2.1872 + function DateFromPackedFormat(packed) { 2.1873 + if (packed == 0xFFFF || packed === 0) return new Date(1995, 0, 1, 0,0,0,0); 2.1874 + 2.1875 + var packedTime = packed & 0x0000ffff; 2.1876 + var packedDate = ((packed & 0xffff0000) >> 16); 2.1877 + 2.1878 + var year = 1980 + ((packedDate & 0xFE00) >> 9); 2.1879 + var month = ((packedDate & 0x01E0) >> 5) -1; 2.1880 + var day = packedDate & 0x001F; 2.1881 + 2.1882 + var hour = (packedTime & 0xF800) >> 11; 2.1883 + var minute = (packedTime & 0x07E0) >> 5; 2.1884 + var second = (packedTime & 0x001F) * 2; 2.1885 + 2.1886 + // Validation and error checking. 2.1887 + // This is not foolproof but will catch most errors. 2.1888 + 2.1889 + // I can't believe how many different ways applications 2.1890 + // can mess up a simple date format. 2.1891 + 2.1892 + if (second >= 60) { minute++; second = 0; } 2.1893 + if (minute >= 60) { hour++; minute = 0; } 2.1894 + if (hour >= 24) { day++; hour = 0; } 2.1895 + var success = false; 2.1896 + var d; 2.1897 + try { 2.1898 + d = new Date(year, month, day, hour, minute, second, 0); 2.1899 + success= true; 2.1900 + } 2.1901 + catch (exc1) { 2.1902 + if (year == 1980 && (month === 0 || day === 0)) { 2.1903 + try { 2.1904 + d = new Date(1980, 0, 1, hour, minute, second, 0); 2.1905 + success= true; 2.1906 + } 2.1907 + catch (exc2) { 2.1908 + try { 2.1909 + d = new Date(1980, 0, 1, 0, 0, 0, 0); 2.1910 + success= true; 2.1911 + } 2.1912 + catch (exc3) { } // how could this fail?? 2.1913 + } 2.1914 + } 2.1915 + else { 2.1916 + try { 2.1917 + if (year < 1980) year = 1980; 2.1918 + if (year > 2030) year = 2030; 2.1919 + if (month < 1) month = 1; 2.1920 + if (month > 12) month = 12; 2.1921 + if (day < 1) day = 1; 2.1922 + if (day > 31) day = 31; 2.1923 + if (minute < 0) minute = 0; 2.1924 + if (minute > 59) minute = 59; 2.1925 + if (second < 0) second = 0; 2.1926 + if (second > 59) second = 59; 2.1927 + d = new Date(year, month-1, day, hour, minute, second, 0); 2.1928 + success= true; 2.1929 + } 2.1930 + catch (exc4){} 2.1931 + } 2.1932 + } 2.1933 + if (!success) this._throwError('Bad date/time value in this ZIP file', null, 'DateFromPackedFormat'); 2.1934 + return d; 2.1935 + } 2.1936 + 2.1937 + 2.1938 + function ReadZipEntries () { 2.1939 + // read only once 2.1940 + if (thisZipFile.entryNames.length === 0){ 2.1941 + var e; 2.1942 + while ((e = ReadZipEntry()) !== null) { 2.1943 + thisZipFile.entries.push(e); 2.1944 + thisZipFile.entryNames.push(e.name); 2.1945 + } 2.1946 + } 2.1947 + } 2.1948 + 2.1949 + 2.1950 + function ReadZipEntry () { 2.1951 + var offset = thisZipFile.binaryStream.position; 2.1952 + var sig = thisZipFile.binaryStream.readNumber(4); 2.1953 + if (sig == ZipFile.Signatures.DirEntry) { 2.1954 + // after all entries, comes the central directory 2.1955 + if (thisZipFile.verbose > 0) { 2.1956 + thisZipFile.status.push("INFO: at offset 0x" + 2.1957 + JSIO.decimalToHexString(offset) + 2.1958 + ", found start of Zip Directory."); 2.1959 + } 2.1960 + // all done reading 2.1961 + return null; 2.1962 + } 2.1963 + if (sig != ZipFile.Signatures.Entry) { 2.1964 + thisZipFile.status.push("WARNING: at offset 0x" + 2.1965 + JSIO.decimalToHexString(offset) + 2.1966 + ", found unexpected signature: 0x" + 2.1967 + JSIO.decimalToHexString(sig)); 2.1968 + return null; 2.1969 + } 2.1970 + 2.1971 + var entry = new ZipEntry(thisZipFile); 2.1972 + entry.offset = offset; 2.1973 + entry.versionNeeded = thisZipFile.binaryStream.readNumber(2); 2.1974 + entry.bitField = thisZipFile.binaryStream.readNumber(2); 2.1975 + entry.compressionMethod = thisZipFile.binaryStream.readNumber(2); 2.1976 + var timeBlob = thisZipFile.binaryStream.readNumber(4); 2.1977 + entry.lastModified = DateFromPackedFormat(timeBlob); 2.1978 + entry.crc32 = thisZipFile.binaryStream.readNumber(4); 2.1979 + entry.compressedSize = thisZipFile.binaryStream.readNumber(4); 2.1980 + entry.uncompressedSize = thisZipFile.binaryStream.readNumber(4); 2.1981 + 2.1982 + if ((entry.bitField & 0x01) == 0x01){ 2.1983 + thisZipFile.status.push("This zipfile uses Encryption, which is not supported by ZipFile.js."); 2.1984 + return null; 2.1985 + } 2.1986 + 2.1987 + entry.utf8 = ((entry.bitField & 0x0800) == 0x0800); 2.1988 + 2.1989 + if ((entry.bitField & 0x0008) == 0x0008){ 2.1990 + thisZipFile.status.push("This zipfile uses a bit 3 trailing data descriptor, which is not supported by ZipFile.js."); 2.1991 + return null; 2.1992 + } 2.1993 + 2.1994 + if (entry.compressedSize == 0xFFFFFFFF || 2.1995 + entry.uncompressedSize == 0xFFFFFFFF) { 2.1996 + thisZipFile.status.push("This zipfile uses ZIP64, which is not supported by ZipFile.js"); 2.1997 + return null; 2.1998 + } 2.1999 + 2.2000 + var filenameLength = thisZipFile.binaryStream.readNumber(2); 2.2001 + var extraFieldLength = thisZipFile.binaryStream.readNumber(2); 2.2002 + 2.2003 + thisZipFile.status.push("INFO: filename length= " + filenameLength); 2.2004 + 2.2005 + // we've read 30 bytes of metadata so far 2.2006 + var bytesRead = 30 + filenameLength + extraFieldLength; 2.2007 + 2.2008 + if (entry.utf8) { 2.2009 + thisZipFile.status.push("INFO: before filename, position= 0x" + 2.2010 + JSIO.decimalToHexString( thisZipFile.binaryStream.position )); 2.2011 + var binReader = 2.2012 + new JSIO.StreamSegmentReader(thisZipFile.binaryStream, 2.2013 + thisZipFile.binaryStream.position, 2.2014 + filenameLength); 2.2015 + var utf8Decoder = new JSIO.TextDecoder.UTF8(binReader); 2.2016 + var textReader = new JSIO.TextReader(utf8Decoder); 2.2017 + entry.name = textReader.readToEnd(); 2.2018 + 2.2019 + // advance the filepointer: 2.2020 + thisZipFile.binaryStream.seek(filenameLength, 2.2021 + JSIO.SeekOrigin.Current, 2.2022 + thisZipFile); 2.2023 + 2.2024 + thisZipFile.status.push("INFO: after filename, position= 0x" + 2.2025 + JSIO.decimalToHexString( thisZipFile.binaryStream.position )); 2.2026 + } 2.2027 + else { 2.2028 + entry.name = thisZipFile.binaryStream.readString(filenameLength); 2.2029 + } 2.2030 + 2.2031 + // There are a bunch of things in the "extra" header, thisZipFile we 2.2032 + // could parse, like timestamps and other things. This class 2.2033 + // only identifies and separates them. 2.2034 + 2.2035 + // More info here: http://www.pkware.com/documents/casestudies/APPNOTE.TXT 2.2036 + 2.2037 + var extraPos = 0; 2.2038 + entry.extra = []; 2.2039 + while (extraPos < extraFieldLength) { 2.2040 + var extraBlock = { 2.2041 + type: thisZipFile.binaryStream.readNumber(2), 2.2042 + size: thisZipFile.binaryStream.readNumber(2) 2.2043 + }; 2.2044 + extraBlock.typeDescription = ZipFile.ExtraFieldTypes[extraBlock.type]; 2.2045 + extraBlock.data = thisZipFile.binaryStream.read(extraBlock.size); 2.2046 + entry.extra.push(extraBlock); 2.2047 + extraPos += 4 + extraBlock.size; 2.2048 + } 2.2049 + 2.2050 + if (thisZipFile.verbose > 1) { 2.2051 + thisZipFile.status.push("INFO: at offset 0x" + 2.2052 + JSIO.decimalToHexString(entry.offset) + 2.2053 + ", found entry '" + entry.name + "' fnl(" + 2.2054 + filenameLength + ") efl(" + 2.2055 + extraFieldLength +")"); 2.2056 + } 2.2057 + 2.2058 + if (extraFieldLength > 0) { 2.2059 + if (thisZipFile.verbose > 0) { 2.2060 + thisZipFile.status.push("INFO: entry " + entry.name + " has " + 2.2061 + extraFieldLength + " bytes of " + 2.2062 + "extra metadata (ID'd but ignored)"); 2.2063 + } 2.2064 + } 2.2065 + 2.2066 + entry.lengthOfHeader = bytesRead; 2.2067 + entry.totalEntrySize = entry.lengthOfHeader + entry.compressedSize; 2.2068 + 2.2069 + // seek past the data without reading it. We will read on Extract() 2.2070 + if (thisZipFile.verbose > 1) { 2.2071 + thisZipFile.status.push("INFO: seek 0x" + 2.2072 + JSIO.decimalToHexString(entry.compressedSize) + 2.2073 + " (" + entry.compressedSize + ") bytes"); 2.2074 + } 2.2075 + 2.2076 + thisZipFile.binaryStream.seek(entry.compressedSize, 2.2077 + JSIO.SeekOrigin.Current, 2.2078 + thisZipFile); 2.2079 + 2.2080 + return entry; 2.2081 + } 2.2082 + 2.2083 + 2.2084 + var parseZipFile = function(bfr){ 2.2085 + try { 2.2086 + if (bfr.req.status == 200) { 2.2087 + var sig = thisZipFile.binaryStream.readNumber(4); 2.2088 + if (sig != ZipFile.Signatures.Entry){ 2.2089 + thisZipFile.status.push("WARNING: this file does not appear to be a zip file"); 2.2090 + } else { 2.2091 + thisZipFile.binaryStream.seek(0, JSIO.SeekOrigin.Begin); 2.2092 + ReadZipEntries(); 2.2093 + if (thisZipFile.verbose > 0) { 2.2094 + thisZipFile.status.push("INFO: read " + thisZipFile.entries.length + " entries"); 2.2095 + } 2.2096 + } 2.2097 + } 2.2098 + else { 2.2099 + thisZipFile.status.push("ERROR: the URL could not be read (" + 2.2100 + bfr.req.status + " " + bfr.req.statusText + ")"); 2.2101 + } 2.2102 + callback(thisZipFile); 2.2103 + } 2.2104 + catch (exc1) 2.2105 + { 2.2106 + thisZipFile.status.push("Exception: " + exc1.message); 2.2107 + callback(thisZipFile); 2.2108 + } 2.2109 + }; 2.2110 + 2.2111 + this.binaryStream = new JSIO.BinaryUrlStream(fileUrl, parseZipFile); 2.2112 + 2.2113 + return this; 2.2114 + }; 2.2115 + 2.2116 + 2.2117 + ZipFile.Signatures = { 2.2118 + Entry : 0x04034b50, 2.2119 + EndOfCentralDirectory : 0x06054b50, 2.2120 + DirEntry : 0x02014b50 2.2121 + }; 2.2122 + 2.2123 + ZipFile.Version = version; 2.2124 + 2.2125 + ZipFile.EncryptionAlgorithm = { 2.2126 + None : 0, 2.2127 + PkzipWeak : 1, 2.2128 + WinZipAes : 2 2.2129 + }; 2.2130 + 2.2131 + ZipFile.ExtraFieldTypes = {}; 2.2132 + ZipFile.ExtraFieldTypes[0x0001] = 'Zip64 Extended Info'; 2.2133 + ZipFile.ExtraFieldTypes[0x0007] = 'AV Info'; 2.2134 + ZipFile.ExtraFieldTypes[0x0008] = 'Extended Language Encoding Data (PFS)'; 2.2135 + ZipFile.ExtraFieldTypes[0x0009] = 'OS/2'; 2.2136 + ZipFile.ExtraFieldTypes[0x000a] = 'NTFS '; 2.2137 + ZipFile.ExtraFieldTypes[0x000c] = 'OpenVMS'; 2.2138 + ZipFile.ExtraFieldTypes[0x000d] = 'UNIX'; 2.2139 + ZipFile.ExtraFieldTypes[0x000e] = 'File Stream and Fork Descriptors'; 2.2140 + ZipFile.ExtraFieldTypes[0x000f] = 'Patch Descriptor'; 2.2141 + ZipFile.ExtraFieldTypes[0x0014] = 'PKCS#7 Store for X.509 Certificates'; 2.2142 + ZipFile.ExtraFieldTypes[0x0015] = 'X.509 Certificate ID and Signature (Individual File)'; 2.2143 + ZipFile.ExtraFieldTypes[0x0016] = 'X.509 Certificate ID (Central Directory)'; 2.2144 + ZipFile.ExtraFieldTypes[0x0017] = 'Strong Encryption Header'; 2.2145 + ZipFile.ExtraFieldTypes[0x0018] = 'Record Management Controls'; 2.2146 + ZipFile.ExtraFieldTypes[0x0019] = 'PKCS#7 Encryption Recipient Certificate List'; 2.2147 + ZipFile.ExtraFieldTypes[0x0065] = 'IBM S/390 (Z390), AS/400 (I400) attributes (uncompressed)'; 2.2148 + ZipFile.ExtraFieldTypes[0x0066] = 'IBM S/390 (Z390), AS/400 (I400) attributes (compressed)'; 2.2149 + ZipFile.ExtraFieldTypes[0x4690] = 'POSZIP 4690 (reserved) '; 2.2150 + ZipFile.ExtraFieldTypes[0x07c8] = 'Macintosh'; 2.2151 + ZipFile.ExtraFieldTypes[0x2605] = 'ZipIt Macintosh'; 2.2152 + ZipFile.ExtraFieldTypes[0x2705] = 'ZipIt Macintosh 1.3.5+'; 2.2153 + ZipFile.ExtraFieldTypes[0x2805] = 'ZipIt Macintosh 1.3.5+'; 2.2154 + ZipFile.ExtraFieldTypes[0x334d] = 'Info-ZIP Macintosh'; 2.2155 + ZipFile.ExtraFieldTypes[0x4341] = 'Acorn/SparkFS '; 2.2156 + ZipFile.ExtraFieldTypes[0x4453] = 'Windows NT security descriptor (binary ACL)'; 2.2157 + ZipFile.ExtraFieldTypes[0x4704] = 'VM/CMS'; 2.2158 + ZipFile.ExtraFieldTypes[0x470f] = 'MVS'; 2.2159 + ZipFile.ExtraFieldTypes[0x4b46] = 'FWKCS MD5'; 2.2160 + ZipFile.ExtraFieldTypes[0x4c41] = 'OS/2 access control list (text ACL)'; 2.2161 + ZipFile.ExtraFieldTypes[0x4d49] = 'Info-ZIP OpenVMS'; 2.2162 + ZipFile.ExtraFieldTypes[0x4f4c] = 'Xceed original location extra field'; 2.2163 + ZipFile.ExtraFieldTypes[0x5356] = 'AOS/VS (ACL)'; 2.2164 + ZipFile.ExtraFieldTypes[0x5455] = 'extended timestamp'; 2.2165 + ZipFile.ExtraFieldTypes[0x554e] = 'Xceed unicode extra field'; 2.2166 + ZipFile.ExtraFieldTypes[0x5855] = 'Info-ZIP UNIX (original, also OS/2, NT, etc)'; 2.2167 + ZipFile.ExtraFieldTypes[0x6375] = 'Info-ZIP Unicode Comment Extra Field'; 2.2168 + ZipFile.ExtraFieldTypes[0x6542] = 'BeOS/BeBox'; 2.2169 + ZipFile.ExtraFieldTypes[0x7075] = 'Info-ZIP Unicode Path Extra Field'; 2.2170 + ZipFile.ExtraFieldTypes[0x756e] = 'ASi UNIX'; 2.2171 + ZipFile.ExtraFieldTypes[0x7855] = 'Info-ZIP UNIX (new)'; 2.2172 + ZipFile.ExtraFieldTypes[0xa220] = 'Microsoft Open Packaging Growth Hint'; 2.2173 + ZipFile.ExtraFieldTypes[0xfd4a] = 'SMS/QDOS'; 2.2174 + 2.2175 +})();
3.1 --- /dev/null Thu Jan 01 00:00:00 1970 +0000 3.2 +++ b/endpoint/WebContent/js/geoxml3-kmz.js Fri Sep 14 13:03:24 2012 +0300 3.3 @@ -0,0 +1,1889 @@ 3.4 +/** 3.5 + * @fileOverview Renders KML on the Google Maps JavaScript API Version 3 3.6 + * @name GeoXML3 3.7 + * @author Sterling Udell, Larry Ross, Brendan Byrd 3.8 + * @see http://code.google.com/p/geoxml3/ 3.9 + * 3.10 + * geoxml3.js 3.11 + * 3.12 + * Renders KML on the Google Maps JavaScript API Version 3 3.13 + * http://code.google.com/p/geoxml3/ 3.14 + * 3.15 + * Copyright 2010 Sterling Udell, Larry Ross 3.16 + * 3.17 + * Licensed under the Apache License, Version 2.0 (the "License"); 3.18 + * you may not use this file except in compliance with the License. 3.19 + * You may obtain a copy of the License at 3.20 + * 3.21 + * http://www.apache.org/licenses/LICENSE-2.0 3.22 + * 3.23 + * Unless required by applicable law or agreed to in writing, software 3.24 + * distributed under the License is distributed on an "AS IS" BASIS, 3.25 + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 3.26 + * See the License for the specific language governing permissions and 3.27 + * limitations under the License. 3.28 + * 3.29 + */ 3.30 + 3.31 +if (!String.prototype.trim) { 3.32 +/** 3.33 + * Remove leading and trailing whitespace. 3.34 + * 3.35 + * @augments String 3.36 + * @return {String} 3.37 + */ 3.38 + String.prototype.trim = function () { 3.39 + return this.replace(/^\s+|\s+$/g, ''); 3.40 + }; 3.41 +} 3.42 + 3.43 +/** 3.44 + * @namespace The GeoXML3 namespace. 3.45 + */ 3.46 +geoXML3 = window.geoXML3 || {instances: []}; 3.47 + 3.48 +/** 3.49 + * Constructor for the root KML parser object. 3.50 + * 3.51 + * <p>All top-level objects and functions are declared under a namespace of geoXML3. 3.52 + * The core object is geoXML3.parser; typically, you'll instantiate a one parser 3.53 + * per map.</p> 3.54 + * 3.55 + * @class Main XML parser. 3.56 + * @param {geoXML3.parserOptions} options 3.57 + */ 3.58 +geoXML3.parser = function (options) { 3.59 + // Private variables 3.60 + var parserOptions = new geoXML3.parserOptions(options); 3.61 + var docs = []; // Individual KML documents 3.62 + var docsByUrl = {}; // Same docs as an hash by cleanURL 3.63 + var kmzMetaData = {}; // Extra files from KMZ data 3.64 + var styles = {}; // Global list of styles 3.65 + var lastPlacemark; 3.66 + var parserName; 3.67 + if (!parserOptions.infoWindow && parserOptions.singleInfoWindow) 3.68 + parserOptions.infoWindow = new google.maps.InfoWindow(); 3.69 + 3.70 + var parseKmlString = function (kmlString, docSet) { 3.71 + // Internal values for the set of documents as a whole 3.72 + var internals = { 3.73 + parser: this, 3.74 + docSet: docSet || [], 3.75 + remaining: 1, 3.76 + parseOnly: !(parserOptions.afterParse || parserOptions.processStyles) 3.77 + }; 3.78 + thisDoc = new Object(); 3.79 + thisDoc.internals = internals; 3.80 + internals.docSet.push(thisDoc); 3.81 + render(geoXML3.xmlParse(kmlString),thisDoc); 3.82 + } 3.83 + 3.84 + var parse = function (urls, docSet) { 3.85 + // Process one or more KML documents 3.86 + if (!parserName) { 3.87 + parserName = 'geoXML3.instances[' + (geoXML3.instances.push(this) - 1) + ']'; 3.88 + } 3.89 + 3.90 + if (typeof urls === 'string') { 3.91 + // Single KML document 3.92 + urls = [urls]; 3.93 + } 3.94 + 3.95 + // Internal values for the set of documents as a whole 3.96 + var internals = { 3.97 + parser: this, 3.98 + docSet: docSet || [], 3.99 + remaining: urls.length, 3.100 + parseOnly: !(parserOptions.afterParse || parserOptions.processStyles) 3.101 + }; 3.102 + var thisDoc, j; 3.103 + for (var i = 0; i < urls.length; i++) { 3.104 + var baseUrl = cleanURL(defileURL(location.pathname), urls[i]); 3.105 + if (docsByUrl[baseUrl]) { 3.106 + // Reloading an existing document 3.107 + thisDoc = docsByUrl[baseUrl]; 3.108 + thisDoc.reload = true; 3.109 + } 3.110 + else { 3.111 + thisDoc = new Object(); 3.112 + thisDoc.baseUrl = baseUrl; 3.113 + internals.docSet.push(thisDoc); 3.114 + } 3.115 + thisDoc.url = urls[i]; 3.116 + thisDoc.internals = internals; 3.117 + fetchDoc(thisDoc.url, thisDoc); 3.118 + } 3.119 + }; 3.120 + 3.121 + function fetchDoc(url, doc, resFunc) { 3.122 + resFunc = resFunc || function (responseXML) { render(responseXML, doc); }; 3.123 + 3.124 + if (typeof ZipFile === 'function' && typeof JSIO === 'object' && typeof JSIO.guessFileType === 'function') { // KMZ support requires these modules loaded 3.125 + contentType = JSIO.guessFileType(doc.baseUrl); 3.126 + if (contentType == JSIO.FileType.Binary || contentType == JSIO.FileType.Unknown) { 3.127 + doc.isCompressed = true; 3.128 + doc.baseDir = doc.baseUrl + '/'; 3.129 + geoXML3.fetchZIP(url, resFunc, doc.internals.parser); 3.130 + return; 3.131 + } 3.132 + } 3.133 + doc.isCompressed = false; 3.134 + doc.baseDir = defileURL(doc.baseUrl); 3.135 + geoXML3.fetchXML(url, resFunc); 3.136 + } 3.137 + 3.138 + var hideDocument = function (doc) { 3.139 + if (!doc) doc = docs[0]; 3.140 + // Hide the map objects associated with a document 3.141 + var i; 3.142 + if (!!doc.markers) { 3.143 + for (i = 0; i < doc.markers.length; i++) { 3.144 + if(!!doc.markers[i].infoWindow) doc.markers[i].infoWindow.close(); 3.145 + doc.markers[i].setVisible(false); 3.146 + } 3.147 + } 3.148 + if (!!doc.ggroundoverlays) { 3.149 + for (i = 0; i < doc.ggroundoverlays.length; i++) { 3.150 + doc.ggroundoverlays[i].setOpacity(0); 3.151 + } 3.152 + } 3.153 + if (!!doc.gpolylines) { 3.154 + for (i=0;i<doc.gpolylines.length;i++) { 3.155 + if(!!doc.gpolylines[i].infoWindow) doc.gpolylines[i].infoWindow.close(); 3.156 + doc.gpolylines[i].setMap(null); 3.157 + } 3.158 + } 3.159 + if (!!doc.gpolygons) { 3.160 + for (i=0;i<doc.gpolygons.length;i++) { 3.161 + if(!!doc.gpolygons[i].infoWindow) doc.gpolygons[i].infoWindow.close(); 3.162 + doc.gpolygons[i].setMap(null); 3.163 + } 3.164 + } 3.165 + }; 3.166 + 3.167 + var showDocument = function (doc) { 3.168 + if (!doc) doc = docs[0]; 3.169 + // Show the map objects associated with a document 3.170 + var i; 3.171 + if (!!doc.markers) { 3.172 + for (i = 0; i < doc.markers.length; i++) { 3.173 + doc.markers[i].setVisible(true); 3.174 + } 3.175 + } 3.176 + if (!!doc.ggroundoverlays) { 3.177 + for (i = 0; i < doc.ggroundoverlays.length; i++) { 3.178 + doc.ggroundoverlays[i].setOpacity(doc.ggroundoverlays[i].percentOpacity_); 3.179 + } 3.180 + } 3.181 + if (!!doc.gpolylines) { 3.182 + for (i=0;i<doc.gpolylines.length;i++) { 3.183 + doc.gpolylines[i].setMap(parserOptions.map); 3.184 + } 3.185 + } 3.186 + if (!!doc.gpolygons) { 3.187 + for (i=0;i<doc.gpolygons.length;i++) { 3.188 + doc.gpolygons[i].setMap(parserOptions.map); 3.189 + } 3.190 + } 3.191 + }; 3.192 + 3.193 + var defaultStyle = { 3.194 + balloon: { 3.195 + bgColor: 'ffffffff', 3.196 + textColor: 'ff000000', 3.197 + text: "<h3>$[name]</h3>\n<div>$[description]</div>\n<div>$[geDirections]</div>", 3.198 + displayMode: 'default' 3.199 + }, 3.200 + icon: { 3.201 + scale: 1.0, 3.202 + dim: { 3.203 + x: 0, 3.204 + y: 0, 3.205 + w: -1, 3.206 + h: -1 3.207 + }, 3.208 + hotSpot: { 3.209 + x: 0.5, 3.210 + y: 0.5, 3.211 + xunits: 'fraction', 3.212 + yunits: 'fraction' 3.213 + } 3.214 + }, 3.215 + line: { 3.216 + color: 'ffffffff', // white (KML default) 3.217 + colorMode: 'normal', 3.218 + width: 1.0 3.219 + }, 3.220 + poly: { 3.221 + color: 'ffffffff', // white (KML default) 3.222 + colorMode: 'normal', 3.223 + fill: true, 3.224 + outline: true 3.225 + } 3.226 + }; 3.227 + 3.228 + var kmlNS = 'http://www.opengis.net/kml/2.2'; 3.229 + var gxNS = 'http://www.google.com/kml/ext/2.2'; 3.230 + var nodeValue = geoXML3.nodeValue; 3.231 + var getBooleanValue = geoXML3.getBooleanValue; 3.232 + var getElementsByTagNameNS = geoXML3.getElementsByTagNameNS; 3.233 + var getElementsByTagName = geoXML3.getElementsByTagName; 3.234 + 3.235 +function processStyleUrl(node) { 3.236 + var styleUrlStr = nodeValue(getElementsByTagName(node, 'styleUrl')[0]); 3.237 + if (!!styleUrlStr && styleUrlStr.indexOf('#') != -1) 3.238 + var styleUrl = styleUrlStr.split('#'); 3.239 + else var styleUrl = ["",""]; 3.240 + return styleUrl; 3.241 +} 3.242 + 3.243 + function processStyle(thisNode, baseUrl, styleID, baseDir) { 3.244 + var style = (baseUrl === '{inline}') ? clone(defaultStyle) : (styles[baseUrl][styleID] = styles[baseUrl][styleID] || clone(defaultStyle)); 3.245 + 3.246 + var styleNodes = getElementsByTagName(thisNode, 'BalloonStyle'); 3.247 + if (!!styleNodes && styleNodes.length > 0) { 3.248 + style.balloon.bgColor = nodeValue(getElementsByTagName(styleNodes[0], 'bgColor')[0], style.balloon.bgColor); 3.249 + style.balloon.textColor = nodeValue(getElementsByTagName(styleNodes[0], 'textColor')[0], style.balloon.textColor); 3.250 + style.balloon.text = nodeValue(getElementsByTagName(styleNodes[0], 'text')[0], style.balloon.text); 3.251 + style.balloon.displayMode = nodeValue(getElementsByTagName(styleNodes[0], 'displayMode')[0], style.balloon.displayMode); 3.252 + } 3.253 + 3.254 + // style.list = (unsupported; doesn't make sense in Google Maps) 3.255 + 3.256 + var styleNodes = getElementsByTagName(thisNode, 'IconStyle'); 3.257 + if (!!styleNodes && styleNodes.length > 0) { 3.258 + var icon = style.icon; 3.259 + 3.260 + icon.scale = parseFloat(nodeValue(getElementsByTagName(styleNodes[0], 'scale')[0], icon.scale)); 3.261 + // style.icon.heading = (unsupported; not supported in API) 3.262 + // style.icon.color = (unsupported; not supported in API) 3.263 + // style.icon.colorMode = (unsupported; not supported in API) 3.264 + 3.265 + styleNodes = getElementsByTagName(thisNode, 'Icon'); 3.266 + if (!!styleNodes && styleNodes.length > 0) { 3.267 + icon.href = nodeValue(getElementsByTagName(styleNodes[0], 'href')[0]); 3.268 + icon.url = cleanURL(baseDir, icon.href); 3.269 + // Detect images buried in KMZ files (and use a base64 encoded URL) 3.270 + if (kmzMetaData[icon.url]) icon.url = kmzMetaData[icon.url].dataUrl; 3.271 + 3.272 + // Support for icon palettes and exact size dimensions 3.273 + icon.dim = { 3.274 + x: parseInt(nodeValue(getElementsByTagNameNS(styleNodes[0], gxNS, 'x')[0], icon.dim.x)), 3.275 + y: parseInt(nodeValue(getElementsByTagNameNS(styleNodes[0], gxNS, 'y')[0], icon.dim.y)), 3.276 + w: parseInt(nodeValue(getElementsByTagNameNS(styleNodes[0], gxNS, 'w')[0], icon.dim.w)), 3.277 + h: parseInt(nodeValue(getElementsByTagNameNS(styleNodes[0], gxNS, 'h')[0], icon.dim.h)) 3.278 + }; 3.279 + 3.280 + styleNodes = getElementsByTagName(styleNodes[0], 'hotSpot')[0]; 3.281 + if (!!styleNodes && styleNodes.length > 0) { 3.282 + icon.hotSpot = { 3.283 + x: styleNodes[0].getAttribute('x'), 3.284 + y: styleNodes[0].getAttribute('y'), 3.285 + xunits: styleNodes[0].getAttribute('xunits'), 3.286 + yunits: styleNodes[0].getAttribute('yunits') 3.287 + }; 3.288 + } 3.289 + 3.290 + // certain occasions where we need the pixel size of the image (like the default settings...) 3.291 + // (NOTE: Scale is applied to entire image, not just the section of the icon palette. So, 3.292 + // if we need scaling, we'll need the img dimensions no matter what.) 3.293 + if ( (icon.dim.w < 0 || icon.dim.h < 0) && (icon.xunits != 'pixels' || icon.yunits == 'fraction') || icon.scale != 1.0) { 3.294 + // (hopefully, this will load by the time we need it...) 3.295 + icon.img = new Image(); 3.296 + icon.img.onload = function() { 3.297 + if (icon.dim.w < 0 || icon.dim.h < 0) { 3.298 + icon.dim.w = this.width; 3.299 + icon.dim.h = this.height; 3.300 + } 3.301 + }; 3.302 + icon.img.src = icon.url; 3.303 + 3.304 + // sometimes the file is already cached and it never calls onLoad 3.305 + if (icon.img.width > 0) { 3.306 + icon.dim.w = icon.img.width; 3.307 + icon.dim.h = icon.img.height; 3.308 + } 3.309 + } 3.310 + } 3.311 + } 3.312 + 3.313 + // style.label = (unsupported; may be possible but not with API) 3.314 + 3.315 + styleNodes = getElementsByTagName(thisNode, 'LineStyle'); 3.316 + if (!!styleNodes && styleNodes.length > 0) { 3.317 + style.line.color = nodeValue(getElementsByTagName(styleNodes[0], 'color')[0], style.line.color); 3.318 + style.line.colorMode = nodeValue(getElementsByTagName(styleNodes[0], 'colorMode')[0], style.line.colorMode); 3.319 + style.line.width = nodeValue(getElementsByTagName(styleNodes[0], 'width')[0], style.line.width); 3.320 + // style.line.outerColor = (unsupported; not supported in API) 3.321 + // style.line.outerWidth = (unsupported; not supported in API) 3.322 + // style.line.physicalWidth = (unsupported; unneccesary in Google Maps) 3.323 + // style.line.labelVisibility = (unsupported; possible to implement) 3.324 + } 3.325 + 3.326 + styleNodes = getElementsByTagName(thisNode, 'PolyStyle'); 3.327 + if (!!styleNodes && styleNodes.length > 0) { 3.328 + style.poly.color = nodeValue( getElementsByTagName(styleNodes[0], 'color')[0], style.poly.color); 3.329 + style.poly.colorMode = nodeValue( getElementsByTagName(styleNodes[0], 'colorMode')[0], style.poly.colorMode); 3.330 + style.poly.outline = getBooleanValue(getElementsByTagName(styleNodes[0], 'outline')[0], style.poly.outline); 3.331 + style.poly.fill = getBooleanValue(getElementsByTagName(styleNodes[0], 'fill')[0], style.poly.fill); 3.332 + } 3.333 + return style; 3.334 + } 3.335 + 3.336 + // from http://stackoverflow.com/questions/122102/what-is-the-most-efficient-way-to-clone-a-javascript-object 3.337 + // http://keithdevens.com/weblog/archive/2007/Jun/07/javascript.clone 3.338 + function clone(obj){ 3.339 + if(obj == null || typeof(obj) != 'object') return obj; 3.340 + if (obj.cloneNode) return obj.cloneNode(true); 3.341 + var temp = new obj.constructor(); 3.342 + for(var key in obj) temp[key] = clone(obj[key]); 3.343 + return temp; 3.344 + } 3.345 + 3.346 + function processStyleMap(thisNode, baseUrl, styleID, baseDir) { 3.347 + var pairs = getElementsByTagName(thisNode, 'Pair'); 3.348 + var map = new Object(); 3.349 + 3.350 + // add each key to the map 3.351 + for (var pr=0;pr<pairs.length;pr++) { 3.352 + var pairKey = nodeValue(getElementsByTagName(pairs[pr], 'key')[0]); 3.353 + var pairStyle = nodeValue(getElementsByTagName(pairs[pr], 'Style')[0]); 3.354 + var pairStyleUrl = processStyleUrl(pairs[pr]); 3.355 + var pairStyleBaseUrl = pairStyleUrl[0] ? cleanURL(baseDir, pairStyleUrl[0]) : baseUrl; 3.356 + var pairStyleID = pairStyleUrl[1]; 3.357 + 3.358 + if (!!pairStyle) { 3.359 + map[pairKey] = processStyle(pairStyle, pairStyleBaseUrl, pairStyleID); 3.360 + } else if (!!pairStyleID && !!styles[pairStyleBaseUrl][pairStyleID]) { 3.361 + map[pairKey] = clone(styles[pairStyleBaseUrl][pairStyleID]); 3.362 + } 3.363 + } 3.364 + if (!!map["normal"]) { 3.365 + styles[baseUrl][styleID] = clone(map["normal"]); 3.366 + } else { 3.367 + styles[baseUrl][styleID] = clone(defaultStyle); 3.368 + } 3.369 + if (!!map["highlight"]) { 3.370 + processStyleID(map["highlight"]); 3.371 + } 3.372 + styles[baseUrl][styleID].map = clone(map); 3.373 + } 3.374 + 3.375 + function processPlacemarkCoords(node, tag) { 3.376 + var parent = getElementsByTagName(node, tag); 3.377 + var coordListA = []; 3.378 + for (var i=0; i<parent.length; i++) { 3.379 + var coordNodes = getElementsByTagName(parent[i], 'coordinates'); 3.380 + if (!coordNodes) { 3.381 + if (coordListA.length > 0) { 3.382 + break; 3.383 + } else { 3.384 + return [{coordinates: []}]; 3.385 + } 3.386 + } 3.387 + 3.388 + for (var j=0; j<coordNodes.length;j++) { 3.389 + var coords = nodeValue(coordNodes[j]).trim(); 3.390 + coords = coords.replace(/,\s+/g, ','); 3.391 + var path = coords.split(/\s+/g); 3.392 + var pathLength = path.length; 3.393 + var coordList = []; 3.394 + for (var k = 0; k < pathLength; k++) { 3.395 + coords = path[k].split(','); 3.396 + if (!isNaN(coords[0]) && !isNaN(coords[1])) { 3.397 + coordList.push({ 3.398 + lat: parseFloat(coords[1]), 3.399 + lng: parseFloat(coords[0]), 3.400 + alt: parseFloat(coords[2]) 3.401 + }); 3.402 + } 3.403 + } 3.404 + coordListA.push({coordinates: coordList}); 3.405 + } 3.406 + } 3.407 + return coordListA; 3.408 + } 3.409 + 3.410 + var render = function (responseXML, doc) { 3.411 + // Callback for retrieving a KML document: parse the KML and display it on the map 3.412 + if (!responseXML) { 3.413 + // Error retrieving the data 3.414 + geoXML3.log('Unable to retrieve ' + doc.url); 3.415 + if (parserOptions.failedParse) parserOptions.failedParse(doc); 3.416 + doc.failed = true; 3.417 + return; 3.418 + } else if (responseXML.parseError && responseXML.parseError.errorCode != 0) { 3.419 + // IE parse error 3.420 + var err = responseXML.parseError; 3.421 + var msg = 'Parse error in line ' + err.line + ', col ' + err.linePos + ' (error code: ' + err.errorCode + ")\n" + 3.422 + "\nError Reason: " + err.reason + 3.423 + 'Error Line: ' + err.srcText; 3.424 + 3.425 + geoXML3.log('Unable to retrieve ' + doc.url + ': ' + msg); 3.426 + if (parserOptions.failedParse) parserOptions.failedParse(doc); 3.427 + doc.failed = true; 3.428 + return; 3.429 + } else if (responseXML.documentElement && responseXML.documentElement.nodeName == 'parsererror') { 3.430 + // Firefox parse error 3.431 + geoXML3.log('Unable to retrieve ' + doc.url + ': ' + responseXML.documentElement.childNodes[0].nodeValue); 3.432 + if (parserOptions.failedParse) parserOptions.failedParse(doc); 3.433 + doc.failed = true; 3.434 + return; 3.435 + } else if (!doc) { 3.436 + throw 'geoXML3 internal error: render called with null document'; 3.437 + } else { //no errors 3.438 + var i; 3.439 + doc.placemarks = []; 3.440 + doc.groundoverlays = []; 3.441 + doc.ggroundoverlays = []; 3.442 + doc.networkLinks = []; 3.443 + doc.gpolygons = []; 3.444 + doc.gpolylines = []; 3.445 + 3.446 + // Check for dependent KML files 3.447 + var nodes = getElementsByTagName(responseXML, 'styleUrl'); 3.448 + var docSet = doc.internals.docSet; 3.449 + 3.450 + for (var i = 0; i < nodes.length; i++) { 3.451 + var url = nodeValue(nodes[i]).split('#')[0]; 3.452 + if (!url) continue; // #id (inside doc) 3.453 + var rUrl = cleanURL( doc.baseDir, url ); 3.454 + if (rUrl === doc.baseUrl) continue; // self 3.455 + if (docsByUrl[rUrl]) continue; // already loaded 3.456 + 3.457 + var thisDoc; 3.458 + var j = docSet.indexOfObjWithItem('baseUrl', rUrl); 3.459 + if (j != -1) { 3.460 + // Already listed to be loaded, but probably in the wrong order. 3.461 + // Load it right away to immediately resolve dependency. 3.462 + thisDoc = docSet[j]; 3.463 + if (thisDoc.failed) continue; // failed to load last time; don't retry it again 3.464 + } 3.465 + else { 3.466 + // Not listed at all; add it in 3.467 + thisDoc = new Object(); 3.468 + thisDoc.url = rUrl; // url can't be trusted inside KMZ files, since it may .. outside of the archive 3.469 + thisDoc.baseUrl = rUrl; 3.470 + thisDoc.internals = doc.internals; 3.471 + 3.472 + doc.internals.docSet.push(thisDoc); 3.473 + doc.internals.remaining++; 3.474 + } 3.475 + 3.476 + // render dependent KML first then re-run renderer 3.477 + fetchDoc(rUrl, thisDoc, function (thisResXML) { 3.478 + render(thisResXML, thisDoc); 3.479 + render(responseXML, doc); 3.480 + }); 3.481 + 3.482 + // to prevent cross-dependency issues, just load the one 3.483 + // file first and re-check the rest later 3.484 + return; 3.485 + } 3.486 + 3.487 + // Parse styles 3.488 + doc.styles = styles[doc.baseUrl] = styles[doc.baseUrl] || {}; 3.489 + var styleID, styleNodes; 3.490 + nodes = getElementsByTagName(responseXML, 'Style'); 3.491 + nodeCount = nodes.length; 3.492 + for (i = 0; i < nodeCount; i++) { 3.493 + thisNode = nodes[i]; 3.494 + var styleID = thisNode.getAttribute('id'); 3.495 + if (!!styleID) processStyle(thisNode, doc.baseUrl, styleID, doc.baseDir); 3.496 + } 3.497 + // Parse StyleMap nodes 3.498 + nodes = getElementsByTagName(responseXML, 'StyleMap'); 3.499 + for (i = 0; i < nodes.length; i++) { 3.500 + thisNode = nodes[i]; 3.501 + var styleID = thisNode.getAttribute('id'); 3.502 + if (!!styleID) processStyleMap(thisNode, doc.baseUrl, styleID, doc.baseDir); 3.503 + } 3.504 + 3.505 + if (!!parserOptions.processStyles || !parserOptions.createMarker) { 3.506 + // Convert parsed styles into GMaps equivalents 3.507 + processStyles(doc); 3.508 + } 3.509 + 3.510 + // Parse placemarks 3.511 + if (!!doc.reload && !!doc.markers) { 3.512 + for (i = 0; i < doc.markers.length; i++) { 3.513 + doc.markers[i].active = false; 3.514 + } 3.515 + } 3.516 + var placemark, node, coords, path, marker, poly; 3.517 + var placemark, coords, path, pathLength, marker, polygonNodes, coordList; 3.518 + var placemarkNodes = getElementsByTagName(responseXML, 'Placemark'); 3.519 + for (pm = 0; pm < placemarkNodes.length; pm++) { 3.520 + // Init the placemark object 3.521 + node = placemarkNodes[pm]; 3.522 + var styleUrl = processStyleUrl(node); 3.523 + placemark = { 3.524 + name: nodeValue(getElementsByTagName(node, 'name')[0]), 3.525 + description: nodeValue(getElementsByTagName(node, 'description')[0]), 3.526 + styleUrl: styleUrl.join('#'), 3.527 + styleBaseUrl: styleUrl[0] ? cleanURL(doc.baseDir, styleUrl[0]) : doc.baseUrl, 3.528 + styleID: styleUrl[1], 3.529 + visibility: getBooleanValue(getElementsByTagName(node, 'visibility')[0], true), 3.530 + balloonVisibility: getBooleanValue(getElementsByTagNameNS(node, gxNS, 'balloonVisibility')[0], !parserOptions.suppressInfoWindows) 3.531 + }; 3.532 + placemark.style = (styles[placemark.styleBaseUrl] && styles[placemark.styleBaseUrl][placemark.styleID]) || clone(defaultStyle); 3.533 + // inline style overrides shared style 3.534 + var inlineStyles = getElementsByTagName(node, 'Style'); 3.535 + if (inlineStyles && (inlineStyles.length > 0)) { 3.536 + var style = processStyle(node, '{inline}', '{inline}'); 3.537 + processStyleID(style); 3.538 + if (style) placemark.style = style; 3.539 + } 3.540 + 3.541 + if (/^https?:\/\//.test(placemark.description)) { 3.542 + placemark.description = ['<a href="', placemark.description, '">', placemark.description, '</a>'].join(''); 3.543 + } 3.544 + 3.545 + // record list of variables for substitution 3.546 + placemark.vars = { 3.547 + display: { 3.548 + name: 'Name', 3.549 + description: 'Description', 3.550 + address: 'Street Address', 3.551 + id: 'ID', 3.552 + Snippet: 'Snippet', 3.553 + geDirections: 'Directions' 3.554 + }, 3.555 + val: { 3.556 + name: placemark.name || '', 3.557 + description: placemark.description || '', 3.558 + address: nodeValue(getElementsByTagName(node, 'address')[0], ''), 3.559 + id: node.getAttribute('id') || '', 3.560 + Snippet: nodeValue(getElementsByTagName(node, 'Snippet')[0], '') 3.561 + }, 3.562 + directions: [ 3.563 + 'f=d', 3.564 + 'source=GeoXML3' 3.565 + ] 3.566 + }; 3.567 + 3.568 + // add extended data to variables 3.569 + var extDataNodes = getElementsByTagName(node, 'ExtendedData'); 3.570 + if (!!extDataNodes && extDataNodes.length > 0) { 3.571 + var dataNodes = getElementsByTagName(extDataNodes[0], 'Data'); 3.572 + for (var d = 0; d < dataNodes.length; d++) { 3.573 + var dn = dataNodes[d]; 3.574 + var name = dn.getAttribute('name'); 3.575 + if (!name) continue; 3.576 + var dName = nodeValue(getElementsByTagName(dn, 'displayName')[0], name); 3.577 + var val = nodeValue(getElementsByTagName(dn, 'value')[0]); 3.578 + 3.579 + placemark.vars.val[name] = val; 3.580 + placemark.vars.display[name] = dName; 3.581 + } 3.582 + } 3.583 + 3.584 + // process MultiGeometry 3.585 + var GeometryNodes = getElementsByTagName(node, 'coordinates'); 3.586 + var Geometry = null; 3.587 + if (!!GeometryNodes && (GeometryNodes.length > 0)) { 3.588 + for (var gn=0;gn<GeometryNodes.length;gn++) { 3.589 + if (GeometryNodes[gn].parentNode && 3.590 + GeometryNodes[gn].parentNode.nodeName) { 3.591 + var GeometryPN = GeometryNodes[gn].parentNode; 3.592 + Geometry = GeometryPN.nodeName; 3.593 + 3.594 + // Extract the coordinates 3.595 + // What sort of placemark? 3.596 + switch(Geometry) { 3.597 + case "Point": 3.598 + placemark.Point = processPlacemarkCoords(node, "Point")[0]; 3.599 + placemark.latlng = new google.maps.LatLng(placemark.Point.coordinates[0].lat, placemark.Point.coordinates[0].lng); 3.600 + pathLength = 1; 3.601 + break; 3.602 + case "LinearRing": 3.603 + // Polygon/line 3.604 + polygonNodes = getElementsByTagName(node, 'Polygon'); 3.605 + // Polygon 3.606 + if (!placemark.Polygon) 3.607 + placemark.Polygon = [{ 3.608 + outerBoundaryIs: {coordinates: []}, 3.609 + innerBoundaryIs: [{coordinates: []}] 3.610 + }]; 3.611 + for (var pg=0;pg<polygonNodes.length;pg++) { 3.612 + placemark.Polygon[pg] = { 3.613 + outerBoundaryIs: {coordinates: []}, 3.614 + innerBoundaryIs: [{coordinates: []}] 3.615 + } 3.616 + placemark.Polygon[pg].outerBoundaryIs = processPlacemarkCoords(polygonNodes[pg], "outerBoundaryIs"); 3.617 + placemark.Polygon[pg].innerBoundaryIs = processPlacemarkCoords(polygonNodes[pg], "innerBoundaryIs"); 3.618 + } 3.619 + coordList = placemark.Polygon[0].outerBoundaryIs; 3.620 + break; 3.621 + 3.622 + case "LineString": 3.623 + pathLength = 0; 3.624 + placemark.LineString = processPlacemarkCoords(node,"LineString"); 3.625 + break; 3.626 + 3.627 + default: 3.628 + break; 3.629 + } 3.630 + } 3.631 + } 3.632 + } 3.633 + 3.634 + // call the custom placemark parse function if it is defined 3.635 + if (!!parserOptions.pmParseFn) parserOptions.pmParseFn(node, placemark); 3.636 + doc.placemarks.push(placemark); 3.637 + 3.638 + // single marker 3.639 + if (placemark.Point) { 3.640 + if (!!google.maps) { 3.641 + doc.bounds = doc.bounds || new google.maps.LatLngBounds(); 3.642 + doc.bounds.extend(placemark.latlng); 3.643 + } 3.644 + 3.645 + // Potential user-defined marker handler 3.646 + var pointCreateFunc = parserOptions.createMarker || createMarker; 3.647 + var found = false; 3.648 + if (!parserOptions.createMarker) { 3.649 + // Check to see if this marker was created on a previous load of this document 3.650 + if (!!doc) { 3.651 + doc.markers = doc.markers || []; 3.652 + if (doc.reload) { 3.653 + for (var j = 0; j < doc.markers.length; j++) { 3.654 + if (doc.markers[j].getPosition().equals(placemark.latlng)) { 3.655 + found = doc.markers[j].active = true; 3.656 + break; 3.657 + } 3.658 + } 3.659 + } 3.660 + } 3.661 + } 3.662 + if (!found) { 3.663 + // Call the marker creator 3.664 + var marker = pointCreateFunc(placemark, doc); 3.665 + if (marker) marker.active = placemark.visibility; 3.666 + } 3.667 + } 3.668 + // polygon/line 3.669 + var poly, line; 3.670 + if (!!doc) { 3.671 + if (placemark.Polygon) doc.gpolygons = doc.gpolygons || []; 3.672 + if (placemark.LineString) doc.gpolylines = doc.gpolylines || []; 3.673 + } 3.674 + 3.675 + var polyCreateFunc = parserOptions.createPolygon || createPolygon; 3.676 + var lineCreateFunc = parserOptions.createLineString || createPolyline; 3.677 + if (placemark.Polygon) { 3.678 + poly = polyCreateFunc(placemark,doc); 3.679 + if (poly) poly.active = placemark.visibility; 3.680 + } 3.681 + if (placemark.LineString) { 3.682 + line = lineCreateFunc(placemark,doc); 3.683 + if (line) line.active = placemark.visibility; 3.684 + } 3.685 + if (!!google.maps) { 3.686 + doc.bounds = doc.bounds || new google.maps.LatLngBounds(); 3.687 + if (poly) doc.bounds.union(poly.bounds); 3.688 + if (line) doc.bounds.union(line.bounds); 3.689 + } 3.690 + 3.691 + } // placemark loop 3.692 + 3.693 + if (!!doc.reload && !!doc.markers) { 3.694 + for (i = doc.markers.length - 1; i >= 0 ; i--) { 3.695 + if (!doc.markers[i].active) { 3.696 + if (!!doc.markers[i].infoWindow) { 3.697 + doc.markers[i].infoWindow.close(); 3.698 + } 3.699 + doc.markers[i].setMap(null); 3.700 + doc.markers.splice(i, 1); 3.701 + } 3.702 + } 3.703 + } 3.704 + 3.705 + // Parse ground overlays 3.706 + if (!!doc.reload && !!doc.groundoverlays) { 3.707 + for (i = 0; i < doc.groundoverlays.length; i++) { 3.708 + doc.groundoverlays[i].active = false; 3.709 + } 3.710 + } 3.711 + 3.712 + if (!!doc) { 3.713 + doc.groundoverlays = doc.groundoverlays || []; 3.714 + } 3.715 + // doc.groundoverlays =[]; 3.716 + var groundOverlay, color, transparency, overlay; 3.717 + var groundNodes = getElementsByTagName(responseXML, 'GroundOverlay'); 3.718 + for (i = 0; i < groundNodes.length; i++) { 3.719 + node = groundNodes[i]; 3.720 + 3.721 + // Detect images buried in KMZ files (and use a base64 encoded URL) 3.722 + var gnUrl = cleanURL( doc.baseDir, nodeValue(getElementsByTagName(node, 'href')[0]) ); 3.723 + if (kmzMetaData[gnUrl]) gnUrl = kmzMetaData[gnUrl].dataUrl; 3.724 + 3.725 + // Init the ground overlay object 3.726 + groundOverlay = { 3.727 + name: nodeValue(getElementsByTagName(node, 'name')[0]), 3.728 + description: nodeValue(getElementsByTagName(node, 'description')[0]), 3.729 + icon: { href: gnUrl }, 3.730 + latLonBox: { 3.731 + north: parseFloat(nodeValue(getElementsByTagName(node, 'north')[0])), 3.732 + east: parseFloat(nodeValue(getElementsByTagName(node, 'east')[0])), 3.733 + south: parseFloat(nodeValue(getElementsByTagName(node, 'south')[0])), 3.734 + west: parseFloat(nodeValue(getElementsByTagName(node, 'west')[0])) 3.735 + } 3.736 + }; 3.737 + if (!!google.maps) { 3.738 + doc.bounds = doc.bounds || new google.maps.LatLngBounds(); 3.739 + doc.bounds.union(new google.maps.LatLngBounds( 3.740 + new google.maps.LatLng(groundOverlay.latLonBox.south, groundOverlay.latLonBox.west), 3.741 + new google.maps.LatLng(groundOverlay.latLonBox.north, groundOverlay.latLonBox.east) 3.742 + )); 3.743 + } 3.744 + 3.745 + // Opacity is encoded in the color node 3.746 + var colorNode = getElementsByTagName(node, 'color'); 3.747 + if (colorNode && colorNode.length > 0) { 3.748 + groundOverlay.opacity = geoXML3.getOpacity(nodeValue(colorNode[0])); 3.749 + } else { 3.750 + groundOverlay.opacity = 1.0; // KML default 3.751 + } 3.752 + 3.753 + doc.groundoverlays.push(groundOverlay); 3.754 + if (!!parserOptions.createOverlay) { 3.755 + // User-defined overlay handler 3.756 + parserOptions.createOverlay(groundOverlay, doc); 3.757 + } else { 3.758 + // Check to see if this overlay was created on a previous load of this document 3.759 + var found = false; 3.760 + if (!!doc) { 3.761 + doc.groundoverlays = doc.groundoverlays || []; 3.762 + if (doc.reload) { 3.763 + overlayBounds = new google.maps.LatLngBounds( 3.764 + new google.maps.LatLng(groundOverlay.latLonBox.south, groundOverlay.latLonBox.west), 3.765 + new google.maps.LatLng(groundOverlay.latLonBox.north, groundOverlay.latLonBox.east) 3.766 + ); 3.767 + var overlays = doc.groundoverlays; 3.768 + for (i = overlays.length; i--;) { 3.769 + if ((overlays[i].bounds().equals(overlayBounds)) && 3.770 + (overlays.url_ === groundOverlay.icon.href)) { 3.771 + found = overlays[i].active = true; 3.772 + break; 3.773 + } 3.774 + } 3.775 + } 3.776 + } 3.777 + 3.778 + if (!found) { 3.779 + // Call the built-in overlay creator 3.780 + overlay = createOverlay(groundOverlay, doc); 3.781 + overlay.active = true; 3.782 + } 3.783 + } 3.784 + if (!!doc.reload && !!doc.groundoverlays && !!doc.groundoverlays.length) { 3.785 + var overlays = doc.groundoverlays; 3.786 + for (i = overlays.length; i--;) { 3.787 + if (!overlays[i].active) { 3.788 + overlays[i].remove(); 3.789 + overlays.splice(i, 1); 3.790 + } 3.791 + } 3.792 + doc.groundoverlays = overlays; 3.793 + } 3.794 + } 3.795 + 3.796 + // Parse network links 3.797 + var networkLink; 3.798 + var docPath = document.location.pathname.split('/'); 3.799 + docPath = docPath.splice(0, docPath.length - 1).join('/'); 3.800 + var linkNodes = getElementsByTagName(responseXML, 'NetworkLink'); 3.801 + for (i = 0; i < linkNodes.length; i++) { 3.802 + node = linkNodes[i]; 3.803 + 3.804 + // Init the network link object 3.805 + networkLink = { 3.806 + name: nodeValue(getElementsByTagName(node, 'name')[0]), 3.807 + link: { 3.808 + href: nodeValue(getElementsByTagName(node, 'href')[0]), 3.809 + refreshMode: nodeValue(getElementsByTagName(node, 'refreshMode')[0]) 3.810 + } 3.811 + }; 3.812 + 3.813 + // Establish the specific refresh mode 3.814 + if (!networkLink.link.refreshMode) { 3.815 + networkLink.link.refreshMode = 'onChange'; 3.816 + } 3.817 + if (networkLink.link.refreshMode === 'onInterval') { 3.818 + networkLink.link.refreshInterval = parseFloat(nodeValue(getElementsByTagName(node, 'refreshInterval')[0])); 3.819 + if (isNaN(networkLink.link.refreshInterval)) { 3.820 + networkLink.link.refreshInterval = 0; 3.821 + } 3.822 + } else if (networkLink.link.refreshMode === 'onChange') { 3.823 + networkLink.link.viewRefreshMode = nodeValue(getElementsByTagName(node, 'viewRefreshMode')[0]); 3.824 + if (!networkLink.link.viewRefreshMode) { 3.825 + networkLink.link.viewRefreshMode = 'never'; 3.826 + } 3.827 + if (networkLink.link.viewRefreshMode === 'onStop') { 3.828 + networkLink.link.viewRefreshTime = nodeValue(getElementsByTagName(node, 'refreshMode')[0]); 3.829 + networkLink.link.viewFormat = nodeValue(getElementsByTagName(node, 'refreshMode')[0]); 3.830 + if (!networkLink.link.viewFormat) { 3.831 + networkLink.link.viewFormat = 'BBOX=[bboxWest],[bboxSouth],[bboxEast],[bboxNorth]'; 3.832 + } 3.833 + } 3.834 + } 3.835 + 3.836 + if (!/^[\/|http]/.test(networkLink.link.href)) { 3.837 + // Fully-qualify the HREF 3.838 + networkLink.link.href = docPath + '/' + networkLink.link.href; 3.839 + } 3.840 + 3.841 + // Apply the link 3.842 + if ((networkLink.link.refreshMode === 'onInterval') && 3.843 + (networkLink.link.refreshInterval > 0)) { 3.844 + // Reload at regular intervals 3.845 + setInterval(parserName + '.parse("' + networkLink.link.href + '")', 3.846 + 1000 * networkLink.link.refreshInterval); 3.847 + } else if (networkLink.link.refreshMode === 'onChange') { 3.848 + if (networkLink.link.viewRefreshMode === 'never') { 3.849 + // Load the link just once 3.850 + doc.internals.parser.parse(networkLink.link.href, doc.internals.docSet); 3.851 + } else if (networkLink.link.viewRefreshMode === 'onStop') { 3.852 + // Reload when the map view changes 3.853 + 3.854 + } 3.855 + } 3.856 + } 3.857 + } 3.858 + 3.859 + if (!!doc.bounds) { 3.860 + doc.internals.bounds = doc.internals.bounds || new google.maps.LatLngBounds(); 3.861 + doc.internals.bounds.union(doc.bounds); 3.862 + } 3.863 + if (!!doc.markers || !!doc.groundoverlays || !!doc.gpolylines || !!doc.gpolygons) { 3.864 + doc.internals.parseOnly = false; 3.865 + } 3.866 + 3.867 + if (!doc.internals.parseOnly) { 3.868 + // geoXML3 is not being used only as a real-time parser, so keep the processed documents around 3.869 + if (!docsByUrl[doc.baseUrl]) { 3.870 + docs.push(doc); 3.871 + docsByUrl[doc.baseUrl] = doc; 3.872 + } 3.873 + else { 3.874 + // internal replacement, which keeps the same memory ref loc in docs and docsByUrl 3.875 + for (var i in docsByUrl[doc.baseUrl]) { 3.876 + docsByUrl[doc.baseUrl][i] = doc[i]; 3.877 + } 3.878 + } 3.879 + } 3.880 + 3.881 + doc.internals.remaining--; 3.882 + if (doc.internals.remaining === 0) { 3.883 + // We're done processing this set of KML documents 3.884 + // Options that get invoked after parsing completes 3.885 + if (parserOptions.zoom && !!doc.internals.bounds && 3.886 + !doc.internals.bounds.isEmpty() && !!parserOptions.map) { 3.887 + parserOptions.map.fitBounds(doc.internals.bounds); 3.888 + } 3.889 + if (parserOptions.afterParse) { 3.890 + parserOptions.afterParse(doc.internals.docSet); 3.891 + } 3.892 + } 3.893 + }; 3.894 + 3.895 + var kmlColor = function (kmlIn, colorMode) { 3.896 + var kmlColor = {}; 3.897 + kmlIn = kmlIn || 'ffffffff'; // white (KML 2.2 default) 3.898 + 3.899 + var aa = kmlIn.substr(0,2); 3.900 + var bb = kmlIn.substr(2,2); 3.901 + var gg = kmlIn.substr(4,2); 3.902 + var rr = kmlIn.substr(6,2); 3.903 + 3.904 + kmlColor.opacity = parseInt(aa, 16) / 256; 3.905 + kmlColor.color = (colorMode === 'random') ? randomColor(rr, gg, bb) : '#' + rr + gg + bb; 3.906 + return kmlColor; 3.907 + }; 3.908 + 3.909 + // Implemented per KML 2.2 <ColorStyle> specs 3.910 + var randomColor = function(rr, gg, bb) { 3.911 + var col = { rr: rr, gg: gg, bb: bb }; 3.912 + for (var k in col) { 3.913 + var v = col[k]; 3.914 + if (v == null) v = 'ff'; 3.915 + 3.916 + // RGB values are limiters for random numbers (ie: 7f would be a random value between 0 and 7f) 3.917 + v = Math.round(Math.random() * parseInt(rr, 16)).toString(16); 3.918 + if (v.length === 1) v = '0' + v; 3.919 + col[k] = v; 3.920 + } 3.921 + 3.922 + return '#' + col.rr + col.gg + col.bb; 3.923 + }; 3.924 + 3.925 + var processStyleID = function (style) { 3.926 + var icon = style.icon; 3.927 + if (!icon.href) return; 3.928 + 3.929 + if (icon.img && !icon.img.complete && (icon.dim.w < 0) && (icon.dim.h < 0) ) { 3.930 + // we're still waiting on the image loading (probably because we've been blocking since the declaration) 3.931 + // so, let's queue this function on the onload stack 3.932 + icon.markerBacklog = []; 3.933 + icon.img.onload = function() { 3.934 + if (icon.dim.w < 0 || icon.dim.h < 0) { 3.935 + icon.dim.w = this.width; 3.936 + icon.dim.h = this.height; 3.937 + } 3.938 + processStyleID(style); 3.939 + 3.940 + // we will undoubtedly get some createMarker queuing, so set this up in advance 3.941 + for (var i = 0; i < icon.markerBacklog.length; i++) { 3.942 + var p = icon.markerBacklog[i][0]; 3.943 + var d = icon.markerBacklog[i][1]; 3.944 + createMarker(p, d); 3.945 + if (p.marker) p.marker.active = true; 3.946 + } 3.947 + delete icon.markerBacklog; 3.948 + }; 3.949 + return; 3.950 + } 3.951 + else if (icon.dim.w < 0 || icon.dim.h < 0) { 3.952 + if (icon.img && icon.img.complete) { 3.953 + // sometimes the file is already cached and it never calls onLoad 3.954 + icon.dim.w = icon.img.width; 3.955 + icon.dim.h = icon.img.height; 3.956 + } 3.957 + else { 3.958 + // settle for a default of 32x32 3.959 + icon.dim.whGuess = true; 3.960 + icon.dim.w = 32; 3.961 + icon.dim.h = 32; 3.962 + } 3.963 + } 3.964 + 3.965 + // pre-scaled variables 3.966 + var rnd = Math.round; 3.967 + var scaled = { 3.968 + x: icon.dim.x * icon.scale, 3.969 + y: icon.dim.y * icon.scale, 3.970 + w: icon.dim.w * icon.scale, 3.971 + h: icon.dim.h * icon.scale, 3.972 + aX: icon.hotSpot.x * icon.scale, 3.973 + aY: icon.hotSpot.y * icon.scale, 3.974 + iW: (icon.img ? icon.img.width : icon.dim.w) * icon.scale, 3.975 + iH: (icon.img ? icon.img.height : icon.dim.h) * icon.scale 3.976 + }; 3.977 + 3.978 + // Figure out the anchor spot 3.979 + var aX, aY; 3.980 + switch (icon.hotSpot.xunits) { 3.981 + case 'fraction': aX = rnd(scaled.aX * icon.dim.w); break; 3.982 + case 'insetPixels': aX = rnd(icon.dim.w * icon.scale - scaled.aX); break; 3.983 + default: aX = rnd(scaled.aX); break; // already pixels 3.984 + } 3.985 + aY = rnd( ((icon.hotSpot.yunits === 'fraction') ? icon.dim.h : 1) * scaled.aY ); // insetPixels Y = pixels Y 3.986 + var iconAnchor = new google.maps.Point(aX, aY); 3.987 + 3.988 + // Sizes 3.989 + // (NOTE: Scale is applied to entire image, not just the section of the icon palette.) 3.990 + var iconSize = icon.dim.whGuess ? null : new google.maps.Size(rnd(scaled.w), rnd(scaled.h)); 3.991 + var iconScale = icon.scale == 1.0 ? null : 3.992 + icon.dim.whGuess ? new google.maps.Size(rnd(scaled.w), rnd(scaled.h)) 3.993 + : new google.maps.Size(rnd(scaled.iW), rnd(scaled.iH)); 3.994 + var iconOrigin = new google.maps.Point(rnd(scaled.x), rnd(scaled.y)); 3.995 + 3.996 + // Detect images buried in KMZ files (and use a base64 encoded URL) 3.997 + if (kmzMetaData[icon.url]) icon.url = kmzMetaData[icon.url].dataUrl; 3.998 + 3.999 + // Init the style object with the KML icon 3.1000 + icon.marker = new google.maps.MarkerImage( 3.1001 + icon.url, // url 3.1002 + iconSize, // size 3.1003 + iconOrigin, // origin 3.1004 + iconAnchor, // anchor 3.1005 + iconScale // scaledSize 3.1006 + ); 3.1007 + 3.1008 + // Look for a predictable shadow 3.1009 + var stdRegEx = /\/(red|blue|green|yellow|lightblue|purple|pink|orange)(-dot)?\.png/; 3.1010 + var shadowSize = new google.maps.Size(59, 32); 3.1011 + var shadowPoint = new google.maps.Point(16, 32); 3.1012 + if (stdRegEx.test(icon.href)) { 3.1013 + // A standard GMap-style marker icon 3.1014 + icon.shadow = new google.maps.MarkerImage( 3.1015 + 'http://maps.google.com/mapfiles/ms/micons/msmarker.shadow.png', // url 3.1016 + shadowSize, // size 3.1017 + null, // origin 3.1018 + shadowPoint, // anchor 3.1019 + shadowSize // scaledSize 3.1020 + ); 3.1021 + } else if (icon.href.indexOf('-pushpin.png') > -1) { 3.1022 + // Pushpin marker icon 3.1023 + icon.shadow = new google.maps.MarkerImage( 3.1024 + 'http://maps.google.com/mapfiles/ms/micons/pushpin_shadow.png', // url 3.1025 + shadowSize, // size 3.1026 + null, // origin 3.1027 + shadowPoint, // anchor 3.1028 + shadowSize // scaledSize 3.1029 + ); 3.1030 + } /* else { 3.1031 + // Other MyMaps KML standard icon 3.1032 + icon.shadow = new google.maps.MarkerImage( 3.1033 + icon.href.replace('.png', '.shadow.png'), // url 3.1034 + shadowSize, // size 3.1035 + null, // origin 3.1036 + anchorPoint, // anchor 3.1037 + shadowSize // scaledSize 3.1038 + ); 3.1039 + } */ 3.1040 + } 3.1041 + 3.1042 + var processStyles = function (doc) { 3.1043 + for (var styleID in doc.styles) { 3.1044 + processStyleID(doc.styles[styleID]); 3.1045 + } 3.1046 + }; 3.1047 + 3.1048 + var createMarker = function (placemark, doc) { 3.1049 + // create a Marker to the map from a placemark KML object 3.1050 + var icon = placemark.style.icon; 3.1051 + 3.1052 + if ( !icon.marker && icon.img ) { 3.1053 + // yay, single point of failure is holding up multiple markers... 3.1054 + icon.markerBacklog = icon.markerBacklog || []; 3.1055 + icon.markerBacklog.push([placemark, doc]); 3.1056 + return; 3.1057 + } 3.1058 + 3.1059 + // Load basic marker properties 3.1060 + var markerOptions = geoXML3.combineOptions(parserOptions.markerOptions, { 3.1061 + map: parserOptions.map, 3.1062 + position: new google.maps.LatLng(placemark.Point.coordinates[0].lat, placemark.Point.coordinates[0].lng), 3.1063 + title: placemark.name, 3.1064 + zIndex: Math.round(placemark.Point.coordinates[0].lat * -100000)<<5, 3.1065 + icon: icon.marker, 3.1066 + shadow: icon.shadow, 3.1067 + flat: !icon.shadow, 3.1068 + visible: placemark.visibility 3.1069 + }); 3.1070 + 3.1071 + // Create the marker on the map 3.1072 + var marker = new google.maps.Marker(markerOptions); 3.1073 + if (!!doc) doc.markers.push(marker); 3.1074 + 3.1075 + // Set up and create the infowindow if it is not suppressed 3.1076 + createInfoWindow(placemark, doc, marker); 3.1077 + placemark.marker = marker; 3.1078 + return marker; 3.1079 + }; 3.1080 + 3.1081 + var createOverlay = function (groundOverlay, doc) { 3.1082 + // Add a ProjectedOverlay to the map from a groundOverlay KML object 3.1083 + 3.1084 + if (!window.ProjectedOverlay) { 3.1085 + throw 'geoXML3 error: ProjectedOverlay not found while rendering GroundOverlay from KML'; 3.1086 + } 3.1087 + 3.1088 + var bounds = new google.maps.LatLngBounds( 3.1089 + new google.maps.LatLng(groundOverlay.latLonBox.south, groundOverlay.latLonBox.west), 3.1090 + new google.maps.LatLng(groundOverlay.latLonBox.north, groundOverlay.latLonBox.east) 3.1091 + ); 3.1092 + var overlayOptions = geoXML3.combineOptions(parserOptions.overlayOptions, {percentOpacity: groundOverlay.opacity*100}); 3.1093 + var overlay = new ProjectedOverlay(parserOptions.map, groundOverlay.icon.href, bounds, overlayOptions); 3.1094 + 3.1095 + if (!!doc) { 3.1096 + doc.ggroundoverlays = doc.ggroundoverlays || []; 3.1097 + doc.ggroundoverlays.push(overlay); 3.1098 + } 3.1099 + 3.1100 + return overlay; 3.1101 + }; 3.1102 + 3.1103 + // Create Polyline 3.1104 + var createPolyline = function(placemark, doc) { 3.1105 + var path = []; 3.1106 + for (var j=0; j<placemark.LineString.length; j++) { 3.1107 + var coords = placemark.LineString[j].coordinates; 3.1108 + var bounds = new google.maps.LatLngBounds(); 3.1109 + for (var i=0;i<coords.length;i++) { 3.1110 + var pt = new google.maps.LatLng(coords[i].lat, coords[i].lng); 3.1111 + path.push(pt); 3.1112 + bounds.extend(pt); 3.1113 + } 3.1114 + } 3.1115 + // point to open the infowindow if triggered 3.1116 + var point = path[Math.floor(path.length/2)]; 3.1117 + // Load basic polyline properties 3.1118 + var kmlStrokeColor = kmlColor(placemark.style.line.color, placemark.style.line.colorMode); 3.1119 + var polyOptions = geoXML3.combineOptions(parserOptions.polylineOptions, { 3.1120 + map: parserOptions.map, 3.1121 + path: path, 3.1122 + strokeColor: kmlStrokeColor.color, 3.1123 + strokeWeight: placemark.style.line.width, 3.1124 + strokeOpacity: kmlStrokeColor.opacity, 3.1125 + title: placemark.name, 3.1126 + visible: placemark.visibility 3.1127 + }); 3.1128 + var p = new google.maps.Polyline(polyOptions); 3.1129 + p.bounds = bounds; 3.1130 + 3.1131 + // setup and create the infoWindow if it is not suppressed 3.1132 + createInfoWindow(placemark, doc, p); 3.1133 + if (!!doc) doc.gpolylines.push(p); 3.1134 + placemark.polyline = p; 3.1135 + return p; 3.1136 + } 3.1137 + 3.1138 + // Create Polygon 3.1139 + var createPolygon = function(placemark, doc) { 3.1140 + var bounds = new google.maps.LatLngBounds(); 3.1141 + var pathsLength = 0; 3.1142 + var paths = []; 3.1143 + for (var polygonPart=0;polygonPart<placemark.Polygon.length;polygonPart++) { 3.1144 + for (var j=0; j<placemark.Polygon[polygonPart].outerBoundaryIs.length; j++) { 3.1145 + var coords = placemark.Polygon[polygonPart].outerBoundaryIs[j].coordinates; 3.1146 + var path = []; 3.1147 + for (var i=0;i<coords.length;i++) { 3.1148 + var pt = new google.maps.LatLng(coords[i].lat, coords[i].lng); 3.1149 + path.push(pt); 3.1150 + bounds.extend(pt); 3.1151 + } 3.1152 + paths.push(path); 3.1153 + pathsLength += path.length; 3.1154 + } 3.1155 + for (var j=0; j<placemark.Polygon[polygonPart].innerBoundaryIs.length; j++) { 3.1156 + var coords = placemark.Polygon[polygonPart].innerBoundaryIs[j].coordinates; 3.1157 + var path = []; 3.1158 + for (var i=0;i<coords.length;i++) { 3.1159 + var pt = new google.maps.LatLng(coords[i].lat, coords[i].lng); 3.1160 + path.push(pt); 3.1161 + bounds.extend(pt); 3.1162 + } 3.1163 + paths.push(path); 3.1164 + pathsLength += path.length; 3.1165 + } 3.1166 + } 3.1167 + 3.1168 + // Load basic polygon properties 3.1169 + var kmlStrokeColor = kmlColor(placemark.style.line.color, placemark.style.line.colorMode); 3.1170 + var kmlFillColor = kmlColor(placemark.style.poly.color, placemark.style.poly.colorMode); 3.1171 + if (!placemark.style.poly.fill) kmlFillColor.opacity = 0.0; 3.1172 + var strokeWeight = placemark.style.line.width; 3.1173 + if (!placemark.style.poly.outline) { 3.1174 + strokeWeight = 0; 3.1175 + kmlStrokeColor.opacity = 0.0; 3.1176 + } 3.1177 + var polyOptions = geoXML3.combineOptions(parserOptions.polygonOptions, { 3.1178 + map: parserOptions.map, 3.1179 + paths: paths, 3.1180 + title: placemark.name, 3.1181 + strokeColor: kmlStrokeColor.color, 3.1182 + strokeWeight: strokeWeight, 3.1183 + strokeOpacity: kmlStrokeColor.opacity, 3.1184 + fillColor: kmlFillColor.color, 3.1185 + fillOpacity: kmlFillColor.opacity, 3.1186 + visible: placemark.visibility 3.1187 + }); 3.1188 + var p = new google.maps.Polygon(polyOptions); 3.1189 + p.bounds = bounds; 3.1190 + 3.1191 + createInfoWindow(placemark, doc, p); 3.1192 + if (!!doc) doc.gpolygons.push(p); 3.1193 + placemark.polygon = p; 3.1194 + return p; 3.1195 + } 3.1196 + 3.1197 + var createInfoWindow = function(placemark, doc, gObj) { 3.1198 + var bStyle = placemark.style.balloon; 3.1199 + var vars = placemark.vars; 3.1200 + 3.1201 + if (!placemark.balloonVisibility || bStyle.displayMode === 'hide') return; 3.1202 + 3.1203 + // define geDirections 3.1204 + if (placemark.latlng) { 3.1205 + vars.directions.push('sll=' + placemark.latlng.toUrlValue()); 3.1206 + 3.1207 + var url = 'http://maps.google.com/maps?' + vars.directions.join('&'); 3.1208 + var address = encodeURIComponent( vars.val.address || placemark.latlng.toUrlValue() ).replace(/\%20/g, '+'); 3.1209 + 3.1210 + vars.val.geDirections = '<a href="' + url + '&daddr=' + address + '" target=_blank>To Here</a> - <a href="' + url + '&saddr=' + address + '" target=_blank>From Here</a>'; 3.1211 + } 3.1212 + else vars.val.geDirections = ''; 3.1213 + 3.1214 + // add in the variables 3.1215 + var iwText = bStyle.text.replace(/\$\[(\w+(\/displayName)?)\]/g, function(txt, n, dn) { return dn ? vars.display[n] : vars.val[n]; }); 3.1216 + var classTxt = 'geoxml3_infowindow geoxml3_style_' + placemark.styleID; 3.1217 + 3.1218 + // color styles 3.1219 + var styleArr = []; 3.1220 + if (bStyle.bgColor != 'ffffffff') styleArr.push('background: ' + kmlColor(bStyle.bgColor ).color + ';'); 3.1221 + if (bStyle.textColor != 'ff000000') styleArr.push('color: ' + kmlColor(bStyle.textColor).color + ';'); 3.1222 + var styleProp = styleArr.length ? ' style="' + styleArr.join(' ') + '"' : ''; 3.1223 + 3.1224 + var infoWindowOptions = geoXML3.combineOptions(parserOptions.infoWindowOptions, { 3.1225 + content: '<div class="' + classTxt + '"' + styleProp + '>' + iwText + '</div>', 3.1226 + pixelOffset: new google.maps.Size(0, 2) 3.1227 + }); 3.1228 + 3.1229 + gObj.infoWindow = parserOptions.infoWindow || new google.maps.InfoWindow(infoWindowOptions); 3.1230 + gObj.infoWindowOptions = infoWindowOptions; 3.1231 + 3.1232 + // Info Window-opening event handler 3.1233 + google.maps.event.addListener(gObj, 'click', function(e) { 3.1234 + var iW = this.infoWindow; 3.1235 + iW.close(); 3.1236 + iW.setOptions(this.infoWindowOptions); 3.1237 + 3.1238 + if (e && e.latLng) iW.setPosition(e.latLng); 3.1239 + else if (this.bounds) iW.setPosition(this.bounds.getCenter()); 3.1240 + 3.1241 + iW.setContent("<div id='geoxml3_infowindow'>"+iW.getContent()+"</div>"); 3.1242 + google.maps.event.addListenerOnce(iW, "domready", function() { 3.1243 + var node = document.getElementById('geoxml3_infowindow'); 3.1244 + var imgArray = node.getElementsByTagName('img'); 3.1245 + for (var i = 0; i < imgArray.length; i++) 3.1246 + { 3.1247 + var imgUrlIE = imgArray[i].getAttribute("src"); 3.1248 + var imgUrl = cleanURL(doc.baseDir, imgUrlIE); 3.1249 + 3.1250 + if (kmzMetaData[imgUrl]) { 3.1251 + imgArray[i].src = kmzMetaData[imgUrl].dataUrl; 3.1252 + } else if (kmzMetaData[imgUrlIE]) { 3.1253 + imgArray[i].src = kmzMetaData[imgUrlIE].dataUrl; 3.1254 + } 3.1255 + } 3.1256 + }); 3.1257 + iW.open(this.map, this.bounds ? null : this); 3.1258 + }); 3.1259 + 3.1260 + } 3.1261 + 3.1262 + return { 3.1263 + // Expose some properties and methods 3.1264 + 3.1265 + options: parserOptions, 3.1266 + docs: docs, 3.1267 + docsByUrl: docsByUrl, 3.1268 + kmzMetaData: kmzMetaData, 3.1269 + 3.1270 + parse: parse, 3.1271 + parseKmlString: parseKmlString, 3.1272 + hideDocument: hideDocument, 3.1273 + showDocument: showDocument, 3.1274 + processStyles: processStyles, 3.1275 + createMarker: createMarker, 3.1276 + createOverlay: createOverlay, 3.1277 + createPolyline: createPolyline, 3.1278 + createPolygon: createPolygon 3.1279 + }; 3.1280 +}; 3.1281 +// End of KML Parser 3.1282 + 3.1283 +// Helper objects and functions 3.1284 +geoXML3.getOpacity = function (kmlColor) { 3.1285 + // Extract opacity encoded in a KML color value. Returns a number between 0 and 1. 3.1286 + if (!!kmlColor && 3.1287 + (kmlColor !== '') && 3.1288 + (kmlColor.length == 8)) { 3.1289 + var transparency = parseInt(kmlColor.substr(0, 2), 16); 3.1290 + return transparency / 255; 3.1291 + } else { 3.1292 + return 1; 3.1293 + } 3.1294 +}; 3.1295 + 3.1296 +// Log a message to the debugging console, if one exists 3.1297 +geoXML3.log = function(msg) { 3.1298 + if (!!window.console) { 3.1299 + console.log(msg); 3.1300 + } else { alert("log:"+msg); } 3.1301 +}; 3.1302 + 3.1303 +/** 3.1304 + * Creates a new parserOptions object. 3.1305 + * @class GeoXML3 parser options. 3.1306 + * @param {Object} overrides Any options you want to declare outside of the defaults should be included here. 3.1307 + * @property {google.maps.Map} map The API map on which geo objects should be rendered. 3.1308 + * @property {google.maps.MarkerOptions} markerOptions If the parser is adding Markers to the map itself, any options specified here will be applied to them. 3.1309 + * @property {google.maps.InfoWindowOptions} infoWindowOptions If the parser is adding Markers to the map itself, any options specified here will be applied to their attached InfoWindows. 3.1310 + * @property {ProjectedOverlay.options} overlayOptions If the parser is adding ProjectedOverlays to the map itself, any options specified here will be applied to them. 3.1311 + */ 3.1312 +geoXML3.parserOptions = function (overrides) { 3.1313 + this.map = null, 3.1314 + /** If true, the parser will automatically move the map to a best-fit of the geodata after parsing of a KML document completes. 3.1315 + * @type Boolean 3.1316 + * @default true 3.1317 + */ 3.1318 + this.zoom = true, 3.1319 + /**#@+ @type Boolean 3.1320 + * @default false */ 3.1321 + /** If true, only a single Marker created by the parser will be able to have its InfoWindow open at once (simulating the behavior of GMaps API v2). */ 3.1322 + this.singleInfoWindow = false, 3.1323 + /** If true, suppresses the rendering of info windows. */ 3.1324 + this.suppressInfoWindows = false, 3.1325 + /** 3.1326 + * Control whether to process styles now or later. 3.1327 + * 3.1328 + * <p>By default, the parser only processes KML <Style> elements into their GMaps equivalents 3.1329 + * if it will be creating its own Markers (the createMarker option is null). Setting this option 3.1330 + * to true will force such processing to happen anyway, useful if you're going to be calling parser.createMarker 3.1331 + * yourself later. OTOH, leaving this option false removes runtime dependency on the GMaps API, enabling 3.1332 + * the use of geoXML3 as a standalone KML parser.</p> 3.1333 + */ 3.1334 + this.processStyles = false, 3.1335 + /**#@-*/ 3.1336 + 3.1337 + this.markerOptions = {}, 3.1338 + this.infoWindowOptions = {}, 3.1339 + this.overlayOptions = {}, 3.1340 + 3.1341 + /**#@+ @event */ 3.1342 + /** This function will be called when parsing of a KML document is complete. 3.1343 + * @param {geoXML3.parser#docs} doc Parsed KML data. */ 3.1344 + this.afterParse = null, 3.1345 + /** This function will be called when parsing of a KML document is complete. 3.1346 + * @param {geoXML3.parser#docs} doc Parsed KML data. */ 3.1347 + this.failedParse = null, 3.1348 + /** 3.1349 + * If supplied, this function will be called once for each marker <Placemark> in the KML document, instead of the parser adding its own Marker to the map. 3.1350 + * @param {geoXML3.parser.render#placemark} placemark Placemark object. 3.1351 + * @param {geoXML3.parser#docs} doc Parsed KML data. 3.1352 + */ 3.1353 + this.createMarker = null, 3.1354 + /** 3.1355 + * If supplied, this function will be called once for each <GroundOverlay> in the KML document, instead of the parser adding its own ProjectedOverlay to the map. 3.1356 + * @param {geoXML3.parser.render#groundOverlay} groundOverlay GroundOverlay object. 3.1357 + * @param {geoXML3.parser#docs} doc Parsed KML data. 3.1358 + */ 3.1359 + this.createOverlay = null 3.1360 + /**#@-*/ 3.1361 + 3.1362 + if (overrides) { 3.1363 + for (var prop in overrides) { 3.1364 + if (overrides.hasOwnProperty(prop)) this[prop] = overrides[prop]; 3.1365 + } 3.1366 + } 3.1367 + return this; 3.1368 +}; 3.1369 + 3.1370 +/** 3.1371 + * Combine two options objects: a set of default values and a set of override values. 3.1372 + * 3.1373 + * @deprecated This has been replaced with {@link geoXML3.parserOptions#combineOptions}. 3.1374 + * @param {geoXML3.parserOptions|Object} overrides Override values. 3.1375 + * @param {geoXML3.parserOptions|Object} defaults Default values. 3.1376 + * @return {geoXML3.parserOptions} Combined result. 3.1377 + */ 3.1378 +geoXML3.combineOptions = function (overrides, defaults) { 3.1379 + var result = {}; 3.1380 + if (!!overrides) { 3.1381 + for (var prop in overrides) { 3.1382 + if (overrides.hasOwnProperty(prop)) result[prop] = overrides[prop]; 3.1383 + } 3.1384 + } 3.1385 + if (!!defaults) { 3.1386 + for (prop in defaults) { 3.1387 + if (defaults.hasOwnProperty(prop) && result[prop] === undefined) result[prop] = defaults[prop]; 3.1388 + } 3.1389 + } 3.1390 + return result; 3.1391 +}; 3.1392 + 3.1393 +/** 3.1394 + * Combine two options objects: a set of default values and a set of override values. 3.1395 + * 3.1396 + * @function 3.1397 + * @param {geoXML3.parserOptions|Object} overrides Override values. 3.1398 + * @param {geoXML3.parserOptions|Object} defaults Default values. 3.1399 + * @return {geoXML3.parserOptions} Combined result. 3.1400 + */ 3.1401 +geoXML3.parserOptions.prototype.combineOptions = geoXML3.combineOptions; 3.1402 + 3.1403 +// Retrieve an XML document from url and pass it to callback as a DOM document 3.1404 +geoXML3.fetchers = []; 3.1405 + 3.1406 +/** 3.1407 + * Parses a XML string. 3.1408 + * 3.1409 + * <p>Parses the given XML string and returns the parsed document in a 3.1410 + * DOM data structure. This function will return an empty DOM node if 3.1411 + * XML parsing is not supported in this browser.</p> 3.1412 + * 3.1413 + * @param {String} str XML string. 3.1414 + * @return {Element|Document} DOM. 3.1415 + */ 3.1416 +geoXML3.xmlParse = function (str) { 3.1417 + if (typeof ActiveXObject != 'undefined' && typeof GetObject != 'undefined') { 3.1418 + var doc = new ActiveXObject('Microsoft.XMLDOM'); 3.1419 + doc.loadXML(str); 3.1420 + return doc; 3.1421 + } 3.1422 + 3.1423 + if (typeof DOMParser != 'undefined') { 3.1424 + return (new DOMParser()).parseFromString(str, 'text/xml'); 3.1425 + } 3.1426 + 3.1427 + return createElement('div', null); 3.1428 +} 3.1429 + 3.1430 +/** 3.1431 + * Fetches a XML document. 3.1432 + * 3.1433 + * <p>Fetches/parses the given XML URL and passes the parsed document (in a 3.1434 + * DOM data structure) to the given callback. Documents are downloaded 3.1435 + * and parsed asynchronously.</p> 3.1436 + * 3.1437 + * @param {String} url URL of XML document. Must be uncompressed XML only. 3.1438 + * @param {Function(Document)} callback Function to call when the document is processed. 3.1439 + */ 3.1440 +geoXML3.fetchXML = function (url, callback) { 3.1441 + function timeoutHandler() { callback(); }; 3.1442 + 3.1443 + var xhrFetcher = new Object(); 3.1444 + if (!!geoXML3.fetchers.length) xhrFetcher = geoXML3.fetchers.pop(); 3.1445 + else if (!!window.XMLHttpRequest) xhrFetcher.fetcher = new window.XMLHttpRequest(); // Most browsers 3.1446 + else if (!!window.ActiveXObject) { // Some IE 3.1447 + // the many versions of IE's XML fetchers 3.1448 + var AXOs = [ 3.1449 + 'MSXML2.XMLHTTP.6.0', 3.1450 + 'MSXML2.XMLHTTP.5.0', 3.1451 + 'MSXML2.XMLHTTP.4.0', 3.1452 + 'MSXML2.XMLHTTP.3.0', 3.1453 + 'MSXML2.XMLHTTP', 3.1454 + 'Microsoft.XMLHTTP', 3.1455 + 'MSXML.XMLHTTP' 3.1456 + ]; 3.1457 + for (var i = 0; i < AXOs.length; i++) { 3.1458 + try { xhrFetcher.fetcher = new ActiveXObject(AXOs[i]); break; } 3.1459 + catch(e) { continue; } 3.1460 + } 3.1461 + if (!xhrFetcher.fetcher) { 3.1462 + geoXML3.log('Unable to create XHR object'); 3.1463 + callback(null); 3.1464 + return null; 3.1465 + } 3.1466 + } 3.1467 + 3.1468 + if (!!xhrFetcher.fetcher.overrideMimeType) xhrFetcher.fetcher.overrideMimeType('text/xml'); 3.1469 + xhrFetcher.fetcher.open('GET', url, true); 3.1470 + xhrFetcher.fetcher.onreadystatechange = function () { 3.1471 + if (xhrFetcher.fetcher.readyState === 4) { 3.1472 + // Retrieval complete 3.1473 + if (!!xhrFetcher.xhrtimeout) clearTimeout(xhrFetcher.xhrtimeout); 3.1474 + if (xhrFetcher.fetcher.status >= 400) { 3.1475 + geoXML3.log('HTTP error ' + xhrFetcher.fetcher.status + ' retrieving ' + url); 3.1476 + callback(); 3.1477 + } 3.1478 + // Returned successfully 3.1479 + else { 3.1480 + if (xhrFetcher.fetcher.responseXML) { 3.1481 + // Sometimes IE will get the data, but won't bother loading it as an XML doc 3.1482 + var xmlDoc = xhrFetcher.fetcher.responseXML; 3.1483 + if (xmlDoc && !xmlDoc.documentElement && !xmlDoc.ownerElement) xmlDoc.loadXML(xhrFetcher.fetcher.responseText); 3.1484 + callback(xmlDoc); 3.1485 + } else // handle valid xml sent with wrong MIME type 3.1486 + callback(geoXML3.xmlParse(xhrFetcher.fetcher.responseText)); 3.1487 + } 3.1488 + 3.1489 + // We're done with this fetcher object 3.1490 + geoXML3.fetchers.push(xhrFetcher); 3.1491 + } 3.1492 + }; 3.1493 + 3.1494 + xhrFetcher.xhrtimeout = setTimeout(timeoutHandler, 60000); 3.1495 + xhrFetcher.fetcher.send(null); 3.1496 + return null; 3.1497 +}; 3.1498 + 3.1499 +var IEversion = function() { 3.1500 + // http://msdn.microsoft.com/workshop/author/dhtml/overview/browserdetection.asp 3.1501 + // Returns the version of Internet Explorer or a -1 3.1502 + // (indicating the use of another browser). 3.1503 + var rv = -1; // Return value assumes failure 3.1504 + if (navigator.appName == 'Microsoft Internet Explorer') { 3.1505 + var ua = navigator.userAgent; 3.1506 + var re = new RegExp("MSIE ([0-9]{1,}[\.0-9]{0,})"); 3.1507 + if (re.exec(ua) != null) { 3.1508 + rv = parseFloat( RegExp.$1 ); 3.1509 + } 3.1510 + } 3.1511 + return rv; 3.1512 +}; 3.1513 + 3.1514 +/** 3.1515 + * Fetches a KMZ document. 3.1516 + * 3.1517 + * <p>Fetches/parses the given ZIP URL, parses each image file, and passes 3.1518 + * the parsed KML document to the given callback. Documents are downloaded 3.1519 + * and parsed asynchronously, though the KML file is always passed after the 3.1520 + * images have been processed, in case the callback requires the image data.</p> 3.1521 + * 3.1522 + * @requires ZipFile.complete.js 3.1523 + * @param {String} url URL of KMZ document. Must be a valid KMZ/ZIP archive. 3.1524 + * @param {Function(Document)} callback Function to call when the document is processed. 3.1525 + * @param {geoXML3.parser} parser A geoXML3.parser object. This is used to populate the KMZ image data. 3.1526 + * @author Brendan Byrd 3.1527 + * @see http://code.google.com/apis/kml/documentation/kmzarchives.html 3.1528 + */ 3.1529 +geoXML3.fetchZIP = function (url, callback, parser) { 3.1530 + // Just need a single 'new' declaration with a really long function... 3.1531 + var zipFile = new ZipFile(url, function (zip) { 3.1532 + // Retrieval complete 3.1533 + 3.1534 + // Check for ERRORs in zip.status 3.1535 + for (var i = 0; i < zip.status.length; i++) { 3.1536 + var msg = zip.status[i]; 3.1537 + if (msg.indexOf("ERROR") == 0) { 3.1538 + geoXML3.log('HTTP/ZIP error retrieving ' + url + ': ' + msg); 3.1539 + callback(); 3.1540 + return; 3.1541 + } 3.1542 + else if (msg.indexOf("WARNING") == 0) { // non-fatal, but still might be useful 3.1543 + geoXML3.log('HTTP/ZIP warning retrieving ' + url + ': ' + msg); 3.1544 + } 3.1545 + } 3.1546 + 3.1547 + // Make sure KMZ structure is according to spec (with a single KML file in the root dir) 3.1548 + var KMLCount = 0; 3.1549 + var KML; 3.1550 + for (var i = 0; i < zip.entries.length; i++) { 3.1551 + var name = zip.entries[i].name; 3.1552 + if (!/\.kml$/.test(name)) continue; 3.1553 + 3.1554 + KMLCount++; 3.1555 + if (KMLCount == 1) KML = i; 3.1556 + else { 3.1557 + geoXML3.log('KMZ warning retrieving ' + url + ': found extra KML "' + name + '" in KMZ; discarding...'); 3.1558 + } 3.1559 + } 3.1560 + 3.1561 + // Returned successfully, but still needs extracting 3.1562 + var baseUrl = cleanURL(defileURL(url), url) + '/'; 3.1563 + var kmlProcessing = { // this is an object just so it gets passed properly 3.1564 + timer: null, 3.1565 + extractLeft: 0, 3.1566 + timerCalls: 0 3.1567 + }; 3.1568 + var extractCb = function(entry, entryContent) { 3.1569 + var mdUrl = cleanURL(baseUrl, entry.name); 3.1570 + var ext = entry.name.substring(entry.name.lastIndexOf(".") + 1).toLowerCase(); 3.1571 + kmlProcessing.extractLeft--; 3.1572 + 3.1573 + if ((typeof entryContent.description == "string") && (entryContent.name == "Error")) { 3.1574 + geoXML3.log('KMZ error extracting ' + mdUrl + ': ' + entryContent.description); 3.1575 + callback(); 3.1576 + return; 3.1577 + } 3.1578 + 3.1579 + // MIME types that can be used in KML 3.1580 + var mime; 3.1581 + if (ext === 'jpg') ext = 'jpeg'; 3.1582 + if (/^(gif|jpeg|png)$/.test(ext)) mime = 'image/' + ext; 3.1583 + else if (ext === 'mp3') mime = 'audio/mpeg'; 3.1584 + else if (ext === 'm4a') mime = 'audio/mp4'; 3.1585 + else if (ext === 'm4a') mime = 'audio/MP4-LATM'; 3.1586 + else mime = 'application/octet-stream'; 3.1587 + 3.1588 + parser.kmzMetaData[mdUrl] = {}; 3.1589 + parser.kmzMetaData[mdUrl].entry = entry; 3.1590 + // ... 3.1591 + parser.kmzMetaData[mdUrl].dataUrl = 'data:' + mime + ';base64,' + base64Encode(entryContent); 3.1592 + // IE cannot handle GET requests beyond 2071 characters, even if it's an inline image 3.1593 + if (/msie/i.test(navigator.userAgent) && !/opera/i.test(navigator.userAgent)) 3.1594 + { 3.1595 + if (((IEversion() < 8.0) && 3.1596 + (parser.kmzMetaData[mdUrl].dataUrl.length > 2071)) || 3.1597 + ((IEversion < 9.0) && 3.1598 + (parser.kmzMetaData[mdUrl].dataUrl.length > 32767))) { 3.1599 + parser.kmzMetaData[mdUrl].dataUrl = 3.1600 + // this is a simple IE icon; to hint at the problem... 3.1601 + '' + 3.1602 + 'oGDMgzSsiyGCAhCETDPMh5XQCBwYBrNBIKWmg0MCQHj8MJU5IoroYCY6AAAgrDIbbQDGIK6DR5UPhlNo0JAlSUNAiDgH7eNAxEDWAKCQM2AAFheVxYAA0AIkFOJ1gBcQQaUQKKA5w7LpcEBwkJaKMUEQA7'; 3.1603 + } 3.1604 + } 3.1605 + parser.kmzMetaData[internalSrc(entry.name)]=parser.kmzMetaData[mdUrl]; 3.1606 + 3.1607 + }; 3.1608 + var kmlExtractCb = function(entry, entryContent) { 3.1609 + if ((typeof entryContent.description == "string") && (entryContent.name == "Error")) { 3.1610 + geoXML3.log('KMZ error extracting ' + mdUrl + ': ' + entryContent.description); 3.1611 + callback(); 3.1612 + return; 3.1613 + } 3.1614 + 3.1615 + // check to see if the KML is the last file extracted 3.1616 + clearTimeout(kmlProcessing.timer); 3.1617 + if (kmlProcessing.extractLeft <= 1) { 3.1618 + kmlProcessing.extractLeft--; 3.1619 + callback(geoXML3.xmlParse(entryContent)); 3.1620 + return; 3.1621 + } 3.1622 + else { 3.1623 + // KML file isn't last yet; it may need to use those files, so wait a bit (100ms) 3.1624 + kmlProcessing.timerCalls++; 3.1625 + if (kmlProcessing.timerCalls < 100) { 3.1626 + kmlProcessing.timer = setTimeout(function() { kmlExtractCb(entry, entryContent); }, 100); 3.1627 + } 3.1628 + else { 3.1629 + geoXML3.log('KMZ warning extracting ' + url + ': entire ZIP has not been extracted after 10 seconds; running through KML, anyway...'); 3.1630 + kmlProcessing.extractLeft--; 3.1631 + callback(geoXML3.xmlParse(entryContent)); 3.1632 + } 3.1633 + } 3.1634 + return; 3.1635 + }; 3.1636 + for (var i = 0; i < zip.entries.length; i++) { 3.1637 + var entry = zip.entries[i]; 3.1638 + var ext = entry.name.substring(entry.name.lastIndexOf(".") + 1).toLowerCase(); 3.1639 + if (!/^(gif|jpe?g|png|kml)$/.test(ext)) continue; // not going to bother to extract files we don't support 3.1640 + if (ext === "kml" && i != KML) continue; // extra KMLs get discarded 3.1641 + if (!parser && ext != "kml") continue; // cannot store images without a parser object 3.1642 + 3.1643 + // extract asynchronously 3.1644 + kmlProcessing.extractLeft++; 3.1645 + if (ext === "kml") entry.extract(kmlExtractCb); 3.1646 + else entry.extract(extractCb); 3.1647 + } 3.1648 + }); 3.1649 + 3.1650 +}; 3.1651 + 3.1652 +/** 3.1653 + * Extract the text value of a DOM node, with leading and trailing whitespace trimmed. 3.1654 + * 3.1655 + * @param {Element} node XML node/element. 3.1656 + * @param {Any} delVal Default value if the node doesn't exist. 3.1657 + * @return {String|Null} 3.1658 + */ 3.1659 +geoXML3.nodeValue = function(node, defVal) { 3.1660 + var retStr=""; 3.1661 + if (!node) { 3.1662 + return (typeof defVal === 'undefined' || defVal === null) ? null : defVal; 3.1663 + } 3.1664 + if(node.nodeType==3||node.nodeType==4||node.nodeType==2){ 3.1665 + retStr+=node.nodeValue; 3.1666 + }else if(node.nodeType==1||node.nodeType==9||node.nodeType==11){ 3.1667 + for(var i=0;i<node.childNodes.length;++i){ 3.1668 + retStr+=arguments.callee(node.childNodes[i]); 3.1669 + } 3.1670 + } 3.1671 + return retStr; 3.1672 +}; 3.1673 + 3.1674 +/** 3.1675 + * Loosely translate various values of a DOM node to a boolean. 3.1676 + * 3.1677 + * @param {Element} node XML node/element. 3.1678 + * @param {Boolean} delVal Default value if the node doesn't exist. 3.1679 + * @return {Boolean|Null} 3.1680 + */ 3.1681 +geoXML3.getBooleanValue = function(node, defVal) { 3.1682 + var nodeContents = geoXML3.nodeValue(node); 3.1683 + if (nodeContents === null) return defVal || false; 3.1684 + nodeContents = parseInt(nodeContents); 3.1685 + if (isNaN(nodeContents)) return true; 3.1686 + if (nodeContents == 0) return false; 3.1687 + else return true; 3.1688 +} 3.1689 + 3.1690 +/** 3.1691 + * Browser-normalized version of getElementsByTagNameNS. 3.1692 + * 3.1693 + * <p>Required because IE8 doesn't define it.</p> 3.1694 + * 3.1695 + * @param {Element|Document} node DOM object. 3.1696 + * @param {String} namespace Full namespace URL to search against. 3.1697 + * @param {String} tagname XML local tag name. 3.1698 + * @return {Array of Elements} 3.1699 + * @author Brendan Byrd 3.1700 + */ 3.1701 +geoXML3.getElementsByTagNameNS = function(node, namespace, tagname) { 3.1702 + if (node && typeof node.getElementsByTagNameNS != 'undefined') return node.getElementsByTagNameNS(namespace, tagname); 3.1703 + if (!node) return []; 3.1704 + 3.1705 + var root = node.documentElement || node.ownerDocument && node.ownerDocument.documentElement; 3.1706 + if (!root || !root.attributes) return []; 3.1707 + 3.1708 + // search for namespace prefix 3.1709 + for (var i = 0; i < root.attributes.length; i++) { 3.1710 + var attr = root.attributes[i]; 3.1711 + if (attr.prefix === 'xmlns' && attr.nodeValue === namespace) return node.getElementsByTagName(attr.baseName + ':' + tagname); 3.1712 + else if (attr.nodeName === 'xmlns' && attr.nodeValue === namespace) { 3.1713 + // default namespace 3.1714 + if (typeof node.selectNodes != 'undefined') { 3.1715 + // Newer IEs have the SelectionNamespace property that can be used with selectNodes 3.1716 + if (!root.ownerDocument.getProperty('SelectionNamespaces')) 3.1717 + root.ownerDocument.setProperty('SelectionNamespaces', "xmlns:defaultNS='" + namespace + "'"); 3.1718 + return node.selectNodes('.//defaultNS:' + tagname); 3.1719 + } 3.1720 + else { 3.1721 + // Otherwise, you can still try to tack on the 'xmlns' attribute to root 3.1722 + root.setAttribute('xmlns:defaultNS', namespace); 3.1723 + return node.getElementsByTagName('defaultNS:' + tagname); 3.1724 + } 3.1725 + } 3.1726 + } 3.1727 + return geoXML3.getElementsByTagName(node, tagname); // try the unqualified version 3.1728 +}; 3.1729 + 3.1730 +/** 3.1731 + * Browser-normalized version of getElementsByTagName. 3.1732 + * 3.1733 + * <p>Required because MSXML 6.0 will treat this function as a NS-qualified function, 3.1734 + * despite the missing NS parameter.</p> 3.1735 + * 3.1736 + * @param {Element|Document} node DOM object. 3.1737 + * @param {String} tagname XML local tag name. 3.1738 + * @return {Array of Elements} 3.1739 + * @author Brendan Byrd 3.1740 + */ 3.1741 +geoXML3.getElementsByTagName = function(node, tagname) { 3.1742 + if (node && typeof node.getElementsByTagNameNS != 'undefined') return node.getElementsByTagName(tagname); // if it has both functions, it should be accurate 3.1743 +// if (node && typeof node.selectNodes != 'undefined') return node.selectNodes(".//*[local-name()='" + tagname + "']"); 3.1744 + return node.getElementsByTagName(tagname); // hope for the best... 3.1745 +} 3.1746 + 3.1747 +/** 3.1748 + * Turn a directory + relative URL into an absolute one. 3.1749 + * 3.1750 + * @private 3.1751 + * @param {String} d Base directory. 3.1752 + * @param {String} s Relative URL. 3.1753 + * @return {String} Absolute URL. 3.1754 + * @author Brendan Byrd 3.1755 + */ 3.1756 +var toAbsURL = function (d, s) { 3.1757 + var p, f, i; 3.1758 + var h = location.protocol + "://" + location.host; 3.1759 + 3.1760 + if (!s.length) return ''; 3.1761 + if (/^\w+:/.test(s)) return s; 3.1762 + if (s.indexOf('/') == 0) return h + s; 3.1763 + 3.1764 + p = d.replace(/\/[^\/]*$/, ''); 3.1765 + f = s.match(/\.\.\//g); 3.1766 + if (f) { 3.1767 + s = s.substring(f.length * 3); 3.1768 + for (i = f.length; i--;) { p = p.substring(0, p.lastIndexOf('/')); } 3.1769 + } 3.1770 + 3.1771 + return h + p + '/' + s; 3.1772 +} 3.1773 + 3.1774 +var internalSrc = function(src) { 3.1775 + //this gets the full url 3.1776 + var url = document.location.href; 3.1777 + //this removes everything after the last slash in the path 3.1778 + url = url.substring(0,url.lastIndexOf("/") + 1); 3.1779 + var internalPath= url+src; 3.1780 + return internalPath; 3.1781 +} 3.1782 + 3.1783 +/** 3.1784 + * Remove current host from URL 3.1785 + * 3.1786 + * @private 3.1787 + * @param {String} s Absolute or relative URL. 3.1788 + * @return {String} Root-based relative URL. 3.1789 + * @author Brendan Byrd 3.1790 + */ 3.1791 +var dehostURL = function (s) { 3.1792 + var h = location.protocol + "://" + location.host; 3.1793 + h = h.replace(/([\.\\\+\*\?\[\^\]\$\(\)])/g, '\\$1'); // quotemeta 3.1794 + return s.replace(new RegExp('^' + h, 'i'), ''); 3.1795 +} 3.1796 + 3.1797 +/** 3.1798 + * Removes all query strings, #IDs, '../' references, and 3.1799 + * hosts from a URL. 3.1800 + * 3.1801 + * @private 3.1802 + * @param {String} d Base directory. 3.1803 + * @param {String} s Absolute or relative URL. 3.1804 + * @return {String} Root-based relative URL. 3.1805 + * @author Brendan Byrd 3.1806 + */ 3.1807 +var cleanURL = function (d, s) { return dehostURL(toAbsURL(d ? d.split('#')[0].split('?')[0] : defileURL(location.pathname), s ? s.split('#')[0].split('?')[0] : '')); } 3.1808 +/** 3.1809 + * Remove filename from URL 3.1810 + * 3.1811 + * @private 3.1812 + * @param {String} s Relative URL. 3.1813 + * @return {String} Base directory. 3.1814 + * @author Brendan Byrd 3.1815 + */ 3.1816 +var defileURL = function (s) { return s ? s.substr(0, s.lastIndexOf('/') + 1) : '/'; } 3.1817 + 3.1818 + 3.1819 +// Some extra Array subs for ease of use 3.1820 +// http://stackoverflow.com/questions/143847/best-way-to-find-an-item-in-a-javascript-array 3.1821 +Array.prototype.hasObject = ( 3.1822 + !Array.indexOf ? function (obj) { 3.1823 + var l = this.length + 1; 3.1824 + while (l--) { 3.1825 + if (this[l - 1] === obj) return true; 3.1826 + } 3.1827 + return false; 3.1828 + } : function (obj) { 3.1829 + return (this.indexOf(obj) !== -1); 3.1830 + } 3.1831 +); 3.1832 +Array.prototype.hasItemInObj = function (name, item) { 3.1833 + var l = this.length + 1; 3.1834 + while (l--) { 3.1835 + if (this[l - 1][name] === item) return true; 3.1836 + } 3.1837 + return false; 3.1838 +}; 3.1839 +if (!Array.prototype.indexOf) { 3.1840 + Array.prototype.indexOf = function (obj, fromIndex) { 3.1841 + if (fromIndex == null) { 3.1842 + fromIndex = 0; 3.1843 + } else if (fromIndex < 0) { 3.1844 + fromIndex = Math.max(0, this.length + fromIndex); 3.1845 + } 3.1846 + for (var i = fromIndex, j = this.length; i < j; i++) { 3.1847 + if (this[i] === obj) return i; 3.1848 + } 3.1849 + return -1; 3.1850 + }; 3.1851 +} 3.1852 +Array.prototype.indexOfObjWithItem = function (name, item, fromIndex) { 3.1853 + if (fromIndex == null) { 3.1854 + fromIndex = 0; 3.1855 + } else if (fromIndex < 0) { 3.1856 + fromIndex = Math.max(0, this.length + fromIndex); 3.1857 + } 3.1858 + for (var i = fromIndex, j = this.length; i < j; i++) { 3.1859 + if (this[i][name] === item) return i; 3.1860 + } 3.1861 + return -1; 3.1862 +}; 3.1863 + 3.1864 +/** 3.1865 + * Borrowed from jquery.base64.js, with some "Array as input" corrections 3.1866 + * 3.1867 + * @private 3.1868 + * @param {Array of charCodes} input An array of byte ASCII codes (0-255). 3.1869 + * @return {String} A base64-encoded string. 3.1870 + * @author Brendan Byrd 3.1871 + */ 3.1872 +var base64Encode = function(input) { 3.1873 + var keyString = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/="; 3.1874 + var output = ""; 3.1875 + var chr1, chr2, chr3, enc1, enc2, enc3, enc4; 3.1876 + var i = 0; 3.1877 + while (i < input.length) { 3.1878 + chr1 = input[i++]; 3.1879 + chr2 = input[i++]; 3.1880 + chr3 = input[i++]; 3.1881 + enc1 = chr1 >> 2; 3.1882 + enc2 = ((chr1 & 3) << 4) | (chr2 >> 4); 3.1883 + enc3 = ((chr2 & 15) << 2) | (chr3 >> 6); 3.1884 + enc4 = chr3 & 63; 3.1885 + 3.1886 + if (chr2 == undefined) enc3 = enc4 = 64; 3.1887 + else if (chr3 == undefined) enc4 = 64; 3.1888 + 3.1889 + output = output + keyString.charAt(enc1) + keyString.charAt(enc2) + keyString.charAt(enc3) + keyString.charAt(enc4); 3.1890 + } 3.1891 + return output; 3.1892 +};
4.1 --- a/endpoint/WebContent/query.jsp Mon Sep 10 15:38:09 2012 +0300 4.2 +++ b/endpoint/WebContent/query.jsp Fri Sep 14 13:03:24 2012 +0300 4.3 @@ -5,9 +5,7 @@ 4.4 <head> 4.5 <meta name="viewport" content="initial-scale=1.0, user-scalable=no" /> 4.6 <meta http-equiv="Content-Type" content="text/html; charset=UTF-8" /> 4.7 - <link href="http://code.google.com/apis/maps/documentation/javascript/examples/default.css" rel="stylesheet" type="text/css" /> 4.8 - <link rel="stylesheet" href="style.css" type="text/css" /> 4.9 - <script type="text/javascript" src="http://maps.googleapis.com/maps/api/js?sensor=false"></script> 4.10 + <link rel="stylesheet" href="style.css" type="text/css" /> 4.11 <script type="text/javascript"> 4.12 function toggleMe(a) { 4.13 var e = document.getElementById(a); 4.14 @@ -22,21 +20,41 @@ 4.15 return true; 4.16 } 4.17 </script> 4.18 +<% if (request.getAttribute("pathToKML") != null) { 4.19 + if ("map_local".equals(request.getAttribute("handle"))) { %> 4.20 + <script type="text/javascript" src="js/geoxml3-kmz.js"></script> 4.21 + <script type="text/javascript" src="js/ProjectedOverlay.js"></script> 4.22 + <%} %> 4.23 + <link href="http://code.google.com/apis/maps/documentation/javascript/examples/default.css" rel="stylesheet" type="text/css" /> 4.24 + <script type="text/javascript" src="http://maps.googleapis.com/maps/api/js?sensor=false"></script> 4.25 <script type="text/javascript"> 4.26 function initialize() { 4.27 + // center at Brahames 4.28 var brahames = new google.maps.LatLng(37.92253, 23.72275); 4.29 var myOptions = { 4.30 zoom: 11, 4.31 center: brahames, 4.32 mapTypeId: google.maps.MapTypeId.ROADMAP 4.33 }; 4.34 - 4.35 + 4.36 + // get KML filename 4.37 + var kml = '<%=request.getAttribute("pathToKML")%>'; 4.38 + // <%=request.getAttribute("handle")%> 4.39 + // create map 4.40 var map = new google.maps.Map(document.getElementById("map_canvas"), myOptions); 4.41 4.42 - var ctaLayer = new google.maps.KmlLayer('<%=request.getAttribute("pathToKML") != null ? request.getAttribute("pathToKML"):""%>'); 4.43 + // display using geoxml3 4.44 + <%if ("map_local".equals(request.getAttribute("handle"))) { %> 4.45 + var myParser = new geoXML3.parser({map: map}); 4.46 + myParser.parse(kml); 4.47 + 4.48 + <%} else {%> 4.49 + var ctaLayer = new google.maps.KmlLayer(kml); 4.50 ctaLayer.setMap(map); 4.51 + <%}%> 4.52 } 4.53 </script> 4.54 +<%}%> 4.55 <title>TELEIOS: Strabon Endpoint</title> 4.56 </head> 4.57 <body topmargin="0" leftmargin="0" link="#FFFFFF" vlink="#FFFFFF" alink="#FFFFFF" onload="initialize()"> 4.58 @@ -101,6 +119,7 @@ 4.59 <OPTION value="plain">Plain result</OPTION> 4.60 <OPTION value="download">Download</OPTION> 4.61 <OPTION value="map">On a map</OPTION> 4.62 + <OPTION value="map_local">On a map (localhost)</OPTION> 4.63 </SELECT></center> 4.64 </td> 4.65 <td colspan=2> </td>
5.1 --- a/endpoint/pom.xml Mon Sep 10 15:38:09 2012 +0300 5.2 +++ b/endpoint/pom.xml Fri Sep 14 13:03:24 2012 +0300 5.3 @@ -195,6 +195,7 @@ 5.4 <directory>${basedir}/WebContent</directory> 5.5 <includes> 5.6 <include>images/**</include> 5.7 + <include>js/**</include> 5.8 </includes> 5.9 </resource> 5.10 </webResources>
6.1 --- a/endpoint/src/main/java/eu/earthobservatory/org/StrabonEndpoint/QueryBean.java Mon Sep 10 15:38:09 2012 +0300 6.2 +++ b/endpoint/src/main/java/eu/earthobservatory/org/StrabonEndpoint/QueryBean.java Fri Sep 14 13:03:24 2012 +0300 6.3 @@ -208,7 +208,7 @@ 6.4 6.5 out.flush(); 6.6 6.7 - } else if ("map".equals(handle) && 6.8 + } else if (("map".equals(handle) || "map_local".equals(handle)) && 6.9 (queryResultFormat == stSPARQLQueryResultFormat.KML || 6.10 queryResultFormat == stSPARQLQueryResultFormat.KMZ) ) { 6.11 // show map (only valid for KML/KMZ) 6.12 @@ -216,6 +216,9 @@ 6.13 // get dispatcher 6.14 dispatcher = request.getRequestDispatcher("query.jsp"); 6.15 6.16 + // re-assign handle 6.17 + request.setAttribute("handle", handle); 6.18 + 6.19 SecureRandom random = new SecureRandom(); 6.20 String temp = new BigInteger(130, random).toString(32); 6.21