offline-editor-js/lib/tpk/TPKLayer.js
2014-05-14 10:02:43 -06:00

600 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
*/
define([
"dojo/_base/declare","esri/geometry/Extent","dojo/query","esri/SpatialReference","tpk/DataStream",
"esri/layers/TileInfo","esri/layers/TiledMapServiceLayer","tiles/TilesStore","tiles/tilingScheme",
"tpk/zip","tpk/xml2json","tpk/autoCenterMap","dojo/Evented"],
function(declare,Extent,query,SpatialReference,DataStream,TileInfo,TiledMapServiceLayer,TilesStore,TilingScheme,zip,X2JS,autoCenter,Evented){
return declare("esri.TPKLayer",[TiledMapServiceLayer,Evented],{
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.
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
//
// Public Properties
//
map: null,
store: null, // Reference to the local database store and hooks to it's functionality
constructor:function(){
this._self = this;
this._inMemTiles = [];
this.store = new TilesStore();
this._validate();
},
extend: function(files){
this._fileEntriesLength = files.length;
this.emit(this.PROGRESS_EVENT,this.PROGRESS_START);
// Create a new array that contains an index of what is in the zip/tpk file. This provides
// a highly optimized search pattern based on each tiles filename. We can then look up the
// index first and then use that index to retrieve the exact tile without having to iterate
// through a large in-memory Array.
this._inMemTilesIndex = files.map(function(tile){
var name = tile.filename.toLocaleUpperCase();
var index = name.indexOf("_ALLLAYERS",0);
if(index != -1){
this.TILE_PATH = name.slice(0,index);
}
console.log("TPK filename " + name);
return name;
}.bind(this));
this._parseInMemFiles(files,function (buffer){
//Parse conf.xml and conf.cdi to get the required setup info
this._parseConfCdi(buffer,function(initExtent,result){
this.initialExtent = (this.fullExtent = initExtent);
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));
},
/**
* 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._inMemTiles.length > 0) {
/* 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);
var count = this._getUrlCountByExtent(this._self,this.map.extent,level);
//console.log("Layer tile count: " + count)
this._getInMemTiles(url,layersDir, level, row, col, count,function (result) {
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);
imgURL = "data:image/png;base64," + result;
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:"TPKLayer library is not supported", 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:"Unable to initialize the database.", 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});
}
},
/**
* Recursive function for pulling out individual files from the .tpk/zip and storing
* them in an array as blobs.
* @param files
* @param callback
* @private
*/
_parseInMemFiles: function(files,callback){
var that = this;
var inMemTilesLength = this._inMemTiles.length;
if(inMemTilesLength < this._fileEntriesLength){
files[0].getData(new zip.BlobWriter(),function(data){
var reader = new FileReader();
reader.onerror = function (event) {
console.error(new Error("_parseInMemFiles Error: " + event.target.error.code).stack);
this.emit(context.PARSING_ERROR, {msg: "Error parsing file: ", err: event.target.error});
}
reader.addEventListener("loadend", function (evt) {
that._inMemTiles.push(this.result);
files.shift();
that._parseInMemFiles(files,callback);
});
reader.readAsArrayBuffer(data); //open bundleX
}.bind(that));
}
if(inMemTilesLength == this._fileEntriesLength){
callback(this._inMemTiles);
}
},
/**
* Parse conf.cdi
* @param tilesInfo
* @param callback
* @private
*/
_parseConfCdi: function(tilesInfo,callback){
var that = this._self;
var m_conf_index = this._inMemTilesIndex.indexOf(this.TILE_PATH + "CONF.CDI");
if(m_conf_index != -1){
var m_conf_i = this._inMemTiles[m_conf_index];
var result = m_conf_i;
var str = this._bin2String(result);
var x2js = new X2JS();
var jsonObj = x2js.xml_str2json( str );
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
);
that._parseConfXml(initExtent,function(initExtent,result){
callback(initExtent,result)
},that)
}
},
/**
* Parse conf.xml
* @param callback
* @param context
* @private
*/
_parseConfXml:function(initExtent,callback,context) {
var m_conf_config = this._inMemTilesIndex.indexOf(this.TILE_PATH + "CONF.XML");
if (m_conf_config != -1) {
var m_conf = this._inMemTiles[m_conf_config];
var result = m_conf;
var str = this._bin2String(result);
var x2js = new X2JS();
var jsonObj = x2js.xml_str2json(str);
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(initExtent,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 callback
* @private
*/
_getInMemTiles: function(url,layersDir,level,row,col,tileCount,callback){
var that = this._self;
var url = url;
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);
}
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 = this._inMemTilesIndex.indexOf(path + ".BUNDLE");
var bundleXIndex = this._inMemTilesIndex.indexOf(path + ".BUNDLX");
if (bundleIndex == -1 || bundleXIndex == -1) {
callback(null) //didn't find anything
}
else{
offset = this._getOffset(level, row, col, snappedRow, snappedCol);
var pointer = that._getPointer(this._inMemTiles[bundleXIndex], offset);
var str = that._bin2Base64(this._inMemTiles[bundleIndex],pointer);
if (that._isDBWriteable)that._storeTile(url, str, db);
callback(str);
}
}
}.bind(that))
},
/**
* Parses a BUNDLEX file as an ArrayBuffer.
* @param array
* @param index
* @param callback returns a pointer based on the given offset
* @private
*/
_bundleXReader: function(array,index,callback){
var bundleX = array[index];
// If you don't capture all the pointers in the first extent
// this will speed things up considerably in any secondary pan events.
// Also on slower devices this may add a performance boost.
if(bundleX instanceof ArrayBuffer){
callback(bundleX);
}
else {
var reader = new FileReader();
reader.onerror = function (event) {
console.error(new Error("_bundleXReader Error: " + event.target.error.code).stack);
context.emit(context.PARSING_ERROR, {msg: "Error parsing bundleX file: ", err: event.target.error});
}
reader.addEventListener("loadend", function (evt) {
array[index] = this.result;
callback(this.result);
});
reader.readAsArrayBuffer(bundleX); //open bundleX
}
},
/**
* Parses a BUNDLE file as an ArrayBuffer.
* @param array
* @param index
* @param callback returns a base64 url based on the given pointer.
* @private
*/
_bundleReader: function(array,index,callback){
var bundle = array[index];
// If you don't capture all the tiles in the first extent
// this will speed things up considerably in any secondary pan events.
// Also on slower devices this may add a performance boost.
if(bundle instanceof ArrayBuffer){
callback(bundle);
}
else{
var reader = new FileReader();
reader.onerror = function(event){
console.error(new Error( "_bundleReader Error: " + event.target.error.code).stack);;
context.emit(context.PARSING_ERROR,{msg: "Error parsing bundle file: ", err : event.target.error});
}
reader.addEventListener("loadend", function(evt) {
array[index] = this.result;
callback(this.result);
});
reader.readAsArrayBuffer(bundle); //open bundle
}
},
/**
* 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});
}
});
},
/**
* Helper method that returns an array of tile urls within a given extent and level
* @returns int
*/
_getUrlCountByExtent: function(layer,extent,level){
var tilingScheme = new TilingScheme(layer);
var level_cell_ids = tilingScheme.getAllCellIdsInExtent(extent,level);
var count = 0;
level_cell_ids.forEach(function(cell_id)
{
count++;
}.bind(this));
return count;
},
/**
* Returns a pointer for reading a BUNDLE binary file as based on the given offset.
* @param blob
* @param offset
* @returns {Uint8}
* @private
*/
_getPointer: function(/* Blob */ blob,offset){
var snip = blob.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;
},
/**
* Converts a blob to a string
* @param blob
* @returns {string}
* @private
*/
_bin2String: function(/* Blob */ blob){
var str = "";
var arr = new Uint8Array(blob,0);
var length = arr.length;
for (var i = 0; i < length; i++) {
str += String.fromCharCode(arr[i])
}
return str;
},
/**
* Given a blob and a position it will return a Base64 tile image
* @param blob
* @param position
* @returns {string}
* @private
*/
_bin2Base64: function(/* Blob */blob,/* int */ position){
var stream = new DataStream(blob, 0,
DataStream.LITTLE_ENDIAN);
stream.seek(position);
var chunk = stream.readInt32(true);
var string = stream.readString(chunk); //Notes: Range limits in Chrome: https://bugs.webkit.org/show_bug.cgi?id=80797
return btoa(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 ) )
}
})
}
)