BaseLayerPicker.js 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265
  1. /*global define*/
  2. define([
  3. '../../Core/defined',
  4. '../../Core/defineProperties',
  5. '../../Core/destroyObject',
  6. '../../Core/DeveloperError',
  7. '../../ThirdParty/knockout',
  8. '../getElement',
  9. './BaseLayerPickerViewModel'
  10. ], function(
  11. defined,
  12. defineProperties,
  13. destroyObject,
  14. DeveloperError,
  15. knockout,
  16. getElement,
  17. BaseLayerPickerViewModel) {
  18. "use strict";
  19. /**
  20. * <span style="display: block; text-align: center;">
  21. * <img src="images/BaseLayerPicker.png" style="border: none; border-radius: 5px;" />
  22. * <br />BaseLayerPicker with its drop-panel open.
  23. * </span>
  24. * <br /><br />
  25. * The BaseLayerPicker is a single button widget that displays a panel of available imagery and
  26. * terrain providers. When imagery is selected, the corresponding imagery layer is created and inserted
  27. * as the base layer of the imagery collection; removing the existing base. When terrain is selected,
  28. * it replaces the current terrain provider. Each item in the available providers list contains a name,
  29. * a representative icon, and a tooltip to display more information when hovered. The list is initially
  30. * empty, and must be configured before use, as illustrated in the below example.
  31. *
  32. * @alias BaseLayerPicker
  33. * @constructor
  34. *
  35. * @param {Element} container The parent HTML container node for this widget.
  36. * @param {Object} options Object with the following properties:
  37. * @param {Globe} options.globe The Globe to use.
  38. * @param {ProviderViewModel[]} [options.imageryProviderViewModels=[]] The array of ProviderViewModel instances to use for imagery.
  39. * @param {ProviderViewModel} [options.selectedImageryProviderViewModel] The view model for the current base imagery layer, if not supplied the first available imagery layer is used.
  40. * @param {ProviderViewModel[]} [options.terrainProviderViewModels=[]] The array of ProviderViewModel instances to use for terrain.
  41. * @param {ProviderViewModel} [options.selectedTerrainProviderViewModel] The view model for the current base terrain layer, if not supplied the first available terrain layer is used.
  42. *
  43. * @exception {DeveloperError} Element with id "container" does not exist in the document.
  44. *
  45. * @see TerrainProvider
  46. * @see ImageryProvider
  47. * @see ImageryLayerCollection
  48. *
  49. * @example
  50. * // In HTML head, include a link to the BaseLayerPicker.css stylesheet,
  51. * // and in the body, include: <div id="baseLayerPickerContainer"
  52. * // style="position:absolute;top:24px;right:24px;width:38px;height:38px;"></div>
  53. *
  54. * //Create the list of available providers we would like the user to select from.
  55. * //This example uses 3, OpenStreetMap, The Black Marble, and a single, non-streaming world image.
  56. * var imageryViewModels = [];
  57. * imageryViewModels.push(new Cesium.ProviderViewModel({
  58. * name : 'Open\u00adStreet\u00adMap',
  59. * iconUrl : Cesium.buildModuleUrl('Widgets/Images/ImageryProviders/openStreetMap.png'),
  60. * tooltip : 'OpenStreetMap (OSM) is a collaborative project to create a free editable \
  61. *map of the world.\nhttp://www.openstreetmap.org',
  62. * creationFunction : function() {
  63. * return new Cesium.OpenStreetMapImageryProvider({
  64. * url : '//a.tile.openstreetmap.org/'
  65. * });
  66. * }
  67. * }));
  68. *
  69. * imageryViewModels.push(new Cesium.ProviderViewModel({
  70. * name : 'Black Marble',
  71. * iconUrl : Cesium.buildModuleUrl('Widgets/Images/ImageryProviders/blackMarble.png'),
  72. * tooltip : 'The lights of cities and villages trace the outlines of civilization \
  73. *in this global view of the Earth at night as seen by NASA/NOAA\'s Suomi NPP satellite.',
  74. * creationFunction : function() {
  75. * return new Cesium.TileMapServiceImageryProvider({
  76. * url : '//cesiumjs.org/blackmarble',
  77. * maximumLevel : 8,
  78. * credit : 'Black Marble imagery courtesy NASA Earth Observatory'
  79. * });
  80. * }
  81. * }));
  82. *
  83. * imageryViewModels.push(new Cesium.ProviderViewModel({
  84. * name : 'Natural Earth\u00a0II',
  85. * iconUrl : Cesium.buildModuleUrl('Widgets/Images/ImageryProviders/naturalEarthII.png'),
  86. * tooltip : 'Natural Earth II, darkened for contrast.\nhttp://www.naturalearthdata.com/',
  87. * creationFunction : function() {
  88. * return new Cesium.TileMapServiceImageryProvider({
  89. * url : Cesium.buildModuleUrl('Assets/Textures/NaturalEarthII')
  90. * });
  91. * }
  92. * }));
  93. *
  94. * //Create a CesiumWidget without imagery, if you haven't already done so.
  95. * var cesiumWidget = new Cesium.CesiumWidget('cesiumContainer', { imageryProvider: false });
  96. *
  97. * //Finally, create the baseLayerPicker widget using our view models.
  98. * var layers = cesiumWidget.imageryLayers;
  99. * var baseLayerPicker = new Cesium.BaseLayerPicker('baseLayerPickerContainer', layers, imageryViewModels);
  100. *
  101. * //Use the first item in the list as the current selection.
  102. * baseLayerPicker.viewModel.selectedItem = imageryViewModels[0];
  103. */
  104. var BaseLayerPicker = function(container, options) {
  105. //>>includeStart('debug', pragmas.debug);
  106. if (!defined(container)) {
  107. throw new DeveloperError('container is required.');
  108. }
  109. //>>includeEnd('debug');
  110. container = getElement(container);
  111. var viewModel = new BaseLayerPickerViewModel(options);
  112. var element = document.createElement('button');
  113. element.type = 'button';
  114. element.className = 'cesium-button cesium-toolbar-button';
  115. element.setAttribute('data-bind', '\
  116. attr: { title: buttonTooltip },\
  117. click: toggleDropDown');
  118. container.appendChild(element);
  119. var imgElement = document.createElement('img');
  120. imgElement.setAttribute('draggable', 'false');
  121. imgElement.className = 'cesium-baseLayerPicker-selected';
  122. imgElement.setAttribute('data-bind', '\
  123. attr: { src: buttonImageUrl }');
  124. element.appendChild(imgElement);
  125. var dropPanel = document.createElement('div');
  126. dropPanel.className = 'cesium-baseLayerPicker-dropDown';
  127. dropPanel.setAttribute('data-bind', '\
  128. css: { "cesium-baseLayerPicker-dropDown-visible" : dropDownVisible }');
  129. container.appendChild(dropPanel);
  130. var imageryTitle = document.createElement('div');
  131. imageryTitle.className = 'cesium-baseLayerPicker-sectionTitle';
  132. imageryTitle.setAttribute('data-bind', 'visible: imageryProviderViewModels.length > 0');
  133. imageryTitle.innerHTML = 'Imagery';
  134. dropPanel.appendChild(imageryTitle);
  135. var imageryChoices = document.createElement('div');
  136. imageryChoices.className = 'cesium-baseLayerPicker-choices';
  137. imageryChoices.setAttribute('data-bind', 'foreach: imageryProviderViewModels');
  138. dropPanel.appendChild(imageryChoices);
  139. var imageryProvider = document.createElement('div');
  140. imageryProvider.className = 'cesium-baseLayerPicker-item';
  141. imageryProvider.setAttribute('data-bind', '\
  142. css: { "cesium-baseLayerPicker-selectedItem" : $data === $parent.selectedImagery },\
  143. attr: { title: tooltip },\
  144. visible: creationCommand.canExecute,\
  145. click: function($data) { $parent.selectedImagery = $data; }');
  146. imageryChoices.appendChild(imageryProvider);
  147. var providerIcon = document.createElement('img');
  148. providerIcon.className = 'cesium-baseLayerPicker-itemIcon';
  149. providerIcon.setAttribute('data-bind', 'attr: { src: iconUrl }');
  150. providerIcon.setAttribute('draggable', 'false');
  151. imageryProvider.appendChild(providerIcon);
  152. var providerLabel = document.createElement('div');
  153. providerLabel.className = 'cesium-baseLayerPicker-itemLabel';
  154. providerLabel.setAttribute('data-bind', 'text: name');
  155. imageryProvider.appendChild(providerLabel);
  156. var terrainTitle = document.createElement('div');
  157. terrainTitle.className = 'cesium-baseLayerPicker-sectionTitle';
  158. terrainTitle.setAttribute('data-bind', 'visible: terrainProviderViewModels.length > 0');
  159. terrainTitle.innerHTML = 'Terrain';
  160. dropPanel.appendChild(terrainTitle);
  161. var terrainChoices = document.createElement('div');
  162. terrainChoices.className = 'cesium-baseLayerPicker-choices';
  163. terrainChoices.setAttribute('data-bind', 'foreach: terrainProviderViewModels');
  164. dropPanel.appendChild(terrainChoices);
  165. var terrainProvider = document.createElement('div');
  166. terrainProvider.className = 'cesium-baseLayerPicker-item';
  167. terrainProvider.setAttribute('data-bind', '\
  168. css: { "cesium-baseLayerPicker-selectedItem" : $data === $parent.selectedTerrain },\
  169. attr: { title: tooltip },\
  170. visible: creationCommand.canExecute,\
  171. click: function($data) { $parent.selectedTerrain = $data; }');
  172. terrainChoices.appendChild(terrainProvider);
  173. var terrainProviderIcon = document.createElement('img');
  174. terrainProviderIcon.className = 'cesium-baseLayerPicker-itemIcon';
  175. terrainProviderIcon.setAttribute('data-bind', 'attr: { src: iconUrl }');
  176. terrainProviderIcon.setAttribute('draggable', 'false');
  177. terrainProvider.appendChild(terrainProviderIcon);
  178. var terrainProviderLabel = document.createElement('div');
  179. terrainProviderLabel.className = 'cesium-baseLayerPicker-itemLabel';
  180. terrainProviderLabel.setAttribute('data-bind', 'text: name');
  181. terrainProvider.appendChild(terrainProviderLabel);
  182. knockout.applyBindings(viewModel, element);
  183. knockout.applyBindings(viewModel, dropPanel);
  184. this._viewModel = viewModel;
  185. this._container = container;
  186. this._element = element;
  187. this._dropPanel = dropPanel;
  188. this._closeDropDown = function(e) {
  189. if (!(element.contains(e.target) || dropPanel.contains(e.target))) {
  190. viewModel.dropDownVisible = false;
  191. }
  192. };
  193. document.addEventListener('mousedown', this._closeDropDown, true);
  194. document.addEventListener('touchstart', this._closeDropDown, true);
  195. };
  196. defineProperties(BaseLayerPicker.prototype, {
  197. /**
  198. * Gets the parent container.
  199. * @memberof BaseLayerPicker.prototype
  200. *
  201. * @type {Element}
  202. */
  203. container : {
  204. get : function() {
  205. return this._container;
  206. }
  207. },
  208. /**
  209. * Gets the view model.
  210. * @memberof BaseLayerPicker.prototype
  211. *
  212. * @type {BaseLayerPickerViewModel}
  213. */
  214. viewModel : {
  215. get : function() {
  216. return this._viewModel;
  217. }
  218. }
  219. });
  220. /**
  221. * @returns {Boolean} true if the object has been destroyed, false otherwise.
  222. */
  223. BaseLayerPicker.prototype.isDestroyed = function() {
  224. return false;
  225. };
  226. /**
  227. * Destroys the widget. Should be called if permanently
  228. * removing the widget from layout.
  229. */
  230. BaseLayerPicker.prototype.destroy = function() {
  231. document.removeEventListener('mousedown', this._closeDropDown, true);
  232. document.removeEventListener('touchstart', this._closeDropDown, true);
  233. knockout.cleanNode(this._element);
  234. knockout.cleanNode(this._dropPanel);
  235. this._container.removeChild(this._element);
  236. this._container.removeChild(this._dropPanel);
  237. return destroyObject(this);
  238. };
  239. return BaseLayerPicker;
  240. });