UNPKG

esri-leaflet

Version:

Leaflet plugins for consuming ArcGIS Online and ArcGIS Server services.

248 lines (214 loc) 7.48 kB
import { CRS, DomEvent, TileLayer, Util } from "leaflet"; import { warn, getUrlParams, setEsriAttribution, removeEsriAttribution, } from "../Util.js"; import mapService from "../Services/MapService.js"; export const TiledMapLayer = TileLayer.extend({ options: { zoomOffsetAllowance: 0.1, errorTileUrl: "", }, statics: { MercatorZoomLevels: { 0: 156543.03392799999, 1: 78271.516963999893, 2: 39135.758482000099, 3: 19567.879240999901, 4: 9783.9396204999593, 5: 4891.9698102499797, 6: 2445.9849051249898, 7: 1222.9924525624899, 8: 611.49622628138002, 9: 305.74811314055802, 10: 152.874056570411, 11: 76.437028285073197, 12: 38.218514142536598, 13: 19.109257071268299, 14: 9.5546285356341496, 15: 4.7773142679493699, 16: 2.38865713397468, 17: 1.1943285668550501, 18: 0.59716428355981699, 19: 0.29858214164761698, 20: 0.14929107082381, 21: 0.07464553541191, 22: 0.0373227677059525, 23: 0.0186613838529763, }, }, initialize(options) { options = Util.setOptions(this, options); // support apikey property being passed in if (options.apikey) { options.token = options.apikey; } // set the urls options = getUrlParams(options); this.tileUrl = `${(options.proxy ? `${options.proxy}?` : "") + options.url}tile/{z}/{y}/{x}${options.requestParams && Object.keys(options.requestParams).length > 0 ? Util.getParamString(options.requestParams) : ""}`; // Remove subdomain in url // https://github.com/Esri/esri-leaflet/issues/991 if (options.url.indexOf("{s}") !== -1 && options.subdomains) { options.url = options.url.replace("{s}", options.subdomains[0]); } this.service = mapService(options); this.service.addEventParent(this); const arcgisonline = new RegExp(/tiles.arcgis(online)?\.com/g); if (arcgisonline.test(options.url)) { this.tileUrl = this.tileUrl.replace("://tiles", "://tiles{s}"); options.subdomains = ["1", "2", "3", "4"]; } if (this.options.token) { this.tileUrl += `?token=${this.options.token}`; } // init layer by calling TileLayers initialize method TileLayer.prototype.initialize.call(this, this.tileUrl, options); }, getTileUrl(tilePoint) { const zoom = this._getZoomForUrl(); return Util.template( this.tileUrl, Util.extend( { s: this._getSubdomain(tilePoint), x: tilePoint.x, y: tilePoint.y, // try lod map first, then just default to zoom level z: this._lodMap && this._lodMap[zoom] !== undefined ? this._lodMap[zoom] : zoom, }, this.options, ), ); }, createTile(coords, done) { const tile = document.createElement("img"); DomEvent.on(tile, "load", Util.bind(this._tileOnLoad, this, done, tile)); DomEvent.on(tile, "error", Util.bind(this._tileOnError, this, done, tile)); if (this.options.crossOrigin) { tile.crossOrigin = ""; } /* Alt tag is set to empty string to keep screen readers from reading URL and for compliance reasons http://www.w3.org/TR/WCAG20-TECHS/H67 */ tile.alt = ""; // if there is no lod map or an lod map with a proper zoom load the tile // otherwise wait for the lod map to become available if ( !this._lodMap || (this._lodMap && this._lodMap[this._getZoomForUrl()] !== undefined) ) { tile.src = this.getTileUrl(coords); } else { this.once( "lodmap", function () { tile.src = this.getTileUrl(coords); }, this, ); } return tile; }, onAdd(map) { // include 'Powered by Esri' in map attribution setEsriAttribution(map); if (!this._lodMap) { this.metadata(function (error, metadata) { if (!error && metadata.spatialReference) { const sr = metadata.spatialReference.latestWkid || metadata.spatialReference.wkid; // display the copyright text from the service using leaflet's attribution control if ( !this.options.attribution && map.attributionControl && metadata.copyrightText ) { this.options.attribution = metadata.copyrightText; map.attributionControl.addAttribution(this.getAttribution()); } // if the service tiles were published in web mercator using conventional LODs but missing levels, we can try and remap them if ( map.options.crs === CRS.EPSG3857 && (sr === 102100 || sr === 3857) ) { this._lodMap = {}; // create the zoom level data const arcgisLODs = metadata.tileInfo.lods; const correctResolutions = TiledMapLayer.MercatorZoomLevels; for (let i = 0; i < arcgisLODs.length; i++) { const arcgisLOD = arcgisLODs[i]; for (const ci in correctResolutions) { const correctRes = correctResolutions[ci]; if ( this._withinPercentage( arcgisLOD.resolution, correctRes, this.options.zoomOffsetAllowance, ) ) { this._lodMap[ci] = arcgisLOD.level; break; } } } this.fire("lodmap"); } else if ( map.options.crs && map.options.crs.code && map.options.crs.code.indexOf(sr) > -1 ) { // if the projection is WGS84, or the developer is using Proj4 to define a custom CRS, no action is required } else { // if the service was cached in a custom projection and an appropriate LOD hasn't been defined in the map, guide the developer to our Proj4 sample warn( "L.esri.TiledMapLayer is using a non-mercator spatial reference. Support may be available through Proj4Leaflet https://developers.arcgis.com/esri-leaflet/samples/non-mercator-projection/", ); } } }, this); } TileLayer.prototype.onAdd.call(this, map); }, onRemove(map) { removeEsriAttribution(map); TileLayer.prototype.onRemove.call(this, map); }, metadata(callback, context) { this.service.metadata(callback, context); return this; }, identify() { return this.service.identify(); }, find() { return this.service.find(); }, query() { return this.service.query(); }, authenticate(token) { const tokenQs = `?token=${token}`; this.tileUrl = this.options.token ? this.tileUrl.replace(/\?token=(.+)/g, tokenQs) : this.tileUrl + tokenQs; this.options.token = token; this.service.authenticate(token); return this; }, _withinPercentage(a, b, percentage) { const diff = Math.abs(a / b - 1); return diff < percentage; }, }); export function tiledMapLayer(url, options) { return new TiledMapLayer(url, options); } export default tiledMapLayer;