BaseLayerPicker.js 13 KB

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