map/esriMap.js

  1. 'use strict';
  2. const basemap = require('./basemap.js');
  3. const mapPrint = require('./print.js');
  4. function esriMap(esriBundle, geoApi) {
  5. const printModule = mapPrint(esriBundle);
  6. class Map {
  7. static get Extent () { return esriBundle.Extent; }
  8. // TODO when jshint parses instance fields properly we can change this from a property to a field
  9. get _passthroughBindings () { return [
  10. 'on', 'reorderLayer', 'addLayer', 'disableKeyboardNavigation', 'removeLayer', 'resize', 'reposition',
  11. 'centerAt', 'setZoom', 'centerAndZoom', 'toScreen', 'setExtent'
  12. ]; }
  13. get _passthroughProperties () { return ['graphicsLayerIds', 'layerIds', 'spatialReference', 'extent']; } // TODO when jshint parses instance fields properly we can change this from a property to a field
  14. constructor (domNode, opts) {
  15. this._passthroughBindings.forEach(bindingName =>
  16. this[bindingName] = (...args) => this._map[bindingName](...args));
  17. this._passthroughProperties.forEach(propName => {
  18. const descriptor = {
  19. enumerable: true,
  20. get: () => this._map[propName]
  21. };
  22. Object.defineProperty(this, propName, descriptor);
  23. });
  24. this._map = new esriBundle.Map(domNode, { extent: Map.getExtentFromJson(opts.extent), lods: opts.lods });
  25. if (opts.proxyUrl) {
  26. this.proxy = opts.proxyUrl;
  27. }
  28. if (opts.basemaps) {
  29. this.basemapGallery = basemap.initBasemaps(esriBundle, opts.basemaps, this._map);
  30. this.basemapGallery.on('selection-change', () => this.resetOverviewMap());
  31. } else {
  32. throw new Error('The basemaps option is required to and at least one basemap must be defined');
  33. }
  34. if (opts.scalebar) {
  35. this.scalebar = new esriBundle.Scalebar({
  36. map: this._map,
  37. attachTo: opts.scalebar.attachTo,
  38. scalebarUnit: opts.scalebar.scalebarUnit
  39. });
  40. this.scalebar.show();
  41. }
  42. if (opts.overviewMap) {
  43. this.initOverviewMap();
  44. }
  45. this.zoomPromise = Promise.resolve();
  46. this.zoomCounter = 0;
  47. }
  48. printLocal (options) { return printModule.printLocal(this._map, options); }
  49. printServer (options) { return printModule.printServer(this._map, options); }
  50. /**
  51. * Select a basemap which has been loaded in the basemapGallery
  52. *
  53. * @param {Object|String} value either an object with an id field or a string
  54. */
  55. selectBasemap (value) {
  56. if (typeof value === 'object') {
  57. value = value.id;
  58. }
  59. this.basemapGallery.select(value);
  60. }
  61. /**
  62. * Create an ESRI Extent object from extent setting JSON object.
  63. *
  64. * @function getExtentFromJson
  65. * @param {Object} extentJson that follows config spec
  66. * @return {Object} an ESRI Extent object
  67. */
  68. static getExtentFromJson (extentJson) {
  69. return esriBundle.Extent({ xmin: extentJson.xmin, ymin: extentJson.ymin,
  70. xmax: extentJson.xmax, ymax: extentJson.ymax,
  71. spatialReference: { wkid: extentJson.spatialReference.wkid } });
  72. }
  73. /**
  74. * Take a JSON object with extent properties and convert it to an ESRI Extent.
  75. * Reprojects to map projection if required.
  76. *
  77. * @param {Object} extent the extent to enhance
  78. * @returns {Extent} extent cast in Extent prototype, and in map spatial reference
  79. */
  80. enhanceConfigExtent (extent) {
  81. const realExtent = Map.getExtentFromJson(extent);
  82. if (geoApi.proj.isSpatialRefEqual(this._map.spatialReference, extent.spatialReference)) {
  83. return realExtent;
  84. } else {
  85. return geoApi.proj.projectEsriExtent(realExtent, this._map.spatialReference);
  86. }
  87. }
  88. /**
  89. * Takes a location object in lat/long, converts to current map spatialReference using
  90. * reprojection method in geoApi, and zooms to the point.
  91. *
  92. * @function zoomToLatLong
  93. * @param {Object} location is a location object, containing geometries in the form of { longitude: <Number>, latitude: <Number> }
  94. */
  95. zoomToPoint ({ longitude, latitude }) {
  96. // get reprojected point and zoom to it
  97. const geoPt = geoApi.proj.localProjectPoint(4326, this._map.spatialReference.wkid,
  98. [parseFloat(longitude), parseFloat(latitude)]);
  99. const zoomPt = geoApi.proj.Point(geoPt[0], geoPt[1], this._map.spatialReference);
  100. // give preference to the layer closest to a 50k scale ratio which is ideal for zoom
  101. const sweetLod = Map.findClosestLOD(this._map.__tileInfo.lods, 50000);
  102. this._map.centerAndZoom(zoomPt, Math.max(sweetLod.level, 0));
  103. }
  104. /**
  105. * Finds the level of detail closest to the provided scale.
  106. *
  107. * @function findClosestLOD
  108. * @param {Array} lods list of levels of detail objects
  109. * @param {Number} scale scale value to search for in the levels of detail
  110. * @return {Object} the level of detail object closest to the scale
  111. */
  112. static findClosestLOD (lods, scale) {
  113. const diffs = lods.map(lod => Math.abs(lod.scale - scale));
  114. const lodIdx = diffs.indexOf(Math.min(...diffs));
  115. return lods[lodIdx];
  116. }
  117. /**
  118. * Calculate north arrow bearing. Angle returned is to to rotate north arrow image.
  119. * http://www.movable-type.co.uk/scripts/latlong.html
  120. * @function getNorthArrowAngle
  121. * @returns {Number} map rotation angle (in degree)
  122. */
  123. getNorthArrowAngle () {
  124. // get center point in longitude and use bottom value for latitude
  125. const pointB = geoApi.proj.localProjectPoint(this._map.extent.spatialReference, 'EPSG:4326',
  126. { x: (this._map.extent.xmin + this._map.extent.xmax) / 2, y: this._map.extent.ymin });
  127. // north value (set longitude to be half of Canada extent (141° W, 52° W))
  128. const pointA = { x: -96, y: 90 };
  129. // set info on longitude and latitude
  130. const dLon = (pointB.x - pointA.x) * Math.PI / 180;
  131. const lat1 = pointA.y * Math.PI / 180;
  132. const lat2 = pointB.y * Math.PI / 180;
  133. // calculate bearing
  134. const y = Math.sin(dLon) * Math.cos(lat2);
  135. const x = Math.cos(lat1) * Math.sin(lat2) - Math.sin(lat1) * Math.cos(lat2) * Math.cos(dLon);
  136. const bearing = Math.atan2(y, x) * 180 / Math.PI;
  137. // return angle (180 is pointiong north)
  138. return ((bearing + 360) % 360).toFixed(1);
  139. }
  140. /**
  141. * Calculate distance between min and max extent to know the pixel ratio between
  142. * screen size and earth distance.
  143. * http://www.movable-type.co.uk/scripts/latlong.html
  144. * @function getScaleRatio
  145. * @param {Number} mapWidth optional the map width to use to calculate ratio
  146. * @returns {Object} contain information about the scale
  147. * - distance: distance between min and max extentId
  148. * - ratio: measure for 1 pixel in earth distance
  149. * - units: array of units [metric, imperial]
  150. */
  151. getScaleRatio (mapWidth = 0) {
  152. const map = this._map;
  153. // get left and right maximum value point to calculate distance from
  154. const pointA = geoApi.proj.localProjectPoint(map.spatialReference, 'EPSG:4326',
  155. { x: map.extent.xmin, y: (map.extent.ymin + map.extent.ymax) / 2 });
  156. const pointB = geoApi.proj.localProjectPoint(map.spatialReference, 'EPSG:4326',
  157. { x: map.extent.xmax, y: (map.extent.ymin + map.extent.ymax) / 2 });
  158. // Haversine formula to calculate distance
  159. const R = 6371e3; // earth radius in meters
  160. const rad = Math.PI / 180;
  161. const phy1 = pointA.y * rad; // radiant
  162. const phy2 = pointB.y * rad; // radiant
  163. const deltaPhy = (pointB.y - pointA.y) * rad; // radiant
  164. const deltaLambda = (pointB.x - pointA.x) * rad; // radiant
  165. const a = Math.sin(deltaPhy / 2) * Math.sin(deltaPhy / 2) +
  166. Math.cos(phy1) * Math.cos(phy2) *
  167. Math.sin(deltaLambda / 2) * Math.sin(deltaLambda / 2);
  168. const c = 2 * Math.atan2(Math.sqrt(a), Math.sqrt(1 - a));
  169. const d = (R * c);
  170. // set map / image width (if mapWidth = 0, use map.width)
  171. const width = mapWidth ? mapWidth : map.width;
  172. // get unit from distance, set distance and ratio (earth size for 1 pixel)
  173. const units = [(d > 1000) ? 'km' : 'm', (d > 1600) ? 'mi' : 'ft'];
  174. const distance = (d > 1000) ? d / 1000 : d;
  175. const ratio = distance / width;
  176. return { distance, ratio, units };
  177. }
  178. /**
  179. * Compares to sets of co-ordinates for extents (valid for both x and y). If center of input co-ordinates falls outside
  180. * map co-ordiantes, function will adjust them so the center is inside the map co-ordinates.
  181. *
  182. * @function clipExtentCoords
  183. * @private
  184. * @param {Numeric} mid middle of the the range to test
  185. * @param {Numeric} max maximum value of the range to test
  186. * @param {Numeric} min minimum value of the range to test
  187. * @param {Numeric} mapMax maximum value of the map range
  188. * @param {Numeric} mapMin minimum value of the map range
  189. * @param {Numeric} len length of the adjusted range, if adjusted
  190. * @return {Array} two element array of Numeric, containing result max and min values
  191. */
  192. static clipExtentCoords (mid, max, min, mapMax, mapMin, len) {
  193. if (mid > mapMax) {
  194. [max, min] = [mapMax, mapMax - len];
  195. } else if (mid < mapMin) {
  196. [max, min] = [mapMin + len, mapMin];
  197. }
  198. return [max, min];
  199. }
  200. /**
  201. * Checks if the center of the given extent is outside of the maximum extent. If it is,
  202. * will determine an adjusted extent with a center inside the maximum extent. Returns both
  203. * an indicator flag if an adjustment happened, and the adjusted extent.
  204. *
  205. * @function enforceBoundary
  206. * @param {Object} extent an ESRI extent to test
  207. * @param {Object} maxExtent an ESRI extent indicating the boundary of the map
  208. * @return {Object} an object with two properties. adjusted - boolean, true if extent was adjusted. newExtent - object, adjusted ESRI extent
  209. */
  210. static enforceBoundary (extent, maxExtent) {
  211. // clone extent
  212. const newExtent = esriBundle.Extent(extent.toJson());
  213. // determine dimensions of adjusted extent.
  214. // same as input, unless input is so large it consumes max.
  215. // in that case, we shrink to the max. This avoids the "washing machine"
  216. // bug where we over-correct past the valid range,
  217. // and achieve infinite oscillating pans
  218. const height = Math.min(extent.getHeight(), maxExtent.getHeight());
  219. const width = Math.min(extent.getWidth(), maxExtent.getWidth());
  220. const center = extent.getCenter();
  221. [newExtent.xmax, newExtent.xmin] =
  222. this.clipExtentCoords(center.x, newExtent.xmax, newExtent.xmin, maxExtent.xmax, maxExtent.xmin, width);
  223. [newExtent.ymax, newExtent.ymin] =
  224. this.clipExtentCoords(center.y, newExtent.ymax, newExtent.ymin, maxExtent.ymax, maxExtent.ymin, height);
  225. return {
  226. newExtent,
  227. adjusted: !extent.contains(newExtent) // true if we adjusted the extent
  228. };
  229. }
  230. initOverviewMap () {
  231. this.overviewMap = new esriBundle.OverviewMap({ map: this._map, expandFactor: 1, visible: true });
  232. this.overviewMap.startup();
  233. }
  234. resetOverviewMap () {
  235. this.overviewMap.destroy();
  236. this.initOverviewMap();
  237. }
  238. /**
  239. * Changes the zoom level by the specified value relative to the current level; can be negative.
  240. * To avoid multiple chained zoom animations when rapidly pressing the zoom in/out icons, we
  241. * update the zoom level only when the one before it resolves with the net zoom change.
  242. *
  243. * @function shiftZoom
  244. * @param {number} byValue a number of zoom levels to shift by
  245. */
  246. shiftZoom (byValue) {
  247. this.zoomCounter += byValue;
  248. this.zoomPromise.then(() => {
  249. if (this.zoomCounter !== 0) {
  250. const zoomValue = this._map.getZoom() + this.zoomCounter;
  251. const zoomPromise = Promise.resolve(this.setZoom(zoomValue));
  252. this.zoomCounter = 0;
  253. // undefined signals we've zoomed in/out as far as we can
  254. if (typeof zoomPromise !== 'undefined') {
  255. this.zoomPromise = zoomPromise;
  256. }
  257. }
  258. });
  259. }
  260. /**
  261. * Sets or gets map default config values.
  262. *
  263. * @function mapDefault
  264. * @param {String} key name of the default property
  265. * @param {Any} [value] optional value to set for the specified default property
  266. */
  267. mapDefault (key, value) {
  268. if (typeof value === 'undefined') {
  269. return esriBundle.esriConfig.defaults.map[key];
  270. } else {
  271. esriBundle.esriConfig.defaults.map[key] = value;
  272. }
  273. }
  274. /**
  275. * Set proxy service URL to avoid same origin issues.
  276. */
  277. set proxy (proxyUrl) { esriBundle.esriConfig.defaults.io.proxyUrl = proxyUrl; }
  278. get proxy () { return esriBundle.esriConfig.defaults.io.proxyUrl; }
  279. set basemapGallery (val) { this._basemapGallery = val; }
  280. get basemapGallery () { return this._basemapGallery; }
  281. set scalebar (val) { this._scalebar = val; }
  282. get scalebar () { return this._scalebar; }
  283. set overviewMap (val) { this._overviewMap = val; }
  284. get overviewMap () { return this._overviewMap; }
  285. }
  286. return Map;
  287. }
  288. /**
  289. * The `MapManager` module exports an object with the following properties:
  290. * - `Extent` esri/geometry type
  291. * - `Map` esri/map type
  292. * - `OverviewMap` esri/dijit/OverviewMap type
  293. * - `Scalebar` sri/dijit/Scalebar type
  294. * - `getExtentFromSetting function to create an ESRI Extent object from extent setting JSON object.
  295. * - `setupMap` function that interates over config settings and apply logic for any items present.
  296. * - `setProxy` function to set proxy service URL to avoid same origin issues
  297. */
  298. // mapManager module, provides function to setup a map
  299. module.exports = (esriBundle, geoApi) => esriMap(esriBundle, geoApi);