offline-editor-js/lib/tpk/TPKLayer.js

721 lines
36 KiB
JavaScript

/**
* Library for reading an ArcGIS Tile Package (.tpk) file and displaying the tiles
* as a map that can be used both online and offline.
*
* Note: you may have to rename your .tpk file to use .zip in order for it to be recognized.
*
* Author: Andy Gup
* Credits: Mansour Raad for his ArcGIS API for Flex TPKLayer,
* and Jon leighton for his super fast ArrayBuffer to Base64 convert.
*/
define([
"dojo/_base/declare","esri/geometry/Extent","dojo/query","esri/SpatialReference",
"esri/layers/TileInfo","esri/layers/TiledMapServiceLayer",
"dojo/Deferred","dojo/promise/all","dojo/Evented"],
function(declare,Extent,query,SpatialReference,TileInfo,TiledMapServiceLayer,
Deferred,all,Evented){
return declare("O.esri.TPK.TPKLayer",[TiledMapServiceLayer,Evented],{
//
// Public Properties
//
map: null,
store: null, // Reference to the local database store and hooks to it's functionality
MAX_DB_SIZE: 75, // Recommended maximum size in MBs
TILE_PATH:"", // The absolute path to the root of bundle/bundleX files e.g. V101/YOSEMITE_MAP/
RECENTER_DELAY: 350, // Millisecond delay before attempting to recent map after an orientation change
PARSING_ERROR: "parsingError", // An error was encountered while parsing a TPK file.
DB_INIT_ERROR: "dbInitError", // An error occurred while initializing the database.
DB_FULL_ERROR: "dbFullError", // No space left in the database.
NO_SUPPORT_ERROR: "libNotSupportedError", // Error indicating this library is not supported in a particular browser.
PROGRESS_START: "start",
PROGRESS_END: "end",
WINDOW_VALIDATED: "windowValidated", // All window functionality checks have passed
DB_VALIDATED: "dbValidated", // All database functionality checks have passed
//
// Events
//
DATABASE_ERROR_EVENT: "databaseErrorEvent", // An error thrown by the database.
VALIDATION_EVENT: "validationEvent", // Library validation checks.
PROGRESS_EVENT: "progress", // Event dispatched while parsing a bundle file.
//
// Private properties
//
_maxDBSize: 75, // User configurable maximum size in MBs.
_isDBWriteable: true, // Manually allow or stop writing to the database.
_isDBValid: false, // Does the browser support IndexedDB or IndexedDBShim
_autoCenter: null, // Auto center the map
_fileEntriesLength: 0, // Number of files in zip
_inMemTilesObject: null, // Stores unzipped files from tpk
_inMemTilesObjectLength: 0,
_zeroLengthFileCounter: 0, // For counting the number of zero length files in the tpk (e.g. directories)
_imageURL: "",
constructor:function(){
this._self = this;
this._inMemTilesIndex = [];
this._inMemTilesObject = {};
this.store = new O.esri.Tiles.TilesStore();
this._validate();
},
extend: function(files){
this._fileEntriesLength = files.length;
this.emit(this.PROGRESS_EVENT,this.PROGRESS_START);
this._parseInMemFiles(files,function (){
//Parse conf.xml and conf.cdi to get the required setup info
this._parseConfCdi(function(initExtent){
this.initialExtent = (this.fullExtent = initExtent);
this._parseConfXml(function(result){
this.tileInfo = new TileInfo(result);
this.spatialReference = new SpatialReference({wkid:this.tileInfo.spatialReference.wkid});
this.loaded = true;
this.onLoad(this);
this.emit(this.PROGRESS_EVENT,this.PROGRESS_END);
}.bind(this._self));
}.bind(this._self));
}.bind(this._self));
},
/**
* Overrides getTileUrl method
* @param level
* @param row
* @param col
* @returns {string}
*/
getTileUrl:function(level,row,col){
this.emit(this.PROGRESS_EVENT,this.PROGRESS_START);
var layersDir = this._self.TILE_PATH + "_alllayers";
var url = this._getCacheFilePath(layersDir,level,row,col);
if(this._inMemTilesObject != {}) {
/* temporary URL returned immediately, as we haven't retrieved the image from the indexeddb yet */
var tileid = "void:/" + level + "/" + row + "/" + col;
if(this.map == null) {
this.map = this.getMap();
}
if(this._autoCenter == null) {
this._autoCenter = new O.esri.TPK.autoCenterMap(this.map,this.RECENTER_DELAY);
this._autoCenter.init();
}
this._getInMemTiles(url,layersDir, level, row, col,tileid,function (result,tileid,url) {
var img = query("img[src=" + tileid + "]")[0];
if (typeof img == "undefined") {
img = new Image();
} //create a blank place holder for undefined images
var imgURL;
if (result) {
console.log("found tile offline", url);
var png = "data:image/png;base64,";
switch(this.tileInfo.format) {
case "JPEG":
imgURL = "data:image/jpg;base64," + result;
break;
case "PNG":
imgURL = png + result;
break;
case "PNG8":
imgURL = png + result;
break;
case "PNG24":
imgURL = png + result;
break;
case "PNG32":
imgURL = png + result;
break;
default:
imgURL = "data:image/jpg;base64," + result;
}
img.style.borderColor = "blue";
}
else {
img.style.borderColor = "green";
console.log("tile is not in the offline store", url);
imgURL = this._imageURL;
}
// when we return a nonexistent url to the image, the TiledMapServiceLayer::_tileErrorHandler() method
// sets img visibility to 'hidden', so we need to show the image back once we have put the data:image
img.style.visibility = "visible";
img.src = imgURL;
//console.log("URL length " + imgURL.length + ", image: " + imgURL);
this.emit(this.PROGRESS_EVENT,this.PROGRESS_END);
return "";
/* this result goes nowhere, seriously */
}.bind(this._self));
return tileid;
}
},
/**
* Optional. Set the maximum database size. Recommended maximum for mobile devices is 100MBs.
* Making the database too large can result in browser crashes and slow performance.
* TPKs can contain a lot of data!
* @param size
*/
setMaxDBSize: function(size){
//Make sure the entry is an integer.
var testRegex = /^\d+$/;
if(testRegex.test(size) && size <= this.MAX_DB_SIZE){
this._maxDBSize = size;
}
else{
console.log("setMaxDBSize Error: invalid entry. Integers only and less than " + this.MAX_DB_SIZE + "MBs");
}
},
/**
* Returns the size of the tiles database.
* @param callback {size , error}. Note, size is in bytes.
*/
getDBSize: function(callback){
this.store.usedSpace(function(size,err){
callback(size,err);
}.bind(this));
},
/**
* Sets whether or not tiles can be written to the database. This function
* can help you manage the size of the tiles database.
* Use this in conjunction with getDBSize() on a map pan or zoom event listener.
* @param value
*/
setDBWriteable: function(/* Boolean */ value){
this._isDBWriteable = value;
},
/**
* Validates whether or not the browser supports this library
* @returns {boolean}
*/
isDBValid: function(){
this._validate();
return this._isDBValid;
},
/**
* Reads a tile into tile database. Works with OfflineTilesBasic.js and OfflineTilesAdvanced.js
* saveToFile() functionality.
* IMPORTANT! The tile must confirm to an object using the pattern shown in _storeTile().
* @param file
* @param callback callback( boolean, error)
*/
loadFromURL: function (tile, callback) // callback(success,msg)
{
if (this.isDBValid()) {
this.store.store(tile, function (success,err) {
//Check the result
if (success) {
console.log("loadFromURL() success.");
callback(true, "");
}
else {
console.log("loadFromURL() Failed.");
callback(false, err);
}
});
}
else {
callback(false, "not supported");
}
},
/**
* Runs specific validation tasks. Reserved for future use.
* Currently only throws console errors. Does not stop execution of the library!
* @private
*/
_validate: function(){
//Verify if basic functionality is supported by the browser
if(!window.File && !window.FileReader && !window.Blob && !window.btoa && !window.DataView){
console.log("TPKLayer library is not supported by this browser");
this.emit(this.VALIDATION_EVENT,{msg:this.NO_SUPPORT_ERROR, err : null});
}
else{
this.emit(this.VALIDATION_EVENT,{msg:this.WINDOW_VALIDATED, err : null});
}
//Verify if IndexedDB is supported and initializes properly
if( /*false &&*/ this.store.isSupported() )
{
this.store.init(function(result){
if(result === false){
console.log("There was an error initializing the TPKLayer database");
this.emit(this.DATABASE_ERROR_EVENT,{msg:this.DB_INIT_ERROR, err: null});
}
else{
this.store.usedSpace(function(size,err){
var mb = this._bytes2MBs(size.sizeBytes);
if(mb > this.MAX_DB_SIZE){
console.log("Database is full!");
this.emit(this.DATABASE_ERROR_EVENT,{msg:this.DB_FULL_ERROR,err : err});
}
this.emit(this.VALIDATION_EVENT,{msg:this.DB_VALIDATED,err : null});
console.log("DB size: " + mb + " MBs, Tile count: " + size.tileCount + ", Error: " + err);
this._isDBValid = true;
}.bind(this));
}
}.bind(this));
}
else
{
console.log("IndexedDB is not supported on your browser.");
this.emit(this.VALIDATION_EVENT,{msg:this.NO_SUPPORT_ERROR, err : null});
}
},
/**
* Function for pulling out individual files from the .tpk/zip and storing
* them in memory.
* @param files
* @param callback
* @private
*/
_parseInMemFiles: function(files,callback){
var inMemTilesLength = this._fileEntriesLength;
this._zeroLengthFileCounter = 0;
var promises = [];
for(var i=0;i < inMemTilesLength;i++){
var deferred = new Deferred();
var name = files[i].filename.toLocaleUpperCase();
var index = name.indexOf("_ALLLAYERS",0);
if(index != -1){
this.TILE_PATH = name.slice(0,index);
}
if(files[i].compressedSize === 0) {
this._zeroLengthFileCounter++;
}
var indexCDI = name.indexOf("CONF.CDI",0);
var indexXML = name.indexOf("CONF.XML",0);
var indexBUNDLE = name.indexOf("BUNDLE",0);
var indexBUNDLX = name.indexOf("BUNDLX",0);
if(indexCDI != -1 || indexXML != -1){
this._unzipConfFiles(files,i,deferred,function(/* deferred */ d, /* token */ t){ console.log("CONF FILE");
d.resolve(t);
});
}
else if(indexBUNDLE != -1 || indexBUNDLX != -1){
this._unzipTileFiles(files,i,deferred,function(/* deferred */ d, /* token */ t){
d.resolve(t);
});
}
else{
deferred.resolve(i);
}
promises.push(deferred);
}
all(promises).then( function(results)
{
callback && callback(results); // jshint ignore:line
});
},
/**
* Calculate the size of an Object based on whether or not the item is enumerable.
* Native Objects don't have a built in size property.
* More info: https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Object/hasOwnProperty
* @param obj
* @returns {number}
* @constructor
*/
ObjectSize: function(obj) {
var size = 0, key;
for (key in obj) {
if (obj.hasOwnProperty(key)) {
size++;
}
}
return size;
},
/**
* Retrieve XML config files
* @param files
* @param token
* @param callback
* @private
*/
_unzipConfFiles: function(files,token,deferred,callback){
files[token].getData(new O.esri.zip.TextWriter(token),function(data){
this._inMemTilesIndex.push("blank");
var name = files[data.token].filename.toLocaleUpperCase();
this._inMemTilesObject[name]= data.string;
var size = this.ObjectSize(this._inMemTilesObject);
if(size > 0){
callback(deferred,data.token);
}
}.bind(this));
},
/**
* Retrieve binary tile files as ArrayBuffers
* @param files
* @param token
* @param callback
* @private
*/
_unzipTileFiles: function(files,token,deferred,callback){
var that = this;
files[token].getData(new O.esri.zip.BlobWriter(token),function(data){
if(data.size !== 0){
var reader = new FileReader();
reader.token = data.token;
reader.onerror = function (event) {
console.error("_unzipTileFiles Error: " + event.target.error.message);
that.emit(that.PARSING_ERROR, {msg: "Error parsing file: ", err: event.target.error});
};
reader.onloadend = function(evt) {
if(reader.token !== undefined){
that._inMemTilesIndex.push("blank");
var name = files[reader.token].filename.toLocaleUpperCase();
that._inMemTilesObject[name]= reader.result;
var size = that.ObjectSize(that._inMemTilesObject);
if(size > 0){
callback(deferred,data.token);
}
}
};
reader.readAsArrayBuffer(data); //open bundleX
}
});
},
/**
* Parse conf.cdi
* @param callback
* @private
*/
_parseConfCdi: function(callback){
var m_conf_i = this._inMemTilesObject[this.TILE_PATH + "CONF.CDI"];
var x2js = new O.esri.TPK.X2JS();
var jsonObj = x2js.xml_str2json( m_conf_i );
var envelopeInfo = jsonObj.EnvelopeN;
var xmin = parseFloat(envelopeInfo.XMin);
var ymin = parseFloat(envelopeInfo.YMin);
var xmax = parseFloat(envelopeInfo.XMax);
var ymax = parseFloat(envelopeInfo.YMax);
var sr = parseInt(envelopeInfo.SpatialReference.WKID);
var initExtent = new Extent(
xmin,ymin,xmax,ymax, new SpatialReference({wkid:sr})
);
callback(initExtent);
},
/**
* Parse conf.xml
* @param callback
* @private
*/
_parseConfXml:function(callback) {
var m_conf = this._inMemTilesObject[this.TILE_PATH + "CONF.XML"];
var x2js = new O.esri.TPK.X2JS();
var jsonObj = x2js.xml_str2json(m_conf);
var cacheInfo = jsonObj.CacheInfo;
var tileInfo = {};
tileInfo.rows = parseInt(cacheInfo.TileCacheInfo.TileRows);
tileInfo.cols = parseInt(cacheInfo.TileCacheInfo.TileCols);
tileInfo.dpi = parseInt(cacheInfo.TileCacheInfo.DPI);
tileInfo.format = cacheInfo.TileImageInfo.CacheTileFormat;
tileInfo.compressionQuality = parseInt(cacheInfo.TileImageInfo.CompressionQuality);
tileInfo.origin = {
x: parseInt(cacheInfo.TileCacheInfo.TileOrigin.X),
y: parseInt(cacheInfo.TileCacheInfo.TileOrigin.Y)
};
tileInfo.spatialReference = {
"wkid": parseInt(cacheInfo.TileCacheInfo.SpatialReference.WKID)
};
var lods = cacheInfo.TileCacheInfo.LODInfos.LODInfo;
var finalLods = [];
for (var i = 0; i < lods.length; i++) {
finalLods.push({
"level": parseFloat(lods[i].LevelID),
"resolution": parseFloat(lods[i].Resolution),
"scale": parseFloat(lods[i].Scale)});
}
tileInfo.lods = finalLods;
callback(tileInfo);
},
/**
* Parses the in-memory tile cache and returns a base64 tile image
* @param layersDir
* @param level
* @param row
* @param col
* @param tileCount - number of tiles in the Extent
* @param tiledId - id of the associated tile being passed
* @param callback
* @private
*/
_getInMemTiles: function(url,layersDir,level,row,col,tileId,callback){
var that = this._self;
var db = this.store;
//First check in the database if the tile exists.
//If not then we store the tile in the database later.
this.store.retrieve(url, function(success, offlineTile){
if( success && offlineTile.img !== "")
{
console.log("Tile found in storage: " + url);
callback(offlineTile.img,tileId,url);
}
else {
console.log("Tile is not in storage: " + url);
var snappedRow = Math.floor(row / 128) * 128;
var snappedCol = Math.floor(col / 128) * 128;
var path = this._getCacheFilePath(layersDir, level, snappedRow, snappedCol).toLocaleUpperCase();
var offset;
var bundleIndex = path + ".BUNDLE";
var bufferI = this._inMemTilesObject[bundleIndex];
var bufferX = this._inMemTilesObject[path + ".BUNDLX"];
if(bufferI !== undefined || bufferX !== undefined) {
offset = this._getOffset(level, row, col, snappedRow, snappedCol);
var pointer = that._getPointer(bufferX, offset);
that._buffer2Base64(bufferI,pointer,function(result){
if (that._isDBWriteable) {
that._storeTile(url, result, db,function(success,err){
if(err){
console.log("TPKLayer - Error writing to database." + err.message);
that.emit(that.DATABASE_ERROR_EVENT,{msg:"Error writing to database. ", err : err});
}
});
}
callback(result,tileId, url);
}.bind(that));
}
else{
console.log("_getInMemTiles Error: Invalid values");
callback(null,tileId,url);
}
}
}.bind(that));
},
/**
* Stores a tile in the local database.
* @param url
* @param base64Str
* @param db
* @param callback
* @private
*/
_storeTile: function(url,base64Str,db,callback){
var tile = {
url: url,
img: base64Str
};
db.store(tile,function(success,err){
callback(success,err);
});
},
/**
* Returns a pointer for reading a BUNDLE binary file as based on the given offset.
* @param buffer
* @param offset
* @returns {Uint8}
* @private
*/
_getPointer: function(/* ArrayBuffer */ buffer,offset){
var snip = buffer.slice(offset);
var dv = new DataView(snip,0,5);
var nume1 = dv.getUint8(0,true);
var nume2 = dv.getUint8(1,true);
var nume3 = dv.getUint8(2,true);
var nume4 = dv.getUint8(3,true);
var nume5 = dv.getUint8(4,true);
var value = nume5;
value = value * 256 + nume4;
value = value * 256 + nume3;
value = value * 256 + nume2;
value = value * 256 + nume1;
return value;
},
/**
* Convert an ArrayBuffer to base64. My testing shows this to be
* much faster than combining Blobs and btoa().
* ALL CREDITS: https://gist.github.com/jonleighton/958841
* NO licensing listed at the gist repo.
* @param arrayBuffer
* @returns {string}
* @private
*/
_base64ArrayBuffer: function(arrayBuffer) {
var base64 = "";
var encodings = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/";
var bytes = new Uint8Array(arrayBuffer);
var byteLength = bytes.byteLength;
var byteRemainder = byteLength % 3;
var mainLength = byteLength - byteRemainder;
var a, b, c, d;
var chunk;
/*jslint bitwise: true */
// Main loop deals with bytes in chunks of 3
for (var i = 0; i < mainLength; i = i + 3) {
// Combine the three bytes into a single integer
chunk = (bytes[i] << 16) | (bytes[i + 1] << 8) | bytes[i + 2];
// Use bitmasks to extract 6-bit segments from the triplet
a = (chunk & 16515072) >> 18; // 16515072 = (2^6 - 1) << 18
b = (chunk & 258048) >> 12; // 258048 = (2^6 - 1) << 12
c = (chunk & 4032) >> 6; // 4032 = (2^6 - 1) << 6
d = chunk & 63; // 63 = 2^6 - 1
// Convert the raw binary segments to the appropriate ASCII encoding
base64 += encodings[a] + encodings[b] + encodings[c] + encodings[d];
}
// Deal with the remaining bytes and padding
if (byteRemainder == 1) {
chunk = bytes[mainLength];
a = (chunk & 252) >> 2; // 252 = (2^6 - 1) << 2
// Set the 4 least significant bits to zero
b = (chunk & 3) << 4; // 3 = 2^2 - 1
base64 += encodings[a] + encodings[b] + "==";
} else if (byteRemainder == 2) {
chunk = (bytes[mainLength] << 8) | bytes[mainLength + 1];
a = (chunk & 64512) >> 10; // 64512 = (2^6 - 1) << 10
b = (chunk & 1008) >> 4; // 1008 = (2^6 - 1) << 4
// Set the 2 least significant bits to zero
c = (chunk & 15) << 2; // 15 = 2^4 - 1
base64 += encodings[a] + encodings[b] + encodings[c] + "=";
}
/*jslint bitwise: false */
return base64;
},
/**
* Given a ArrayBuffer and a position it will return a Base64 tile image
* @param arrayBuffer
* @param position
* @returns {string}
* @private
*/
_buffer2Base64: function(/* ArrayBuffer */arrayBuffer,/* int */ position,callback){
var view = new DataView(arrayBuffer,position);
var chunk = view.getInt32(0,true);
var buffer = view.buffer.slice(position + 4,position + 4 + chunk);
var string = this._base64ArrayBuffer(buffer);
callback(string);
},
/**
* Converts an integer to hex
* @param value
* @returns {string}
* @private
*/
_int2HexString: function(/* int */ value){
var text = value.toString(16).toUpperCase();
if (text.length === 1)
{
return "000" + text;
}
if (text.length === 2)
{
return "00" + text;
}
if (text.length === 3)
{
return "0" + text;
}
return text.substr(0, text.length);
},
/**
* Determines where to start reading a BUNDLEX binary file
* @param level
* @param row
* @param col
* @param startRow
* @param startCol
* @returns {number}
* @private
*/
_getOffset: function(/* int */level, /* number */row,/* number */col, /* number */startRow, /* number */ startCol){
var recordNumber = 128 * (col - startCol) + (row - startRow);
return 16 + recordNumber * 5;
},
/**
* Returns a hexadecimal representation of a cache file path
* @param layerDir
* @param level
* @param row
* @param col
* @returns {string}
* @private
*/
_getCacheFilePath: function(/* String */ layerDir, /* int */level, /* int */row, /* int */ col){
var arr = [];
arr.push(layerDir);
arr.push("/");
arr.push("L");
arr.push(level < 10 ? "0" + level : level);
arr.push("/");
arr.push("R");
arr.push(this._int2HexString(row));
arr.push("C");
arr.push(this._int2HexString(col));
return arr.join("");
},
/**
* Returns database size in MBs.
* @returns {string}
* @private
*/
_bytes2MBs: function(bytes){
return (bytes >>> 20 ) + '.' + ( bytes & (2*0x3FF ) ); // jshint ignore:line
}
});
}
);