layer/layerRec/layerInterface.js

  1. 'use strict';
  2. // Controls Interface class is used to provide something to the UI that it can bind to.
  3. // It helps the UI keep in line with the layer state.
  4. // Due to bindings, we cannot destroy & recreate an interface when a legend item
  5. // goes from 'Unknown Placeholder' to 'Specific Layer Type'. This means we cannot
  6. // do object heirarchies, as to go from PlaceholderInterface to FeatureLayerInterface
  7. // would require a new object. Instead, we have a class that exposes all possible
  8. // methods and properties as error throwing stubs. Then we replace those functions
  9. // with real ones once we know the flavour of interface we want.
  10. class LayerInterface {
  11. /**
  12. * @param {Object} source object that provides info to the interface. usually a LayerRecord or FeatureClass
  13. */
  14. constructor (source) {
  15. this._source = source;
  16. // TODO revisit isPlaceholder after grand refactor
  17. this._isPlaceholder = true;
  18. }
  19. get isPlaceholder () { return this._isPlaceholder; } // returns Boolean
  20. // these expose ui controls available on the interface and indicate which ones are disabled
  21. get symbology () { return undefined; } // returns Array
  22. // can be group or node name
  23. get name () { return undefined; } // returns String
  24. // these are needed for the type flag
  25. get layerType () { return undefined; } // returns String
  26. get parentLayerType () { return undefined; } // returns String
  27. get geometryType () { return undefined; } // returns String
  28. get featureCount () { return undefined; } // returns Integer
  29. get extent () { return undefined; } // returns Object (Esri Extent)
  30. // layer states
  31. get state () { return undefined; } // returns String
  32. // these return the current values of the corresponding controls
  33. get visibility () { return undefined; } // returns Boolean
  34. get opacity () { return undefined; } // returns Decimal
  35. get query () { return undefined; } // returns Boolean
  36. get snapshot () { return undefined; } // returns Boolean
  37. get highlightFeature () { return undefined; } // returns Boolean
  38. // fetches attributes for use in the datatable
  39. get formattedAttributes () { return undefined; } // returns Promise of Object
  40. // returns a feature name of an attribute set
  41. getFeatureName () { return undefined; } // returns String
  42. // formats an attribute to standard details
  43. attributesToDetails () { return undefined; } // returns Array
  44. // these set values to the corresponding controls
  45. // param: boolean
  46. setVisibility () { return undefined; }
  47. // param: numeric (between 0 and 1)
  48. setOpacity () { return undefined; }
  49. // param: boolean
  50. setQuery () { return undefined; }
  51. // param: integer, boolean (optional)
  52. fetchGraphic () { return undefined; } // returns promise that resolves with object containing graphics info
  53. // param: esriMap
  54. zoomToBoundary () { return undefined; } // returns promise that resolves after zoom completes
  55. // param: esriMap, array of lods, boolean, optional boolean
  56. zoomToScale () { return undefined; } // returns promise that resolves after zoom completes
  57. // param: Integer, esriMap, Object {x: Numeric, y: Numeric}
  58. zoomToGraphic () { return undefined; } // returns promise that resolves after zoom completes
  59. // param: string
  60. setDefinitionQuery () { return undefined; }
  61. // param: spatialReference object
  62. validateProjection () { return undefined; } // returns Boolean
  63. // param: integer
  64. isOffScale () { return undefined; } // returns Object {offScale: Boolean, zoomIn: Boolean}
  65. // updates what this interface is pointing to, in terms of layer data source.
  66. // often, the interface starts with a placeholder to avoid errors and return
  67. // defaults. This update happens after a layer has loaded, and new now want
  68. // the interface reading off the real FC.
  69. // TODO docs
  70. updateSource (newSource) {
  71. this._source = newSource;
  72. }
  73. convertToSingleLayer (layerRecord) {
  74. this._source = layerRecord;
  75. this._isPlaceholder = false;
  76. newProp(this, 'symbology', standardGetSymbology);
  77. newProp(this, 'state', standardGetState);
  78. newProp(this, 'visibility', standardGetVisibility);
  79. newProp(this, 'opacity', standardGetOpacity);
  80. newProp(this, 'query', standardGetQuery);
  81. newProp(this, 'name', standardGetName);
  82. newProp(this, 'geometryType', standardGetGeometryType);
  83. newProp(this, 'layerType', standardGetLayerType);
  84. newProp(this, 'parentLayerType', standardGetParentLayerType);
  85. newProp(this, 'featureCount', standardGetFeatureCount);
  86. newProp(this, 'extent', standardGetExtent);
  87. this.setVisibility = standardSetVisibility;
  88. this.setOpacity = standardSetOpacity;
  89. this.setQuery = standardSetQuery;
  90. this.zoomToBoundary = standardZoomToBoundary;
  91. this.validateProjection = standardValidateProjection;
  92. this.zoomToScale = standardZoomToScale;
  93. this.isOffScale = standardIsOffScale;
  94. }
  95. convertToFeatureLayer (layerRecord) {
  96. this.convertToSingleLayer(layerRecord);
  97. newProp(this, 'snapshot', featureGetSnapshot);
  98. newProp(this, 'formattedAttributes', standardGetFormattedAttributes);
  99. newProp(this, 'geometryType', featureGetGeometryType);
  100. newProp(this, 'featureCount', featureGetFeatureCount);
  101. newProp(this, 'highlightFeature', featureGetHighlightFeature);
  102. this.getFeatureName = featureGetFeatureName;
  103. this.attributesToDetails = featureAttributesToDetails;
  104. this.fetchGraphic = featureFetchGraphic;
  105. this.setDefinitionQuery = featureSetDefinitionQuery;
  106. this.zoomToGraphic = featureZoomToGraphic;
  107. }
  108. convertToDynamicLeaf (dynamicFC) {
  109. this._source = dynamicFC;
  110. this._isPlaceholder = false;
  111. newProp(this, 'symbology', dynamicLeafGetSymbology);
  112. newProp(this, 'state', dynamicLeafGetState);
  113. newProp(this, 'name', dynamicLeafGetName);
  114. newProp(this, 'visibility', dynamicLeafGetVisibility);
  115. newProp(this, 'opacity', dynamicLeafGetOpacity);
  116. newProp(this, 'query', dynamicLeafGetQuery);
  117. newProp(this, 'formattedAttributes', dynamicLeafGetFormattedAttributes);
  118. newProp(this, 'geometryType', dynamicLeafGetGeometryType);
  119. newProp(this, 'layerType', dynamicLeafGetLayerType);
  120. newProp(this, 'parentLayerType', dynamicLeafGetParentLayerType);
  121. newProp(this, 'featureCount', dynamicLeafGetFeatureCount);
  122. newProp(this, 'extent', dynamicLeafGetExtent);
  123. newProp(this, 'highlightFeature', dynamicLeafGetHighlightFeature);
  124. this.setVisibility = dynamicLeafSetVisibility;
  125. this.setOpacity = dynamicLeafSetOpacity;
  126. this.setQuery = dynamicLeafSetQuery;
  127. this.zoomToBoundary = dynamicLeafZoomToBoundary;
  128. this.zoomToScale = dynamicLeafZoomToScale;
  129. this.getFeatureName = featureGetFeatureName;
  130. this.attributesToDetails = dynamicLeafAttributesToDetails;
  131. this.fetchGraphic = dynamicLeafFetchGraphic;
  132. this.setDefinitionQuery = dynamicLeafSetDefinitionQuery;
  133. this.isOffScale = dynamicLeafIsOffScale;
  134. this.zoomToGraphic = dynamicLeafZoomToGraphic;
  135. }
  136. convertToPlaceholder (placeholderFC) {
  137. this._source = placeholderFC;
  138. this._isPlaceholder = true;
  139. newProp(this, 'symbology', standardGetSymbology);
  140. newProp(this, 'name', standardGetName);
  141. newProp(this, 'state', standardGetState);
  142. newProp(this, 'layerType', standardGetLayerType);
  143. newProp(this, 'parentLayerType', standardGetParentLayerType);
  144. }
  145. // TODO we might need a `converToDynamicLayer`. It calls `converToSingleLayer` and then
  146. // removes zoomToScale method, as this method will fail if called directly on a
  147. // dynamic layer proxy. As is, nothing should ever make that call. But we can
  148. // do this to be safe.
  149. }
  150. /**
  151. * Worker function to add or override a get property on an object
  152. *
  153. * @function newProp
  154. * @private
  155. * @param {Object} target the object that will receive the new property
  156. * @param {String} propName name of the get property
  157. * @param {Function} getter the function defining the guts of the get property.
  158. */
  159. function newProp(target, propName, getter) {
  160. Object.defineProperty(target, propName, {
  161. get: getter,
  162. enumerable: true,
  163. configurable: true
  164. });
  165. }
  166. // these functions are upgrades to the duds above.
  167. // we don't use arrow notation, as we want the `this` to point at the object
  168. // that these functions get smashed into.
  169. function standardGetState() {
  170. /* jshint validthis: true */
  171. return this._source.state;
  172. }
  173. function dynamicLeafGetState() {
  174. /* jshint validthis: true */
  175. return this._source.state;
  176. }
  177. function standardGetVisibility() {
  178. /* jshint validthis: true */
  179. return this._source.visibility;
  180. }
  181. function dynamicLeafGetVisibility() {
  182. /* jshint validthis: true */
  183. return this._source.getVisibility();
  184. }
  185. function standardGetName() {
  186. /* jshint validthis: true */
  187. return this._source.name;
  188. }
  189. function dynamicLeafGetName() {
  190. /* jshint validthis: true */
  191. return this._source.name;
  192. }
  193. function standardGetOpacity() {
  194. /* jshint validthis: true */
  195. return this._source.opacity;
  196. }
  197. function dynamicLeafGetOpacity() {
  198. /* jshint validthis: true */
  199. return this._source.opacity;
  200. }
  201. function standardGetLayerType() {
  202. /* jshint validthis: true */
  203. return this._source.layerType;
  204. }
  205. function dynamicLeafGetLayerType() {
  206. /* jshint validthis: true */
  207. return this._source.layerType;
  208. }
  209. function standardGetParentLayerType() {
  210. /* jshint validthis: true */
  211. return this._source.parentLayerType;
  212. }
  213. function dynamicLeafGetParentLayerType() {
  214. /* jshint validthis: true */
  215. return this._source.parentLayerType;
  216. }
  217. function standardGetExtent() {
  218. /* jshint validthis: true */
  219. return this._source.extent;
  220. }
  221. function dynamicLeafGetExtent() {
  222. /* jshint validthis: true */
  223. return this._source.extent;
  224. }
  225. function standardGetQuery() {
  226. /* jshint validthis: true */
  227. return this._source.isQueryable();
  228. }
  229. function dynamicLeafGetQuery() {
  230. /* jshint validthis: true */
  231. return this._source.queryable;
  232. }
  233. function standardGetFormattedAttributes() {
  234. /* jshint validthis: true */
  235. return this._source.getFormattedAttributes();
  236. }
  237. function dynamicLeafGetFormattedAttributes() {
  238. /* jshint validthis: true */
  239. // TODO code-wise this looks identical to standardGetFormattedAttributes.
  240. // however in this case, ._source is a DynamicFC, not a LayerRecord.
  241. // This is safer. Deleting this would avoid the duplication. Decide.
  242. return this._source.getFormattedAttributes();
  243. }
  244. function standardGetSymbology() {
  245. /* jshint validthis: true */
  246. return this._source.symbology;
  247. }
  248. function dynamicLeafGetSymbology() {
  249. /* jshint validthis: true */
  250. // TODO code-wise this looks identical to standardGetSymbology.
  251. // however in this case, ._source is a DynamicFC, not a LayerRecord.
  252. // This is safer. Deleting this would avoid the duplication. Decide.
  253. return this._source.symbology;
  254. }
  255. function standardGetGeometryType() {
  256. /* jshint validthis: true */
  257. return undefined;
  258. }
  259. function featureGetGeometryType() {
  260. /* jshint validthis: true */
  261. return this._source.getGeomType();
  262. }
  263. function dynamicLeafGetGeometryType() {
  264. /* jshint validthis: true */
  265. // TEST STATUS none
  266. return this._source.geomType;
  267. }
  268. function standardGetFeatureCount() {
  269. /* jshint validthis: true */
  270. return undefined;
  271. }
  272. function featureGetFeatureCount() {
  273. /* jshint validthis: true */
  274. return this._source.featureCount;
  275. }
  276. function dynamicLeafGetFeatureCount() {
  277. /* jshint validthis: true */
  278. return this._source.featureCount;
  279. }
  280. function standardSetVisibility(value) {
  281. /* jshint validthis: true */
  282. this._source.visibility = value;
  283. }
  284. function dynamicLeafSetVisibility(value) {
  285. /* jshint validthis: true */
  286. this._source.setVisibility(value);
  287. }
  288. function standardSetOpacity(value) {
  289. /* jshint validthis: true */
  290. this._source.opacity = value;
  291. }
  292. function dynamicLeafSetOpacity(value) {
  293. /* jshint validthis: true */
  294. this._source.opacity = value;
  295. }
  296. function standardSetQuery(value) {
  297. /* jshint validthis: true */
  298. this._source.setQueryable(value);
  299. }
  300. function dynamicLeafSetQuery(value) {
  301. /* jshint validthis: true */
  302. this._source.queryable = value;
  303. }
  304. function featureGetSnapshot() {
  305. /* jshint validthis: true */
  306. return this._source.isSnapshot;
  307. }
  308. function featureGetHighlightFeature() {
  309. return true;
  310. }
  311. function dynamicLeafGetHighlightFeature() {
  312. /* jshint validthis: true */
  313. return this._source.highlightFeature;
  314. }
  315. function standardZoomToBoundary(map) {
  316. /* jshint validthis: true */
  317. this._source.zoomToBoundary(map);
  318. }
  319. function dynamicLeafZoomToBoundary(map) {
  320. /* jshint validthis: true */
  321. this._source.zoomToBoundary(map);
  322. }
  323. function standardZoomToScale(map, lods, zoomIn, positionOverLayer = true) {
  324. /* jshint validthis: true */
  325. return this._source.zoomToScale(map, lods, zoomIn, positionOverLayer);
  326. }
  327. function dynamicLeafZoomToScale(map, lods, zoomIn, positionOverLayer = true) {
  328. /* jshint validthis: true */
  329. return this._source.zoomToScale(map, lods, zoomIn, positionOverLayer);
  330. }
  331. function featureGetFeatureName(objId, attribs) {
  332. /* jshint validthis: true */
  333. return this._source.getFeatureName(objId, attribs);
  334. }
  335. function featureAttributesToDetails(attribs, fields) {
  336. /* jshint validthis: true */
  337. return this._source.attributesToDetails(attribs, fields);
  338. }
  339. function dynamicLeafAttributesToDetails(attribs, fields) {
  340. /* jshint validthis: true */
  341. return this._source._parent.attributesToDetails(attribs, fields);
  342. }
  343. function featureFetchGraphic(oid, ignoreLocal = false) {
  344. /* jshint validthis: true */
  345. return this._source.fetchGraphic(oid, ignoreLocal);
  346. }
  347. function dynamicLeafFetchGraphic(oid, ignoreLocal = false) {
  348. /* jshint validthis: true */
  349. return this._source.fetchGraphic(oid, ignoreLocal);
  350. }
  351. function featureZoomToGraphic(oid, map, offsetFraction) {
  352. /* jshint validthis: true */
  353. return this._source.zoomToGraphic(oid, map, offsetFraction);
  354. }
  355. function dynamicLeafZoomToGraphic(oid, map, offsetFraction) {
  356. /* jshint validthis: true */
  357. return this._source.zoomToGraphic(oid, map, offsetFraction);
  358. }
  359. function featureSetDefinitionQuery(query) {
  360. /* jshint validthis: true */
  361. this._source.setDefinitionQuery(query);
  362. }
  363. function dynamicLeafSetDefinitionQuery(query) {
  364. /* jshint validthis: true */
  365. this._source.setDefinitionQuery(query);
  366. }
  367. function standardValidateProjection(spatialReference) {
  368. /* jshint validthis: true */
  369. return this._source.validateProjection(spatialReference);
  370. }
  371. function standardIsOffScale(mapScale) {
  372. /* jshint validthis: true */
  373. return this._source.isOffScale(mapScale);
  374. }
  375. function dynamicLeafIsOffScale(mapScale) {
  376. /* jshint validthis: true */
  377. return this._source.isOffScale(mapScale);
  378. }
  379. module.exports = () => ({
  380. LayerInterface
  381. });