define([ "dojo/query", "dojo/request", "dojo/_base/declare", "tiles/base64utils", "tiles/TilesStore", "tiles/tilingScheme", "tiles/FileSaver" ], function(query, request, declare,Base64Utils,TilesStore,TilingScheme,FileSaver) { "use strict"; return declare([],{ /** * Utility method to get the basemap layer reference * @param map * @returns {Number} layerId */ getBasemapLayer: function(map) { var layerId = map.layerIds[0]; return map.getLayer(layerId); }, /** * Method that extends a layer object with the offline capability. * After extending one layer, you can call layer.goOffline() or layer.goOnline() * @param layer * @param callback * @returns {callback} callback(boolean, string) */ extend: function(layer,callback) { console.log("extending layer", layer.url); layer._lastTileUrl = ""; /* we add some methods to the layer object */ /* we don't want to extend the tiled layer class, as it is a capability that we want to add only to one instance */ /* we also add some additional attributes inside an "offline" object */ layer._getTileUrl = layer.getTileUrl; layer.offline = { online: true, store: new TilesStore(), proxyPath: "../lib/resource-proxy/proxy.php" }; if( /*false &&*/ layer.offline.store.isSupported() ) { layer.offline.store.init(callback); } else { return callback(false, "indexedDB not supported"); } layer.resampling = false; /** * Internal method that overrides the getTileUrl() method. * If application is offline then tiles are written to local storage. * Retrieves tiles as requested by the ArcGIS API for JavaScript. * If a tile is in cache it is returned. * If it is not in cache then one is retrieved over the internet. * @param level * @param row * @param col * @returns {String} URL */ layer.getTileUrl = function(level,row,col) { console.assert(!isNaN(level) && !isNaN(row) && !isNaN(col), "bad tile requested"); console.log("looking for tile",level,row,col); var url = this._getTileUrl(level,row,col); console.log("LIBRARY ONLINE " + this.offline.online) if( this.offline.online ) { console.log("fetching url online: ", url); layer._lastTileUrl = url; return url; } url = url.split("?")[0]; /* temporary URL returned immediately, as we haven't retrieved the image from the indexeddb yet */ var tileid = "void:/"+level+"/"+row+"/"+col; this.offline.store.retrieve(url, function(success, offlineTile) { /* when the .get() callback is called we replace the temporary URL originally returned by the data:image url */ // search for the img with src="void:"+level+"-"+row+"-"+col and replace with actual url var img = query("img[src="+tileid+"]")[0]; var imgURL; console.assert(img !== "undefined", "undefined image detected"); if( success ) { img.style.borderColor = "blue"; console.log("found tile offline", url); imgURL = "data:image;base64," + offlineTile.img; } 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; return ""; /* this result goes nowhere, seriously */ }); return tileid; }; /** * Returns an object that contains the number of tiles that would need to be downloaded * for the specified extent and zoom level, and the estimated byte size of such tiles. * This method is useful to give the user an indication of the required time and space * before launching the actual download operation. The byte size estimation is very rough. * @param extent * @param level * @param tileSize * @returns {{level: *, tileCount: Number, sizeBytes: number}} */ layer.getLevelEstimation = function(extent, level, tileSize) { var tilingScheme = new TilingScheme(this); var cellIds = tilingScheme.getAllCellIdsInExtent(extent,level); var levelEstimation = { level: level, tileCount: cellIds.length, sizeBytes: cellIds.length * tileSize }; return levelEstimation; }; /** * Retrieves tiles and stores them in the local cache. * @param minLevel * @param maxLevel * @param extent * @param reportProgress */ layer.prepareForOffline = function(minLevel, maxLevel, extent, reportProgress) { /* create list of tiles to store */ var tilingScheme = new TilingScheme(this); var cells = []; var level; for(level=minLevel; level<=maxLevel; level++) { var level_cell_ids = tilingScheme.getAllCellIdsInExtent(extent,level); level_cell_ids.forEach(function(cell_id) { cells.push({ level: level, row: cell_id[1], col: cell_id[0]}); }); // if the number of requested tiles is excessive, we just stop if( cells.length > 5000 && level !== maxLevel) { console.log("enough is enough!"); break; } } /* launch tile download */ this._doNextTile(0, cells, reportProgress); }; /** * This method puts the layer in offline mode. When in offline mode, * the layer will not fetch any tile from the remote server. It * will look up the tiles in the indexed db database and display them in the layer. * If the tile can't be found in the local database it will show up blank * (even if there is actual connectivity). The pair of methods goOffline() and * goOnline()allows the developer to manually control the behaviour of the layer. * Used in conjunction with the offline dectection library, you can put the layer in * the appropriate mode when the offline condition changes. */ layer.goOffline = function() { this.offline.online = false; }; /** * This method puts the layer in online mode. When in online mode, the layer will * behave as regular layers, fetching all tiles from the remote server. * If there is no internet connectivity the tiles may appear thanks to the browsers cache, * but no attempt will be made to look up tiles in the local database. */ layer.goOnline = function() { this.offline.online = true; this.refresh(); }; /** * Determines if application is online or offline * @returns {boolean} */ layer.isOnline = function() { return this.offline.online; }; /** * Clears the local cache of tiles. * @param callback callback(boolean, errors) */ layer.deleteAllTiles = function(callback) // callback(success) or callback(false, error) { var store = this.offline.store; store.deleteAll(callback); }; /** * Gets the size in bytes of the local tile cache. * @param callback callback(size, error) */ layer.getOfflineUsage = function(callback) // callback({size: <>, tileCount: <>}) or callback(null,error) { var store = this.offline.store; store.usedSpace(callback); }; /** * Gets polygons representing all cached cell ids within a particular * zoom level and bounded by an extent. * @param callback callback(polygon, error) */ layer.getTilePolygons = function(callback) // callback(Polygon polygon) or callback(null, error) { var store = this.offline.store; var tilingScheme = new TilingScheme(this); store.getAllTiles(function(url,img,err) { if(url) { var components = url.split("/"); var level = parseInt(components[ components.length - 3],10); var col = parseInt(components[ components.length - 2],10); var row = parseInt(components[ components.length - 1],10); var cellId = [row,col]; var polygon = tilingScheme.getCellPolygonFromCellId(cellId, level); //if( level == 15) callback(polygon); } else { callback(null,err); } }); }; /** * Saves tile cache into a portable csv format. * @param fileName * @param callback callback( boolean, error) */ layer.saveToFile = function(fileName, callback) // callback(success, msg) { var store = this.offline.store; var csv = []; csv.push("url,img"); store.getAllTiles(function(url,img,evt) { if(evt==="end") { var blob = new Blob([ csv.join("\r\n") ], {type:"text/plain;charset=utf-8"}); var saver = FileSaver.saveAs(blob, fileName); if( saver.readyState === saver.DONE ) { if( saver.error ) { return callback(false,"Error saving file " + fileName); } return callback(true, "Saved " + (csv.length-1) + " tiles (" + Math.floor(blob.size / 1024 / 1024 * 100) / 100 + " Mb) into " + fileName); } saver.onerror = function() { callback(false,"Error saving file " + fileName); }; saver.onwriteend = function() { callback(true, "Saved " + (csv.length-1) + " tiles (" + Math.floor(blob.size / 1024 / 1024 * 100) / 100 + " Mb) into " + fileName); }; } else { csv.push(url+","+img); } }); }; /** * Reads a csv file into local tile cache. * @param file * @param callback callback( boolean, error) */ layer.loadFromFile = function(file, callback) // callback(success,msg) { console.log("reading",file); var store = this.offline.store; var i; if (window.File && window.FileReader && window.FileList && window.Blob) { // Great success! All the File APIs are supported. var reader = new FileReader(); reader.onload = function(evt) { var csvContent = evt.target.result; var tiles = csvContent.split("\r\n"); var tileCount = 0; var pair, tile; if(tiles[0] !== "url,img") { return callback(false, "File " + file.name + " doesn't contain tiles that can be loaded"); } for(i=1; i