createVerticesFromQuantizedTerrainMesh.js 33 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057
  1. /**
  2. * Cesium - https://github.com/CesiumGS/cesium
  3. *
  4. * Copyright 2011-2020 Cesium Contributors
  5. *
  6. * Licensed under the Apache License, Version 2.0 (the "License");
  7. * you may not use this file except in compliance with the License.
  8. * You may obtain a copy of the License at
  9. *
  10. * http://www.apache.org/licenses/LICENSE-2.0
  11. *
  12. * Unless required by applicable law or agreed to in writing, software
  13. * distributed under the License is distributed on an "AS IS" BASIS,
  14. * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  15. * See the License for the specific language governing permissions and
  16. * limitations under the License.
  17. *
  18. * Columbus View (Pat. Pend.)
  19. *
  20. * Portions licensed separately.
  21. * See https://github.com/CesiumGS/cesium/blob/master/LICENSE.md for full licensing details.
  22. */
  23. define(['./when-54c2dc71', './Check-6c0211bc', './Math-1124a290', './Cartesian2-33d2657c', './Transforms-8be64844', './RuntimeError-2109023a', './WebGLConstants-76bb35d1', './ComponentDatatype-a26dd044', './AttributeCompression-75249b5e', './IndexDatatype-25023891', './IntersectionTests-afc38163', './Plane-fa30fc46', './WebMercatorProjection-054890ef', './createTaskProcessorWorker', './EllipsoidTangentPlane-ce6e380f', './OrientedBoundingBox-8897f474', './TerrainEncoding-6954276f'], function (when, Check, _Math, Cartesian2, Transforms, RuntimeError, WebGLConstants, ComponentDatatype, AttributeCompression, IndexDatatype, IntersectionTests, Plane, WebMercatorProjection, createTaskProcessorWorker, EllipsoidTangentPlane, OrientedBoundingBox, TerrainEncoding) { 'use strict';
  24. /**
  25. * Provides terrain or other geometry for the surface of an ellipsoid. The surface geometry is
  26. * organized into a pyramid of tiles according to a {@link TilingScheme}. This type describes an
  27. * interface and is not intended to be instantiated directly.
  28. *
  29. * @alias TerrainProvider
  30. * @constructor
  31. *
  32. * @see EllipsoidTerrainProvider
  33. * @see CesiumTerrainProvider
  34. * @see VRTheWorldTerrainProvider
  35. * @see GoogleEarthEnterpriseTerrainProvider
  36. */
  37. function TerrainProvider() {
  38. Check.DeveloperError.throwInstantiationError();
  39. }
  40. Object.defineProperties(TerrainProvider.prototype, {
  41. /**
  42. * Gets an event that is raised when the terrain provider encounters an asynchronous error.. By subscribing
  43. * to the event, you will be notified of the error and can potentially recover from it. Event listeners
  44. * are passed an instance of {@link TileProviderError}.
  45. * @memberof TerrainProvider.prototype
  46. * @type {Event}
  47. */
  48. errorEvent: {
  49. get: Check.DeveloperError.throwInstantiationError,
  50. },
  51. /**
  52. * Gets the credit to display when this terrain provider is active. Typically this is used to credit
  53. * the source of the terrain. This function should
  54. * not be called before {@link TerrainProvider#ready} returns true.
  55. * @memberof TerrainProvider.prototype
  56. * @type {Credit}
  57. */
  58. credit: {
  59. get: Check.DeveloperError.throwInstantiationError,
  60. },
  61. /**
  62. * Gets the tiling scheme used by the provider. This function should
  63. * not be called before {@link TerrainProvider#ready} returns true.
  64. * @memberof TerrainProvider.prototype
  65. * @type {TilingScheme}
  66. */
  67. tilingScheme: {
  68. get: Check.DeveloperError.throwInstantiationError,
  69. },
  70. /**
  71. * Gets a value indicating whether or not the provider is ready for use.
  72. * @memberof TerrainProvider.prototype
  73. * @type {Boolean}
  74. */
  75. ready: {
  76. get: Check.DeveloperError.throwInstantiationError,
  77. },
  78. /**
  79. * Gets a promise that resolves to true when the provider is ready for use.
  80. * @memberof TerrainProvider.prototype
  81. * @type {Promise.<Boolean>}
  82. * @readonly
  83. */
  84. readyPromise: {
  85. get: Check.DeveloperError.throwInstantiationError,
  86. },
  87. /**
  88. * Gets a value indicating whether or not the provider includes a water mask. The water mask
  89. * indicates which areas of the globe are water rather than land, so they can be rendered
  90. * as a reflective surface with animated waves. This function should not be
  91. * called before {@link TerrainProvider#ready} returns true.
  92. * @memberof TerrainProvider.prototype
  93. * @type {Boolean}
  94. */
  95. hasWaterMask: {
  96. get: Check.DeveloperError.throwInstantiationError,
  97. },
  98. /**
  99. * Gets a value indicating whether or not the requested tiles include vertex normals.
  100. * This function should not be called before {@link TerrainProvider#ready} returns true.
  101. * @memberof TerrainProvider.prototype
  102. * @type {Boolean}
  103. */
  104. hasVertexNormals: {
  105. get: Check.DeveloperError.throwInstantiationError,
  106. },
  107. /**
  108. * Gets an object that can be used to determine availability of terrain from this provider, such as
  109. * at points and in rectangles. This function should not be called before
  110. * {@link TerrainProvider#ready} returns true. This property may be undefined if availability
  111. * information is not available.
  112. * @memberof TerrainProvider.prototype
  113. * @type {TileAvailability}
  114. */
  115. availability: {
  116. get: Check.DeveloperError.throwInstantiationError,
  117. },
  118. });
  119. var regularGridIndicesCache = [];
  120. /**
  121. * Gets a list of indices for a triangle mesh representing a regular grid. Calling
  122. * this function multiple times with the same grid width and height returns the
  123. * same list of indices. The total number of vertices must be less than or equal
  124. * to 65536.
  125. *
  126. * @param {Number} width The number of vertices in the regular grid in the horizontal direction.
  127. * @param {Number} height The number of vertices in the regular grid in the vertical direction.
  128. * @returns {Uint16Array|Uint32Array} The list of indices. Uint16Array gets returned for 64KB or less and Uint32Array for 4GB or less.
  129. */
  130. TerrainProvider.getRegularGridIndices = function (width, height) {
  131. //>>includeStart('debug', pragmas.debug);
  132. if (width * height >= _Math.CesiumMath.FOUR_GIGABYTES) {
  133. throw new Check.DeveloperError(
  134. "The total number of vertices (width * height) must be less than 4,294,967,296."
  135. );
  136. }
  137. //>>includeEnd('debug');
  138. var byWidth = regularGridIndicesCache[width];
  139. if (!when.defined(byWidth)) {
  140. regularGridIndicesCache[width] = byWidth = [];
  141. }
  142. var indices = byWidth[height];
  143. if (!when.defined(indices)) {
  144. if (width * height < _Math.CesiumMath.SIXTY_FOUR_KILOBYTES) {
  145. indices = byWidth[height] = new Uint16Array(
  146. (width - 1) * (height - 1) * 6
  147. );
  148. } else {
  149. indices = byWidth[height] = new Uint32Array(
  150. (width - 1) * (height - 1) * 6
  151. );
  152. }
  153. addRegularGridIndices(width, height, indices, 0);
  154. }
  155. return indices;
  156. };
  157. var regularGridAndEdgeIndicesCache = [];
  158. /**
  159. * @private
  160. */
  161. TerrainProvider.getRegularGridIndicesAndEdgeIndices = function (width, height) {
  162. //>>includeStart('debug', pragmas.debug);
  163. if (width * height >= _Math.CesiumMath.FOUR_GIGABYTES) {
  164. throw new Check.DeveloperError(
  165. "The total number of vertices (width * height) must be less than 4,294,967,296."
  166. );
  167. }
  168. //>>includeEnd('debug');
  169. var byWidth = regularGridAndEdgeIndicesCache[width];
  170. if (!when.defined(byWidth)) {
  171. regularGridAndEdgeIndicesCache[width] = byWidth = [];
  172. }
  173. var indicesAndEdges = byWidth[height];
  174. if (!when.defined(indicesAndEdges)) {
  175. var indices = TerrainProvider.getRegularGridIndices(width, height);
  176. var edgeIndices = getEdgeIndices(width, height);
  177. var westIndicesSouthToNorth = edgeIndices.westIndicesSouthToNorth;
  178. var southIndicesEastToWest = edgeIndices.southIndicesEastToWest;
  179. var eastIndicesNorthToSouth = edgeIndices.eastIndicesNorthToSouth;
  180. var northIndicesWestToEast = edgeIndices.northIndicesWestToEast;
  181. indicesAndEdges = byWidth[height] = {
  182. indices: indices,
  183. westIndicesSouthToNorth: westIndicesSouthToNorth,
  184. southIndicesEastToWest: southIndicesEastToWest,
  185. eastIndicesNorthToSouth: eastIndicesNorthToSouth,
  186. northIndicesWestToEast: northIndicesWestToEast,
  187. };
  188. }
  189. return indicesAndEdges;
  190. };
  191. var regularGridAndSkirtAndEdgeIndicesCache = [];
  192. /**
  193. * @private
  194. */
  195. TerrainProvider.getRegularGridAndSkirtIndicesAndEdgeIndices = function (
  196. width,
  197. height
  198. ) {
  199. //>>includeStart('debug', pragmas.debug);
  200. if (width * height >= _Math.CesiumMath.FOUR_GIGABYTES) {
  201. throw new Check.DeveloperError(
  202. "The total number of vertices (width * height) must be less than 4,294,967,296."
  203. );
  204. }
  205. //>>includeEnd('debug');
  206. var byWidth = regularGridAndSkirtAndEdgeIndicesCache[width];
  207. if (!when.defined(byWidth)) {
  208. regularGridAndSkirtAndEdgeIndicesCache[width] = byWidth = [];
  209. }
  210. var indicesAndEdges = byWidth[height];
  211. if (!when.defined(indicesAndEdges)) {
  212. var gridVertexCount = width * height;
  213. var gridIndexCount = (width - 1) * (height - 1) * 6;
  214. var edgeVertexCount = width * 2 + height * 2;
  215. var edgeIndexCount = Math.max(0, edgeVertexCount - 4) * 6;
  216. var vertexCount = gridVertexCount + edgeVertexCount;
  217. var indexCount = gridIndexCount + edgeIndexCount;
  218. var edgeIndices = getEdgeIndices(width, height);
  219. var westIndicesSouthToNorth = edgeIndices.westIndicesSouthToNorth;
  220. var southIndicesEastToWest = edgeIndices.southIndicesEastToWest;
  221. var eastIndicesNorthToSouth = edgeIndices.eastIndicesNorthToSouth;
  222. var northIndicesWestToEast = edgeIndices.northIndicesWestToEast;
  223. var indices = IndexDatatype.IndexDatatype.createTypedArray(vertexCount, indexCount);
  224. addRegularGridIndices(width, height, indices, 0);
  225. TerrainProvider.addSkirtIndices(
  226. westIndicesSouthToNorth,
  227. southIndicesEastToWest,
  228. eastIndicesNorthToSouth,
  229. northIndicesWestToEast,
  230. gridVertexCount,
  231. indices,
  232. gridIndexCount
  233. );
  234. indicesAndEdges = byWidth[height] = {
  235. indices: indices,
  236. westIndicesSouthToNorth: westIndicesSouthToNorth,
  237. southIndicesEastToWest: southIndicesEastToWest,
  238. eastIndicesNorthToSouth: eastIndicesNorthToSouth,
  239. northIndicesWestToEast: northIndicesWestToEast,
  240. indexCountWithoutSkirts: gridIndexCount,
  241. };
  242. }
  243. return indicesAndEdges;
  244. };
  245. /**
  246. * @private
  247. */
  248. TerrainProvider.addSkirtIndices = function (
  249. westIndicesSouthToNorth,
  250. southIndicesEastToWest,
  251. eastIndicesNorthToSouth,
  252. northIndicesWestToEast,
  253. vertexCount,
  254. indices,
  255. offset
  256. ) {
  257. var vertexIndex = vertexCount;
  258. offset = addSkirtIndices(
  259. westIndicesSouthToNorth,
  260. vertexIndex,
  261. indices,
  262. offset
  263. );
  264. vertexIndex += westIndicesSouthToNorth.length;
  265. offset = addSkirtIndices(
  266. southIndicesEastToWest,
  267. vertexIndex,
  268. indices,
  269. offset
  270. );
  271. vertexIndex += southIndicesEastToWest.length;
  272. offset = addSkirtIndices(
  273. eastIndicesNorthToSouth,
  274. vertexIndex,
  275. indices,
  276. offset
  277. );
  278. vertexIndex += eastIndicesNorthToSouth.length;
  279. addSkirtIndices(northIndicesWestToEast, vertexIndex, indices, offset);
  280. };
  281. function getEdgeIndices(width, height) {
  282. var westIndicesSouthToNorth = new Array(height);
  283. var southIndicesEastToWest = new Array(width);
  284. var eastIndicesNorthToSouth = new Array(height);
  285. var northIndicesWestToEast = new Array(width);
  286. var i;
  287. for (i = 0; i < width; ++i) {
  288. northIndicesWestToEast[i] = i;
  289. southIndicesEastToWest[i] = width * height - 1 - i;
  290. }
  291. for (i = 0; i < height; ++i) {
  292. eastIndicesNorthToSouth[i] = (i + 1) * width - 1;
  293. westIndicesSouthToNorth[i] = (height - i - 1) * width;
  294. }
  295. return {
  296. westIndicesSouthToNorth: westIndicesSouthToNorth,
  297. southIndicesEastToWest: southIndicesEastToWest,
  298. eastIndicesNorthToSouth: eastIndicesNorthToSouth,
  299. northIndicesWestToEast: northIndicesWestToEast,
  300. };
  301. }
  302. function addRegularGridIndices(width, height, indices, offset) {
  303. var index = 0;
  304. for (var j = 0; j < height - 1; ++j) {
  305. for (var i = 0; i < width - 1; ++i) {
  306. var upperLeft = index;
  307. var lowerLeft = upperLeft + width;
  308. var lowerRight = lowerLeft + 1;
  309. var upperRight = upperLeft + 1;
  310. indices[offset++] = upperLeft;
  311. indices[offset++] = lowerLeft;
  312. indices[offset++] = upperRight;
  313. indices[offset++] = upperRight;
  314. indices[offset++] = lowerLeft;
  315. indices[offset++] = lowerRight;
  316. ++index;
  317. }
  318. ++index;
  319. }
  320. }
  321. function addSkirtIndices(edgeIndices, vertexIndex, indices, offset) {
  322. var previousIndex = edgeIndices[0];
  323. var length = edgeIndices.length;
  324. for (var i = 1; i < length; ++i) {
  325. var index = edgeIndices[i];
  326. indices[offset++] = previousIndex;
  327. indices[offset++] = index;
  328. indices[offset++] = vertexIndex;
  329. indices[offset++] = vertexIndex;
  330. indices[offset++] = index;
  331. indices[offset++] = vertexIndex + 1;
  332. previousIndex = index;
  333. ++vertexIndex;
  334. }
  335. return offset;
  336. }
  337. /**
  338. * Specifies the quality of terrain created from heightmaps. A value of 1.0 will
  339. * ensure that adjacent heightmap vertices are separated by no more than
  340. * {@link Globe.maximumScreenSpaceError} screen pixels and will probably go very slowly.
  341. * A value of 0.5 will cut the estimated level zero geometric error in half, allowing twice the
  342. * screen pixels between adjacent heightmap vertices and thus rendering more quickly.
  343. * @type {Number}
  344. */
  345. TerrainProvider.heightmapTerrainQuality = 0.25;
  346. /**
  347. * Determines an appropriate geometric error estimate when the geometry comes from a heightmap.
  348. *
  349. * @param {Ellipsoid} ellipsoid The ellipsoid to which the terrain is attached.
  350. * @param {Number} tileImageWidth The width, in pixels, of the heightmap associated with a single tile.
  351. * @param {Number} numberOfTilesAtLevelZero The number of tiles in the horizontal direction at tile level zero.
  352. * @returns {Number} An estimated geometric error.
  353. */
  354. TerrainProvider.getEstimatedLevelZeroGeometricErrorForAHeightmap = function (
  355. ellipsoid,
  356. tileImageWidth,
  357. numberOfTilesAtLevelZero
  358. ) {
  359. return (
  360. (ellipsoid.maximumRadius *
  361. 2 *
  362. Math.PI *
  363. TerrainProvider.heightmapTerrainQuality) /
  364. (tileImageWidth * numberOfTilesAtLevelZero)
  365. );
  366. };
  367. /**
  368. * Requests the geometry for a given tile. This function should not be called before
  369. * {@link TerrainProvider#ready} returns true. The result must include terrain data and
  370. * may optionally include a water mask and an indication of which child tiles are available.
  371. * @function
  372. *
  373. * @param {Number} x The X coordinate of the tile for which to request geometry.
  374. * @param {Number} y The Y coordinate of the tile for which to request geometry.
  375. * @param {Number} level The level of the tile for which to request geometry.
  376. * @param {Request} [request] The request object. Intended for internal use only.
  377. *
  378. * @returns {Promise.<TerrainData>|undefined} A promise for the requested geometry. If this method
  379. * returns undefined instead of a promise, it is an indication that too many requests are already
  380. * pending and the request will be retried later.
  381. */
  382. TerrainProvider.prototype.requestTileGeometry =
  383. Check.DeveloperError.throwInstantiationError;
  384. /**
  385. * Gets the maximum geometric error allowed in a tile at a given level. This function should not be
  386. * called before {@link TerrainProvider#ready} returns true.
  387. * @function
  388. *
  389. * @param {Number} level The tile level for which to get the maximum geometric error.
  390. * @returns {Number} The maximum geometric error.
  391. */
  392. TerrainProvider.prototype.getLevelMaximumGeometricError =
  393. Check.DeveloperError.throwInstantiationError;
  394. /**
  395. * Determines whether data for a tile is available to be loaded.
  396. * @function
  397. *
  398. * @param {Number} x The X coordinate of the tile for which to request geometry.
  399. * @param {Number} y The Y coordinate of the tile for which to request geometry.
  400. * @param {Number} level The level of the tile for which to request geometry.
  401. * @returns {Boolean} Undefined if not supported by the terrain provider, otherwise true or false.
  402. */
  403. TerrainProvider.prototype.getTileDataAvailable =
  404. Check.DeveloperError.throwInstantiationError;
  405. /**
  406. * Makes sure we load availability data for a tile
  407. * @function
  408. *
  409. * @param {Number} x The X coordinate of the tile for which to request geometry.
  410. * @param {Number} y The Y coordinate of the tile for which to request geometry.
  411. * @param {Number} level The level of the tile for which to request geometry.
  412. * @returns {undefined|Promise<void>} Undefined if nothing need to be loaded or a Promise that resolves when all required tiles are loaded
  413. */
  414. TerrainProvider.prototype.loadTileDataAvailability =
  415. Check.DeveloperError.throwInstantiationError;
  416. var maxShort = 32767;
  417. var cartesian3Scratch = new Cartesian2.Cartesian3();
  418. var scratchMinimum = new Cartesian2.Cartesian3();
  419. var scratchMaximum = new Cartesian2.Cartesian3();
  420. var cartographicScratch = new Cartesian2.Cartographic();
  421. var toPack = new Cartesian2.Cartesian2();
  422. var scratchNormal = new Cartesian2.Cartesian3();
  423. var scratchToENU = new Transforms.Matrix4();
  424. var scratchFromENU = new Transforms.Matrix4();
  425. function createVerticesFromQuantizedTerrainMesh(
  426. parameters,
  427. transferableObjects
  428. ) {
  429. var quantizedVertices = parameters.quantizedVertices;
  430. var quantizedVertexCount = quantizedVertices.length / 3;
  431. var octEncodedNormals = parameters.octEncodedNormals;
  432. var edgeVertexCount =
  433. parameters.westIndices.length +
  434. parameters.eastIndices.length +
  435. parameters.southIndices.length +
  436. parameters.northIndices.length;
  437. var includeWebMercatorT = parameters.includeWebMercatorT;
  438. var rectangle = Cartesian2.Rectangle.clone(parameters.rectangle);
  439. var west = rectangle.west;
  440. var south = rectangle.south;
  441. var east = rectangle.east;
  442. var north = rectangle.north;
  443. var ellipsoid = Cartesian2.Ellipsoid.clone(parameters.ellipsoid);
  444. var exaggeration = parameters.exaggeration;
  445. var minimumHeight = parameters.minimumHeight * exaggeration;
  446. var maximumHeight = parameters.maximumHeight * exaggeration;
  447. var center = parameters.relativeToCenter;
  448. var fromENU = Transforms.Transforms.eastNorthUpToFixedFrame(center, ellipsoid);
  449. var toENU = Transforms.Matrix4.inverseTransformation(fromENU, new Transforms.Matrix4());
  450. var southMercatorY;
  451. var oneOverMercatorHeight;
  452. if (includeWebMercatorT) {
  453. southMercatorY = WebMercatorProjection.WebMercatorProjection.geodeticLatitudeToMercatorAngle(
  454. south
  455. );
  456. oneOverMercatorHeight =
  457. 1.0 /
  458. (WebMercatorProjection.WebMercatorProjection.geodeticLatitudeToMercatorAngle(north) -
  459. southMercatorY);
  460. }
  461. var uBuffer = quantizedVertices.subarray(0, quantizedVertexCount);
  462. var vBuffer = quantizedVertices.subarray(
  463. quantizedVertexCount,
  464. 2 * quantizedVertexCount
  465. );
  466. var heightBuffer = quantizedVertices.subarray(
  467. quantizedVertexCount * 2,
  468. 3 * quantizedVertexCount
  469. );
  470. var hasVertexNormals = when.defined(octEncodedNormals);
  471. var uvs = new Array(quantizedVertexCount);
  472. var heights = new Array(quantizedVertexCount);
  473. var positions = new Array(quantizedVertexCount);
  474. var webMercatorTs = includeWebMercatorT
  475. ? new Array(quantizedVertexCount)
  476. : [];
  477. var minimum = scratchMinimum;
  478. minimum.x = Number.POSITIVE_INFINITY;
  479. minimum.y = Number.POSITIVE_INFINITY;
  480. minimum.z = Number.POSITIVE_INFINITY;
  481. var maximum = scratchMaximum;
  482. maximum.x = Number.NEGATIVE_INFINITY;
  483. maximum.y = Number.NEGATIVE_INFINITY;
  484. maximum.z = Number.NEGATIVE_INFINITY;
  485. var minLongitude = Number.POSITIVE_INFINITY;
  486. var maxLongitude = Number.NEGATIVE_INFINITY;
  487. var minLatitude = Number.POSITIVE_INFINITY;
  488. var maxLatitude = Number.NEGATIVE_INFINITY;
  489. for (var i = 0; i < quantizedVertexCount; ++i) {
  490. var rawU = uBuffer[i];
  491. var rawV = vBuffer[i];
  492. var u = rawU / maxShort;
  493. var v = rawV / maxShort;
  494. var height = _Math.CesiumMath.lerp(
  495. minimumHeight,
  496. maximumHeight,
  497. heightBuffer[i] / maxShort
  498. );
  499. cartographicScratch.longitude = _Math.CesiumMath.lerp(west, east, u);
  500. cartographicScratch.latitude = _Math.CesiumMath.lerp(south, north, v);
  501. cartographicScratch.height = height;
  502. minLongitude = Math.min(cartographicScratch.longitude, minLongitude);
  503. maxLongitude = Math.max(cartographicScratch.longitude, maxLongitude);
  504. minLatitude = Math.min(cartographicScratch.latitude, minLatitude);
  505. maxLatitude = Math.max(cartographicScratch.latitude, maxLatitude);
  506. var position = ellipsoid.cartographicToCartesian(cartographicScratch);
  507. uvs[i] = new Cartesian2.Cartesian2(u, v);
  508. heights[i] = height;
  509. positions[i] = position;
  510. if (includeWebMercatorT) {
  511. webMercatorTs[i] =
  512. (WebMercatorProjection.WebMercatorProjection.geodeticLatitudeToMercatorAngle(
  513. cartographicScratch.latitude
  514. ) -
  515. southMercatorY) *
  516. oneOverMercatorHeight;
  517. }
  518. Transforms.Matrix4.multiplyByPoint(toENU, position, cartesian3Scratch);
  519. Cartesian2.Cartesian3.minimumByComponent(cartesian3Scratch, minimum, minimum);
  520. Cartesian2.Cartesian3.maximumByComponent(cartesian3Scratch, maximum, maximum);
  521. }
  522. var westIndicesSouthToNorth = copyAndSort(parameters.westIndices, function (
  523. a,
  524. b
  525. ) {
  526. return uvs[a].y - uvs[b].y;
  527. });
  528. var eastIndicesNorthToSouth = copyAndSort(parameters.eastIndices, function (
  529. a,
  530. b
  531. ) {
  532. return uvs[b].y - uvs[a].y;
  533. });
  534. var southIndicesEastToWest = copyAndSort(parameters.southIndices, function (
  535. a,
  536. b
  537. ) {
  538. return uvs[b].x - uvs[a].x;
  539. });
  540. var northIndicesWestToEast = copyAndSort(parameters.northIndices, function (
  541. a,
  542. b
  543. ) {
  544. return uvs[a].x - uvs[b].x;
  545. });
  546. var orientedBoundingBox;
  547. var boundingSphere;
  548. if (exaggeration !== 1.0) {
  549. // Bounding volumes need to be recomputed since the tile payload assumes no exaggeration.
  550. boundingSphere = Transforms.BoundingSphere.fromPoints(positions);
  551. orientedBoundingBox = OrientedBoundingBox.OrientedBoundingBox.fromRectangle(
  552. rectangle,
  553. minimumHeight,
  554. maximumHeight,
  555. ellipsoid
  556. );
  557. }
  558. var occludeePointInScaledSpace;
  559. if (exaggeration !== 1.0 || minimumHeight < 0.0) {
  560. // Horizon culling point needs to be recomputed since the tile payload assumes no exaggeration.
  561. var occluder = new TerrainEncoding.EllipsoidalOccluder(ellipsoid);
  562. occludeePointInScaledSpace = occluder.computeHorizonCullingPointPossiblyUnderEllipsoid(
  563. center,
  564. positions,
  565. minimumHeight
  566. );
  567. }
  568. var hMin = minimumHeight;
  569. hMin = Math.min(
  570. hMin,
  571. findMinMaxSkirts(
  572. parameters.westIndices,
  573. parameters.westSkirtHeight,
  574. heights,
  575. uvs,
  576. rectangle,
  577. ellipsoid,
  578. toENU,
  579. minimum,
  580. maximum
  581. )
  582. );
  583. hMin = Math.min(
  584. hMin,
  585. findMinMaxSkirts(
  586. parameters.southIndices,
  587. parameters.southSkirtHeight,
  588. heights,
  589. uvs,
  590. rectangle,
  591. ellipsoid,
  592. toENU,
  593. minimum,
  594. maximum
  595. )
  596. );
  597. hMin = Math.min(
  598. hMin,
  599. findMinMaxSkirts(
  600. parameters.eastIndices,
  601. parameters.eastSkirtHeight,
  602. heights,
  603. uvs,
  604. rectangle,
  605. ellipsoid,
  606. toENU,
  607. minimum,
  608. maximum
  609. )
  610. );
  611. hMin = Math.min(
  612. hMin,
  613. findMinMaxSkirts(
  614. parameters.northIndices,
  615. parameters.northSkirtHeight,
  616. heights,
  617. uvs,
  618. rectangle,
  619. ellipsoid,
  620. toENU,
  621. minimum,
  622. maximum
  623. )
  624. );
  625. var aaBox = new EllipsoidTangentPlane.AxisAlignedBoundingBox(minimum, maximum, center);
  626. var encoding = new TerrainEncoding.TerrainEncoding(
  627. aaBox,
  628. hMin,
  629. maximumHeight,
  630. fromENU,
  631. hasVertexNormals,
  632. includeWebMercatorT
  633. );
  634. var vertexStride = encoding.getStride();
  635. var size =
  636. quantizedVertexCount * vertexStride + edgeVertexCount * vertexStride;
  637. var vertexBuffer = new Float32Array(size);
  638. var bufferIndex = 0;
  639. for (var j = 0; j < quantizedVertexCount; ++j) {
  640. if (hasVertexNormals) {
  641. var n = j * 2.0;
  642. toPack.x = octEncodedNormals[n];
  643. toPack.y = octEncodedNormals[n + 1];
  644. if (exaggeration !== 1.0) {
  645. var normal = AttributeCompression.AttributeCompression.octDecode(
  646. toPack.x,
  647. toPack.y,
  648. scratchNormal
  649. );
  650. var fromENUNormal = Transforms.Transforms.eastNorthUpToFixedFrame(
  651. positions[j],
  652. ellipsoid,
  653. scratchFromENU
  654. );
  655. var toENUNormal = Transforms.Matrix4.inverseTransformation(
  656. fromENUNormal,
  657. scratchToENU
  658. );
  659. Transforms.Matrix4.multiplyByPointAsVector(toENUNormal, normal, normal);
  660. normal.z *= exaggeration;
  661. Cartesian2.Cartesian3.normalize(normal, normal);
  662. Transforms.Matrix4.multiplyByPointAsVector(fromENUNormal, normal, normal);
  663. Cartesian2.Cartesian3.normalize(normal, normal);
  664. AttributeCompression.AttributeCompression.octEncode(normal, toPack);
  665. }
  666. }
  667. bufferIndex = encoding.encode(
  668. vertexBuffer,
  669. bufferIndex,
  670. positions[j],
  671. uvs[j],
  672. heights[j],
  673. toPack,
  674. webMercatorTs[j]
  675. );
  676. }
  677. var edgeTriangleCount = Math.max(0, (edgeVertexCount - 4) * 2);
  678. var indexBufferLength = parameters.indices.length + edgeTriangleCount * 3;
  679. var indexBuffer = IndexDatatype.IndexDatatype.createTypedArray(
  680. quantizedVertexCount + edgeVertexCount,
  681. indexBufferLength
  682. );
  683. indexBuffer.set(parameters.indices, 0);
  684. var percentage = 0.0001;
  685. var lonOffset = (maxLongitude - minLongitude) * percentage;
  686. var latOffset = (maxLatitude - minLatitude) * percentage;
  687. var westLongitudeOffset = -lonOffset;
  688. var westLatitudeOffset = 0.0;
  689. var eastLongitudeOffset = lonOffset;
  690. var eastLatitudeOffset = 0.0;
  691. var northLongitudeOffset = 0.0;
  692. var northLatitudeOffset = latOffset;
  693. var southLongitudeOffset = 0.0;
  694. var southLatitudeOffset = -latOffset;
  695. // Add skirts.
  696. var vertexBufferIndex = quantizedVertexCount * vertexStride;
  697. addSkirt(
  698. vertexBuffer,
  699. vertexBufferIndex,
  700. westIndicesSouthToNorth,
  701. encoding,
  702. heights,
  703. uvs,
  704. octEncodedNormals,
  705. ellipsoid,
  706. rectangle,
  707. parameters.westSkirtHeight,
  708. exaggeration,
  709. southMercatorY,
  710. oneOverMercatorHeight,
  711. westLongitudeOffset,
  712. westLatitudeOffset
  713. );
  714. vertexBufferIndex += parameters.westIndices.length * vertexStride;
  715. addSkirt(
  716. vertexBuffer,
  717. vertexBufferIndex,
  718. southIndicesEastToWest,
  719. encoding,
  720. heights,
  721. uvs,
  722. octEncodedNormals,
  723. ellipsoid,
  724. rectangle,
  725. parameters.southSkirtHeight,
  726. exaggeration,
  727. southMercatorY,
  728. oneOverMercatorHeight,
  729. southLongitudeOffset,
  730. southLatitudeOffset
  731. );
  732. vertexBufferIndex += parameters.southIndices.length * vertexStride;
  733. addSkirt(
  734. vertexBuffer,
  735. vertexBufferIndex,
  736. eastIndicesNorthToSouth,
  737. encoding,
  738. heights,
  739. uvs,
  740. octEncodedNormals,
  741. ellipsoid,
  742. rectangle,
  743. parameters.eastSkirtHeight,
  744. exaggeration,
  745. southMercatorY,
  746. oneOverMercatorHeight,
  747. eastLongitudeOffset,
  748. eastLatitudeOffset
  749. );
  750. vertexBufferIndex += parameters.eastIndices.length * vertexStride;
  751. addSkirt(
  752. vertexBuffer,
  753. vertexBufferIndex,
  754. northIndicesWestToEast,
  755. encoding,
  756. heights,
  757. uvs,
  758. octEncodedNormals,
  759. ellipsoid,
  760. rectangle,
  761. parameters.northSkirtHeight,
  762. exaggeration,
  763. southMercatorY,
  764. oneOverMercatorHeight,
  765. northLongitudeOffset,
  766. northLatitudeOffset
  767. );
  768. TerrainProvider.addSkirtIndices(
  769. westIndicesSouthToNorth,
  770. southIndicesEastToWest,
  771. eastIndicesNorthToSouth,
  772. northIndicesWestToEast,
  773. quantizedVertexCount,
  774. indexBuffer,
  775. parameters.indices.length
  776. );
  777. transferableObjects.push(vertexBuffer.buffer, indexBuffer.buffer);
  778. return {
  779. vertices: vertexBuffer.buffer,
  780. indices: indexBuffer.buffer,
  781. westIndicesSouthToNorth: westIndicesSouthToNorth,
  782. southIndicesEastToWest: southIndicesEastToWest,
  783. eastIndicesNorthToSouth: eastIndicesNorthToSouth,
  784. northIndicesWestToEast: northIndicesWestToEast,
  785. vertexStride: vertexStride,
  786. center: center,
  787. minimumHeight: minimumHeight,
  788. maximumHeight: maximumHeight,
  789. boundingSphere: boundingSphere,
  790. orientedBoundingBox: orientedBoundingBox,
  791. occludeePointInScaledSpace: occludeePointInScaledSpace,
  792. encoding: encoding,
  793. indexCountWithoutSkirts: parameters.indices.length,
  794. };
  795. }
  796. function findMinMaxSkirts(
  797. edgeIndices,
  798. edgeHeight,
  799. heights,
  800. uvs,
  801. rectangle,
  802. ellipsoid,
  803. toENU,
  804. minimum,
  805. maximum
  806. ) {
  807. var hMin = Number.POSITIVE_INFINITY;
  808. var north = rectangle.north;
  809. var south = rectangle.south;
  810. var east = rectangle.east;
  811. var west = rectangle.west;
  812. if (east < west) {
  813. east += _Math.CesiumMath.TWO_PI;
  814. }
  815. var length = edgeIndices.length;
  816. for (var i = 0; i < length; ++i) {
  817. var index = edgeIndices[i];
  818. var h = heights[index];
  819. var uv = uvs[index];
  820. cartographicScratch.longitude = _Math.CesiumMath.lerp(west, east, uv.x);
  821. cartographicScratch.latitude = _Math.CesiumMath.lerp(south, north, uv.y);
  822. cartographicScratch.height = h - edgeHeight;
  823. var position = ellipsoid.cartographicToCartesian(
  824. cartographicScratch,
  825. cartesian3Scratch
  826. );
  827. Transforms.Matrix4.multiplyByPoint(toENU, position, position);
  828. Cartesian2.Cartesian3.minimumByComponent(position, minimum, minimum);
  829. Cartesian2.Cartesian3.maximumByComponent(position, maximum, maximum);
  830. hMin = Math.min(hMin, cartographicScratch.height);
  831. }
  832. return hMin;
  833. }
  834. function addSkirt(
  835. vertexBuffer,
  836. vertexBufferIndex,
  837. edgeVertices,
  838. encoding,
  839. heights,
  840. uvs,
  841. octEncodedNormals,
  842. ellipsoid,
  843. rectangle,
  844. skirtLength,
  845. exaggeration,
  846. southMercatorY,
  847. oneOverMercatorHeight,
  848. longitudeOffset,
  849. latitudeOffset
  850. ) {
  851. var hasVertexNormals = when.defined(octEncodedNormals);
  852. var north = rectangle.north;
  853. var south = rectangle.south;
  854. var east = rectangle.east;
  855. var west = rectangle.west;
  856. if (east < west) {
  857. east += _Math.CesiumMath.TWO_PI;
  858. }
  859. var length = edgeVertices.length;
  860. for (var i = 0; i < length; ++i) {
  861. var index = edgeVertices[i];
  862. var h = heights[index];
  863. var uv = uvs[index];
  864. cartographicScratch.longitude =
  865. _Math.CesiumMath.lerp(west, east, uv.x) + longitudeOffset;
  866. cartographicScratch.latitude =
  867. _Math.CesiumMath.lerp(south, north, uv.y) + latitudeOffset;
  868. cartographicScratch.height = h - skirtLength;
  869. var position = ellipsoid.cartographicToCartesian(
  870. cartographicScratch,
  871. cartesian3Scratch
  872. );
  873. if (hasVertexNormals) {
  874. var n = index * 2.0;
  875. toPack.x = octEncodedNormals[n];
  876. toPack.y = octEncodedNormals[n + 1];
  877. if (exaggeration !== 1.0) {
  878. var normal = AttributeCompression.AttributeCompression.octDecode(
  879. toPack.x,
  880. toPack.y,
  881. scratchNormal
  882. );
  883. var fromENUNormal = Transforms.Transforms.eastNorthUpToFixedFrame(
  884. cartesian3Scratch,
  885. ellipsoid,
  886. scratchFromENU
  887. );
  888. var toENUNormal = Transforms.Matrix4.inverseTransformation(
  889. fromENUNormal,
  890. scratchToENU
  891. );
  892. Transforms.Matrix4.multiplyByPointAsVector(toENUNormal, normal, normal);
  893. normal.z *= exaggeration;
  894. Cartesian2.Cartesian3.normalize(normal, normal);
  895. Transforms.Matrix4.multiplyByPointAsVector(fromENUNormal, normal, normal);
  896. Cartesian2.Cartesian3.normalize(normal, normal);
  897. AttributeCompression.AttributeCompression.octEncode(normal, toPack);
  898. }
  899. }
  900. var webMercatorT;
  901. if (encoding.hasWebMercatorT) {
  902. webMercatorT =
  903. (WebMercatorProjection.WebMercatorProjection.geodeticLatitudeToMercatorAngle(
  904. cartographicScratch.latitude
  905. ) -
  906. southMercatorY) *
  907. oneOverMercatorHeight;
  908. }
  909. vertexBufferIndex = encoding.encode(
  910. vertexBuffer,
  911. vertexBufferIndex,
  912. position,
  913. uv,
  914. cartographicScratch.height,
  915. toPack,
  916. webMercatorT
  917. );
  918. }
  919. }
  920. function copyAndSort(typedArray, comparator) {
  921. var copy;
  922. if (typeof typedArray.slice === "function") {
  923. copy = typedArray.slice();
  924. if (typeof copy.sort !== "function") {
  925. // Sliced typed array isn't sortable, so we can't use it.
  926. copy = undefined;
  927. }
  928. }
  929. if (!when.defined(copy)) {
  930. copy = Array.prototype.slice.call(typedArray);
  931. }
  932. copy.sort(comparator);
  933. return copy;
  934. }
  935. var createVerticesFromQuantizedTerrainMesh$1 = createTaskProcessorWorker(
  936. createVerticesFromQuantizedTerrainMesh
  937. );
  938. return createVerticesFromQuantizedTerrainMesh$1;
  939. });
  940. //# sourceMappingURL=createVerticesFromQuantizedTerrainMesh.js.map