offline-editor-js/lib/tpk/TPKLayer.js
2014-05-23 12:16:16 -06:00

610 lines
31 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","tiles/TilesStore","tiles/tilingScheme",
"tpk/zip","tpk/xml2json","tpk/autoCenterMap","dojo/Evented"],
function(declare,Extent,query,SpatialReference,TileInfo,TiledMapServiceLayer,TilesStore,TilingScheme,zip,X2JS,autoCenter,Evented){
return declare("esri.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
VALIDATION_ERROR: "validationError", // Library validation error.
DATABASE_ERROR: "databaseError", // An error thrown by the database.
PARSING_ERROR: 'parsingError', // An error was encountered while parsing a TPK file.
DB_INIT_ERROR: "dbInitError", // An error occurred while initializing the database.
NO_SUPPORT_ERROR: "libNotSupportedError", // Error indicating this library is not supported in a particular browser.
PROGRESS_EVENT: "progress", // Event dispatched while parsing a bundle file.
PROGRESS_START: "start",
PROGRESS_END: "end",
//
// Private properties
//
_maxDBSize: 75, // User configurable maximum size in MBs.
_isDBWriteable: true, // Manually allow or stop writing to the database.
_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)
constructor:function(){
this._self = this;
this._inMemTilesIndex = [];
this._inMemTilesObject = {};
this.store = new TilesStore();
this._validate();
},
extend: function(files){
this._fileEntriesLength = files.length;
this.emit(this.PROGRESS_EVENT,this.PROGRESS_START);
this._parseInMemFiles(files,function (buffer){
//Parse conf.xml and conf.cdi to get the required setup info
this._parseConfCdi(buffer,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 autoCenter(this.map,this.RECENTER_DELAY);
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);
switch(this.tileInfo.format) {
case "JPEG":
imgURL = "data:image/jpg;base64," + result;
break;
case "PNG8":
imgURL = "data:image/png;base64," + result;
break;
}
img.style.borderColor = "blue";
}
else {
img.style.borderColor = "green";
console.log("tile is not in the offline store", url);
imgURL = "";
}
// 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
*/
isDBWriteable: function(/* Boolean */ value){
this._isDBWriteable = value;
},
/**
* 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.error(new Error( "This library is not supported on your browser!").stack);
this.emit(this.VALIDATION_ERROR,{msg:this.NO_SUPPORT_ERROR, err : null});
}
//Verify if IndexedDB is supported and initializes properly
if( /*false &&*/ this.store.isSupported() )
{
this.store.init(function(result){
if(result == false){
console.error(new Error( "There was an error initializing the database.").stack);
this.emit(this.DATABASE_ERROR,{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.error(new Error( "Database is full!").stack);
this.emit(this.DATABASE_ERROR,{msg:"Database full! ",err : err});
}
console.log("DB size: " + mb + " MBs, Tile count: " + size.tileCount + ", Error: " + err)
}.bind(this))
}
}.bind(this));
}
else
{
console.error(new Error( "IndexedDB is not supported on your browser.").stack);
this.emit(this.VALIDATION_ERROR,{msg:"IndexedDB is not supported.", 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;
for(var i=0;i < inMemTilesLength;i++){
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);
if(indexCDI != -1 || indexXML != -1){
this._unzipConfFiles(files,i,callback);
}
else{
this._unzipTileFiles(files,i,callback);
}
}
},
/**
* 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,callback){
files[token].getData(new 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 == this._fileEntriesLength - this._zeroLengthFileCounter - 1){
callback(this._inMemTilesObject);
}
}.bind(this));
},
/**
* Retrieve binary tile files as ArrayBuffers
* @param files
* @param token
* @param callback
* @private
*/
_unzipTileFiles: function(files,token,callback){
var that = this;
files[token].getData(new zip.BlobWriter(token),function(data){
if(data.size != 0){
var reader = new FileReader();
reader.token = data.token;
reader.onerror = function (event) {
console.error(new Error("_unzipTileFiles Error: " + event.target.error.code).stack);
that.emit(that.PARSING_ERROR, {msg: "Error parsing file: ", err: event.target.error});
}
reader.addEventListener("loadend", function (evt) {
if(this.token != undefined){
that._inMemTilesIndex.push("blank");
var name = files[this.token].filename.toLocaleUpperCase();
that._inMemTilesObject[name]= this.result;
var size = that.ObjectSize(that._inMemTilesObject);
if(size == that._fileEntriesLength - that._zeroLengthFileCounter - 1){
callback(that._inMemTilesObject);
}
}
});
reader.readAsArrayBuffer(data); //open bundleX
}
});
},
/**
* Parse conf.cdi
* @param tilesInfo
* @param callback
* @private
*/
_parseConfCdi: function(tilesInfo,callback){
var that = this._self;
var m_conf_i = this._inMemTilesObject[this.TILE_PATH + "CONF.CDI"];
var x2js = new 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 initExtent = new Extent(
xmin,ymin,xmax,ymax
);
callback(initExtent);
},
/**
* Parse conf.xml
* @param callback
* @private
*/
_parseConfXml:function(callback) {
var m_conf = this._inMemTilesObject[this.TILE_PATH + "CONF.XML"];
var x2js = new 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 )
{
console.log("Tile found in indexedDB: " + url)
callback(offlineTile.img,tileId,url);
}
else {
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";
offset = this._getOffset(level, row, col, snappedRow, snappedCol);
var pointer = that._getPointer(this._inMemTilesObject[path + ".BUNDLX"], offset);
that._buffer2Base64(this._inMemTilesObject[bundleIndex],pointer,function(result){
if (that._isDBWriteable)that._storeTile(url, result, db);
callback(result,tileId, url);
}.bind(that));
}
}.bind(that))
},
/**
* Stores a tile in the local database.
* @param url
* @param base64Str
* @param db
* @private
*/
_storeTile: function(url,base64Str,db){
var tile = {
url: url,
img: base64Str
};
db.store(tile,function(success,err){
if(err){
console.error(new Error( "Error writing to database." + err).stack);
this.emit(this.DATABASE_ERROR,{msg:"Error writing to database. ", err : 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
// 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] + '='
}
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 + 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 ) )
}
})
}
)