upsampleQuantizedTerrainMesh.js 31 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047
  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', './createTaskProcessorWorker', './EllipsoidTangentPlane-ce6e380f', './OrientedBoundingBox-8897f474', './TerrainEncoding-6954276f'], function (when, Check, _Math, Cartesian2, Transforms, RuntimeError, WebGLConstants, ComponentDatatype, AttributeCompression, IndexDatatype, IntersectionTests, Plane, createTaskProcessorWorker, EllipsoidTangentPlane, OrientedBoundingBox, TerrainEncoding) { 'use strict';
  24. /**
  25. * Contains functions for operating on 2D triangles.
  26. *
  27. * @namespace Intersections2D
  28. */
  29. var Intersections2D = {};
  30. /**
  31. * Splits a 2D triangle at given axis-aligned threshold value and returns the resulting
  32. * polygon on a given side of the threshold. The resulting polygon may have 0, 1, 2,
  33. * 3, or 4 vertices.
  34. *
  35. * @param {Number} threshold The threshold coordinate value at which to clip the triangle.
  36. * @param {Boolean} keepAbove true to keep the portion of the triangle above the threshold, or false
  37. * to keep the portion below.
  38. * @param {Number} u0 The coordinate of the first vertex in the triangle, in counter-clockwise order.
  39. * @param {Number} u1 The coordinate of the second vertex in the triangle, in counter-clockwise order.
  40. * @param {Number} u2 The coordinate of the third vertex in the triangle, in counter-clockwise order.
  41. * @param {Number[]} [result] The array into which to copy the result. If this parameter is not supplied,
  42. * a new array is constructed and returned.
  43. * @returns {Number[]} The polygon that results after the clip, specified as a list of
  44. * vertices. The vertices are specified in counter-clockwise order.
  45. * Each vertex is either an index from the existing list (identified as
  46. * a 0, 1, or 2) or -1 indicating a new vertex not in the original triangle.
  47. * For new vertices, the -1 is followed by three additional numbers: the
  48. * index of each of the two original vertices forming the line segment that
  49. * the new vertex lies on, and the fraction of the distance from the first
  50. * vertex to the second one.
  51. *
  52. * @example
  53. * var result = Cesium.Intersections2D.clipTriangleAtAxisAlignedThreshold(0.5, false, 0.2, 0.6, 0.4);
  54. * // result === [2, 0, -1, 1, 0, 0.25, -1, 1, 2, 0.5]
  55. */
  56. Intersections2D.clipTriangleAtAxisAlignedThreshold = function (
  57. threshold,
  58. keepAbove,
  59. u0,
  60. u1,
  61. u2,
  62. result
  63. ) {
  64. //>>includeStart('debug', pragmas.debug);
  65. if (!when.defined(threshold)) {
  66. throw new Check.DeveloperError("threshold is required.");
  67. }
  68. if (!when.defined(keepAbove)) {
  69. throw new Check.DeveloperError("keepAbove is required.");
  70. }
  71. if (!when.defined(u0)) {
  72. throw new Check.DeveloperError("u0 is required.");
  73. }
  74. if (!when.defined(u1)) {
  75. throw new Check.DeveloperError("u1 is required.");
  76. }
  77. if (!when.defined(u2)) {
  78. throw new Check.DeveloperError("u2 is required.");
  79. }
  80. //>>includeEnd('debug');
  81. if (!when.defined(result)) {
  82. result = [];
  83. } else {
  84. result.length = 0;
  85. }
  86. var u0Behind;
  87. var u1Behind;
  88. var u2Behind;
  89. if (keepAbove) {
  90. u0Behind = u0 < threshold;
  91. u1Behind = u1 < threshold;
  92. u2Behind = u2 < threshold;
  93. } else {
  94. u0Behind = u0 > threshold;
  95. u1Behind = u1 > threshold;
  96. u2Behind = u2 > threshold;
  97. }
  98. var numBehind = u0Behind + u1Behind + u2Behind;
  99. var u01Ratio;
  100. var u02Ratio;
  101. var u12Ratio;
  102. var u10Ratio;
  103. var u20Ratio;
  104. var u21Ratio;
  105. if (numBehind === 1) {
  106. if (u0Behind) {
  107. u01Ratio = (threshold - u0) / (u1 - u0);
  108. u02Ratio = (threshold - u0) / (u2 - u0);
  109. result.push(1);
  110. result.push(2);
  111. if (u02Ratio !== 1.0) {
  112. result.push(-1);
  113. result.push(0);
  114. result.push(2);
  115. result.push(u02Ratio);
  116. }
  117. if (u01Ratio !== 1.0) {
  118. result.push(-1);
  119. result.push(0);
  120. result.push(1);
  121. result.push(u01Ratio);
  122. }
  123. } else if (u1Behind) {
  124. u12Ratio = (threshold - u1) / (u2 - u1);
  125. u10Ratio = (threshold - u1) / (u0 - u1);
  126. result.push(2);
  127. result.push(0);
  128. if (u10Ratio !== 1.0) {
  129. result.push(-1);
  130. result.push(1);
  131. result.push(0);
  132. result.push(u10Ratio);
  133. }
  134. if (u12Ratio !== 1.0) {
  135. result.push(-1);
  136. result.push(1);
  137. result.push(2);
  138. result.push(u12Ratio);
  139. }
  140. } else if (u2Behind) {
  141. u20Ratio = (threshold - u2) / (u0 - u2);
  142. u21Ratio = (threshold - u2) / (u1 - u2);
  143. result.push(0);
  144. result.push(1);
  145. if (u21Ratio !== 1.0) {
  146. result.push(-1);
  147. result.push(2);
  148. result.push(1);
  149. result.push(u21Ratio);
  150. }
  151. if (u20Ratio !== 1.0) {
  152. result.push(-1);
  153. result.push(2);
  154. result.push(0);
  155. result.push(u20Ratio);
  156. }
  157. }
  158. } else if (numBehind === 2) {
  159. if (!u0Behind && u0 !== threshold) {
  160. u10Ratio = (threshold - u1) / (u0 - u1);
  161. u20Ratio = (threshold - u2) / (u0 - u2);
  162. result.push(0);
  163. result.push(-1);
  164. result.push(1);
  165. result.push(0);
  166. result.push(u10Ratio);
  167. result.push(-1);
  168. result.push(2);
  169. result.push(0);
  170. result.push(u20Ratio);
  171. } else if (!u1Behind && u1 !== threshold) {
  172. u21Ratio = (threshold - u2) / (u1 - u2);
  173. u01Ratio = (threshold - u0) / (u1 - u0);
  174. result.push(1);
  175. result.push(-1);
  176. result.push(2);
  177. result.push(1);
  178. result.push(u21Ratio);
  179. result.push(-1);
  180. result.push(0);
  181. result.push(1);
  182. result.push(u01Ratio);
  183. } else if (!u2Behind && u2 !== threshold) {
  184. u02Ratio = (threshold - u0) / (u2 - u0);
  185. u12Ratio = (threshold - u1) / (u2 - u1);
  186. result.push(2);
  187. result.push(-1);
  188. result.push(0);
  189. result.push(2);
  190. result.push(u02Ratio);
  191. result.push(-1);
  192. result.push(1);
  193. result.push(2);
  194. result.push(u12Ratio);
  195. }
  196. } else if (numBehind !== 3) {
  197. // Completely in front of threshold
  198. result.push(0);
  199. result.push(1);
  200. result.push(2);
  201. }
  202. // else Completely behind threshold
  203. return result;
  204. };
  205. /**
  206. * Compute the barycentric coordinates of a 2D position within a 2D triangle.
  207. *
  208. * @param {Number} x The x coordinate of the position for which to find the barycentric coordinates.
  209. * @param {Number} y The y coordinate of the position for which to find the barycentric coordinates.
  210. * @param {Number} x1 The x coordinate of the triangle's first vertex.
  211. * @param {Number} y1 The y coordinate of the triangle's first vertex.
  212. * @param {Number} x2 The x coordinate of the triangle's second vertex.
  213. * @param {Number} y2 The y coordinate of the triangle's second vertex.
  214. * @param {Number} x3 The x coordinate of the triangle's third vertex.
  215. * @param {Number} y3 The y coordinate of the triangle's third vertex.
  216. * @param {Cartesian3} [result] The instance into to which to copy the result. If this parameter
  217. * is undefined, a new instance is created and returned.
  218. * @returns {Cartesian3} The barycentric coordinates of the position within the triangle.
  219. *
  220. * @example
  221. * var result = Cesium.Intersections2D.computeBarycentricCoordinates(0.0, 0.0, 0.0, 1.0, -1, -0.5, 1, -0.5);
  222. * // result === new Cesium.Cartesian3(1.0 / 3.0, 1.0 / 3.0, 1.0 / 3.0);
  223. */
  224. Intersections2D.computeBarycentricCoordinates = function (
  225. x,
  226. y,
  227. x1,
  228. y1,
  229. x2,
  230. y2,
  231. x3,
  232. y3,
  233. result
  234. ) {
  235. //>>includeStart('debug', pragmas.debug);
  236. if (!when.defined(x)) {
  237. throw new Check.DeveloperError("x is required.");
  238. }
  239. if (!when.defined(y)) {
  240. throw new Check.DeveloperError("y is required.");
  241. }
  242. if (!when.defined(x1)) {
  243. throw new Check.DeveloperError("x1 is required.");
  244. }
  245. if (!when.defined(y1)) {
  246. throw new Check.DeveloperError("y1 is required.");
  247. }
  248. if (!when.defined(x2)) {
  249. throw new Check.DeveloperError("x2 is required.");
  250. }
  251. if (!when.defined(y2)) {
  252. throw new Check.DeveloperError("y2 is required.");
  253. }
  254. if (!when.defined(x3)) {
  255. throw new Check.DeveloperError("x3 is required.");
  256. }
  257. if (!when.defined(y3)) {
  258. throw new Check.DeveloperError("y3 is required.");
  259. }
  260. //>>includeEnd('debug');
  261. var x1mx3 = x1 - x3;
  262. var x3mx2 = x3 - x2;
  263. var y2my3 = y2 - y3;
  264. var y1my3 = y1 - y3;
  265. var inverseDeterminant = 1.0 / (y2my3 * x1mx3 + x3mx2 * y1my3);
  266. var ymy3 = y - y3;
  267. var xmx3 = x - x3;
  268. var l1 = (y2my3 * xmx3 + x3mx2 * ymy3) * inverseDeterminant;
  269. var l2 = (-y1my3 * xmx3 + x1mx3 * ymy3) * inverseDeterminant;
  270. var l3 = 1.0 - l1 - l2;
  271. if (when.defined(result)) {
  272. result.x = l1;
  273. result.y = l2;
  274. result.z = l3;
  275. return result;
  276. }
  277. return new Cartesian2.Cartesian3(l1, l2, l3);
  278. };
  279. /**
  280. * Compute the intersection between 2 line segments
  281. *
  282. * @param {Number} x00 The x coordinate of the first line's first vertex.
  283. * @param {Number} y00 The y coordinate of the first line's first vertex.
  284. * @param {Number} x01 The x coordinate of the first line's second vertex.
  285. * @param {Number} y01 The y coordinate of the first line's second vertex.
  286. * @param {Number} x10 The x coordinate of the second line's first vertex.
  287. * @param {Number} y10 The y coordinate of the second line's first vertex.
  288. * @param {Number} x11 The x coordinate of the second line's second vertex.
  289. * @param {Number} y11 The y coordinate of the second line's second vertex.
  290. * @param {Cartesian2} [result] The instance into to which to copy the result. If this parameter
  291. * is undefined, a new instance is created and returned.
  292. * @returns {Cartesian2} The intersection point, undefined if there is no intersection point or lines are coincident.
  293. *
  294. * @example
  295. * var result = Cesium.Intersections2D.computeLineSegmentLineSegmentIntersection(0.0, 0.0, 0.0, 2.0, -1, 1, 1, 1);
  296. * // result === new Cesium.Cartesian2(0.0, 1.0);
  297. */
  298. Intersections2D.computeLineSegmentLineSegmentIntersection = function (
  299. x00,
  300. y00,
  301. x01,
  302. y01,
  303. x10,
  304. y10,
  305. x11,
  306. y11,
  307. result
  308. ) {
  309. //>>includeStart('debug', pragmas.debug);
  310. Check.Check.typeOf.number("x00", x00);
  311. Check.Check.typeOf.number("y00", y00);
  312. Check.Check.typeOf.number("x01", x01);
  313. Check.Check.typeOf.number("y01", y01);
  314. Check.Check.typeOf.number("x10", x10);
  315. Check.Check.typeOf.number("y10", y10);
  316. Check.Check.typeOf.number("x11", x11);
  317. Check.Check.typeOf.number("y11", y11);
  318. //>>includeEnd('debug');
  319. var numerator1A = (x11 - x10) * (y00 - y10) - (y11 - y10) * (x00 - x10);
  320. var numerator1B = (x01 - x00) * (y00 - y10) - (y01 - y00) * (x00 - x10);
  321. var denominator1 = (y11 - y10) * (x01 - x00) - (x11 - x10) * (y01 - y00);
  322. // If denominator = 0, then lines are parallel. If denominator = 0 and both numerators are 0, then coincident
  323. if (denominator1 === 0) {
  324. return;
  325. }
  326. var ua1 = numerator1A / denominator1;
  327. var ub1 = numerator1B / denominator1;
  328. if (ua1 >= 0 && ua1 <= 1 && ub1 >= 0 && ub1 <= 1) {
  329. if (!when.defined(result)) {
  330. result = new Cartesian2.Cartesian2();
  331. }
  332. result.x = x00 + ua1 * (x01 - x00);
  333. result.y = y00 + ua1 * (y01 - y00);
  334. return result;
  335. }
  336. };
  337. var maxShort = 32767;
  338. var halfMaxShort = (maxShort / 2) | 0;
  339. var clipScratch = [];
  340. var clipScratch2 = [];
  341. var verticesScratch = [];
  342. var cartographicScratch = new Cartesian2.Cartographic();
  343. var cartesian3Scratch = new Cartesian2.Cartesian3();
  344. var uScratch = [];
  345. var vScratch = [];
  346. var heightScratch = [];
  347. var indicesScratch = [];
  348. var normalsScratch = [];
  349. var horizonOcclusionPointScratch = new Cartesian2.Cartesian3();
  350. var boundingSphereScratch = new Transforms.BoundingSphere();
  351. var orientedBoundingBoxScratch = new OrientedBoundingBox.OrientedBoundingBox();
  352. var decodeTexCoordsScratch = new Cartesian2.Cartesian2();
  353. var octEncodedNormalScratch = new Cartesian2.Cartesian3();
  354. function upsampleQuantizedTerrainMesh(parameters, transferableObjects) {
  355. var isEastChild = parameters.isEastChild;
  356. var isNorthChild = parameters.isNorthChild;
  357. var minU = isEastChild ? halfMaxShort : 0;
  358. var maxU = isEastChild ? maxShort : halfMaxShort;
  359. var minV = isNorthChild ? halfMaxShort : 0;
  360. var maxV = isNorthChild ? maxShort : halfMaxShort;
  361. var uBuffer = uScratch;
  362. var vBuffer = vScratch;
  363. var heightBuffer = heightScratch;
  364. var normalBuffer = normalsScratch;
  365. uBuffer.length = 0;
  366. vBuffer.length = 0;
  367. heightBuffer.length = 0;
  368. normalBuffer.length = 0;
  369. var indices = indicesScratch;
  370. indices.length = 0;
  371. var vertexMap = {};
  372. var parentVertices = parameters.vertices;
  373. var parentIndices = parameters.indices;
  374. parentIndices = parentIndices.subarray(0, parameters.indexCountWithoutSkirts);
  375. var encoding = TerrainEncoding.TerrainEncoding.clone(parameters.encoding);
  376. var hasVertexNormals = encoding.hasVertexNormals;
  377. var exaggeration = parameters.exaggeration;
  378. var vertexCount = 0;
  379. var quantizedVertexCount = parameters.vertexCountWithoutSkirts;
  380. var parentMinimumHeight = parameters.minimumHeight;
  381. var parentMaximumHeight = parameters.maximumHeight;
  382. var parentUBuffer = new Array(quantizedVertexCount);
  383. var parentVBuffer = new Array(quantizedVertexCount);
  384. var parentHeightBuffer = new Array(quantizedVertexCount);
  385. var parentNormalBuffer = hasVertexNormals
  386. ? new Array(quantizedVertexCount * 2)
  387. : undefined;
  388. var threshold = 20;
  389. var height;
  390. var i, n;
  391. var u, v;
  392. for (i = 0, n = 0; i < quantizedVertexCount; ++i, n += 2) {
  393. var texCoords = encoding.decodeTextureCoordinates(
  394. parentVertices,
  395. i,
  396. decodeTexCoordsScratch
  397. );
  398. height = encoding.decodeHeight(parentVertices, i) / exaggeration;
  399. u = _Math.CesiumMath.clamp((texCoords.x * maxShort) | 0, 0, maxShort);
  400. v = _Math.CesiumMath.clamp((texCoords.y * maxShort) | 0, 0, maxShort);
  401. parentHeightBuffer[i] = _Math.CesiumMath.clamp(
  402. (((height - parentMinimumHeight) /
  403. (parentMaximumHeight - parentMinimumHeight)) *
  404. maxShort) |
  405. 0,
  406. 0,
  407. maxShort
  408. );
  409. if (u < threshold) {
  410. u = 0;
  411. }
  412. if (v < threshold) {
  413. v = 0;
  414. }
  415. if (maxShort - u < threshold) {
  416. u = maxShort;
  417. }
  418. if (maxShort - v < threshold) {
  419. v = maxShort;
  420. }
  421. parentUBuffer[i] = u;
  422. parentVBuffer[i] = v;
  423. if (hasVertexNormals) {
  424. var encodedNormal = encoding.getOctEncodedNormal(
  425. parentVertices,
  426. i,
  427. octEncodedNormalScratch
  428. );
  429. parentNormalBuffer[n] = encodedNormal.x;
  430. parentNormalBuffer[n + 1] = encodedNormal.y;
  431. }
  432. if (
  433. ((isEastChild && u >= halfMaxShort) ||
  434. (!isEastChild && u <= halfMaxShort)) &&
  435. ((isNorthChild && v >= halfMaxShort) ||
  436. (!isNorthChild && v <= halfMaxShort))
  437. ) {
  438. vertexMap[i] = vertexCount;
  439. uBuffer.push(u);
  440. vBuffer.push(v);
  441. heightBuffer.push(parentHeightBuffer[i]);
  442. if (hasVertexNormals) {
  443. normalBuffer.push(parentNormalBuffer[n]);
  444. normalBuffer.push(parentNormalBuffer[n + 1]);
  445. }
  446. ++vertexCount;
  447. }
  448. }
  449. var triangleVertices = [];
  450. triangleVertices.push(new Vertex());
  451. triangleVertices.push(new Vertex());
  452. triangleVertices.push(new Vertex());
  453. var clippedTriangleVertices = [];
  454. clippedTriangleVertices.push(new Vertex());
  455. clippedTriangleVertices.push(new Vertex());
  456. clippedTriangleVertices.push(new Vertex());
  457. var clippedIndex;
  458. var clipped2;
  459. for (i = 0; i < parentIndices.length; i += 3) {
  460. var i0 = parentIndices[i];
  461. var i1 = parentIndices[i + 1];
  462. var i2 = parentIndices[i + 2];
  463. var u0 = parentUBuffer[i0];
  464. var u1 = parentUBuffer[i1];
  465. var u2 = parentUBuffer[i2];
  466. triangleVertices[0].initializeIndexed(
  467. parentUBuffer,
  468. parentVBuffer,
  469. parentHeightBuffer,
  470. parentNormalBuffer,
  471. i0
  472. );
  473. triangleVertices[1].initializeIndexed(
  474. parentUBuffer,
  475. parentVBuffer,
  476. parentHeightBuffer,
  477. parentNormalBuffer,
  478. i1
  479. );
  480. triangleVertices[2].initializeIndexed(
  481. parentUBuffer,
  482. parentVBuffer,
  483. parentHeightBuffer,
  484. parentNormalBuffer,
  485. i2
  486. );
  487. // Clip triangle on the east-west boundary.
  488. var clipped = Intersections2D.clipTriangleAtAxisAlignedThreshold(
  489. halfMaxShort,
  490. isEastChild,
  491. u0,
  492. u1,
  493. u2,
  494. clipScratch
  495. );
  496. // Get the first clipped triangle, if any.
  497. clippedIndex = 0;
  498. if (clippedIndex >= clipped.length) {
  499. continue;
  500. }
  501. clippedIndex = clippedTriangleVertices[0].initializeFromClipResult(
  502. clipped,
  503. clippedIndex,
  504. triangleVertices
  505. );
  506. if (clippedIndex >= clipped.length) {
  507. continue;
  508. }
  509. clippedIndex = clippedTriangleVertices[1].initializeFromClipResult(
  510. clipped,
  511. clippedIndex,
  512. triangleVertices
  513. );
  514. if (clippedIndex >= clipped.length) {
  515. continue;
  516. }
  517. clippedIndex = clippedTriangleVertices[2].initializeFromClipResult(
  518. clipped,
  519. clippedIndex,
  520. triangleVertices
  521. );
  522. // Clip the triangle against the North-south boundary.
  523. clipped2 = Intersections2D.clipTriangleAtAxisAlignedThreshold(
  524. halfMaxShort,
  525. isNorthChild,
  526. clippedTriangleVertices[0].getV(),
  527. clippedTriangleVertices[1].getV(),
  528. clippedTriangleVertices[2].getV(),
  529. clipScratch2
  530. );
  531. addClippedPolygon(
  532. uBuffer,
  533. vBuffer,
  534. heightBuffer,
  535. normalBuffer,
  536. indices,
  537. vertexMap,
  538. clipped2,
  539. clippedTriangleVertices,
  540. hasVertexNormals
  541. );
  542. // If there's another vertex in the original clipped result,
  543. // it forms a second triangle. Clip it as well.
  544. if (clippedIndex < clipped.length) {
  545. clippedTriangleVertices[2].clone(clippedTriangleVertices[1]);
  546. clippedTriangleVertices[2].initializeFromClipResult(
  547. clipped,
  548. clippedIndex,
  549. triangleVertices
  550. );
  551. clipped2 = Intersections2D.clipTriangleAtAxisAlignedThreshold(
  552. halfMaxShort,
  553. isNorthChild,
  554. clippedTriangleVertices[0].getV(),
  555. clippedTriangleVertices[1].getV(),
  556. clippedTriangleVertices[2].getV(),
  557. clipScratch2
  558. );
  559. addClippedPolygon(
  560. uBuffer,
  561. vBuffer,
  562. heightBuffer,
  563. normalBuffer,
  564. indices,
  565. vertexMap,
  566. clipped2,
  567. clippedTriangleVertices,
  568. hasVertexNormals
  569. );
  570. }
  571. }
  572. var uOffset = isEastChild ? -maxShort : 0;
  573. var vOffset = isNorthChild ? -maxShort : 0;
  574. var westIndices = [];
  575. var southIndices = [];
  576. var eastIndices = [];
  577. var northIndices = [];
  578. var minimumHeight = Number.MAX_VALUE;
  579. var maximumHeight = -minimumHeight;
  580. var cartesianVertices = verticesScratch;
  581. cartesianVertices.length = 0;
  582. var ellipsoid = Cartesian2.Ellipsoid.clone(parameters.ellipsoid);
  583. var rectangle = Cartesian2.Rectangle.clone(parameters.childRectangle);
  584. var north = rectangle.north;
  585. var south = rectangle.south;
  586. var east = rectangle.east;
  587. var west = rectangle.west;
  588. if (east < west) {
  589. east += _Math.CesiumMath.TWO_PI;
  590. }
  591. for (i = 0; i < uBuffer.length; ++i) {
  592. u = Math.round(uBuffer[i]);
  593. if (u <= minU) {
  594. westIndices.push(i);
  595. u = 0;
  596. } else if (u >= maxU) {
  597. eastIndices.push(i);
  598. u = maxShort;
  599. } else {
  600. u = u * 2 + uOffset;
  601. }
  602. uBuffer[i] = u;
  603. v = Math.round(vBuffer[i]);
  604. if (v <= minV) {
  605. southIndices.push(i);
  606. v = 0;
  607. } else if (v >= maxV) {
  608. northIndices.push(i);
  609. v = maxShort;
  610. } else {
  611. v = v * 2 + vOffset;
  612. }
  613. vBuffer[i] = v;
  614. height = _Math.CesiumMath.lerp(
  615. parentMinimumHeight,
  616. parentMaximumHeight,
  617. heightBuffer[i] / maxShort
  618. );
  619. if (height < minimumHeight) {
  620. minimumHeight = height;
  621. }
  622. if (height > maximumHeight) {
  623. maximumHeight = height;
  624. }
  625. heightBuffer[i] = height;
  626. cartographicScratch.longitude = _Math.CesiumMath.lerp(west, east, u / maxShort);
  627. cartographicScratch.latitude = _Math.CesiumMath.lerp(south, north, v / maxShort);
  628. cartographicScratch.height = height;
  629. ellipsoid.cartographicToCartesian(cartographicScratch, cartesian3Scratch);
  630. cartesianVertices.push(cartesian3Scratch.x);
  631. cartesianVertices.push(cartesian3Scratch.y);
  632. cartesianVertices.push(cartesian3Scratch.z);
  633. }
  634. var boundingSphere = Transforms.BoundingSphere.fromVertices(
  635. cartesianVertices,
  636. Cartesian2.Cartesian3.ZERO,
  637. 3,
  638. boundingSphereScratch
  639. );
  640. var orientedBoundingBox = OrientedBoundingBox.OrientedBoundingBox.fromRectangle(
  641. rectangle,
  642. minimumHeight,
  643. maximumHeight,
  644. ellipsoid,
  645. orientedBoundingBoxScratch
  646. );
  647. var occluder = new TerrainEncoding.EllipsoidalOccluder(ellipsoid);
  648. var horizonOcclusionPoint = occluder.computeHorizonCullingPointFromVerticesPossiblyUnderEllipsoid(
  649. boundingSphere.center,
  650. cartesianVertices,
  651. 3,
  652. boundingSphere.center,
  653. minimumHeight,
  654. horizonOcclusionPointScratch
  655. );
  656. var heightRange = maximumHeight - minimumHeight;
  657. var vertices = new Uint16Array(
  658. uBuffer.length + vBuffer.length + heightBuffer.length
  659. );
  660. for (i = 0; i < uBuffer.length; ++i) {
  661. vertices[i] = uBuffer[i];
  662. }
  663. var start = uBuffer.length;
  664. for (i = 0; i < vBuffer.length; ++i) {
  665. vertices[start + i] = vBuffer[i];
  666. }
  667. start += vBuffer.length;
  668. for (i = 0; i < heightBuffer.length; ++i) {
  669. vertices[start + i] =
  670. (maxShort * (heightBuffer[i] - minimumHeight)) / heightRange;
  671. }
  672. var indicesTypedArray = IndexDatatype.IndexDatatype.createTypedArray(
  673. uBuffer.length,
  674. indices
  675. );
  676. var encodedNormals;
  677. if (hasVertexNormals) {
  678. var normalArray = new Uint8Array(normalBuffer);
  679. transferableObjects.push(
  680. vertices.buffer,
  681. indicesTypedArray.buffer,
  682. normalArray.buffer
  683. );
  684. encodedNormals = normalArray.buffer;
  685. } else {
  686. transferableObjects.push(vertices.buffer, indicesTypedArray.buffer);
  687. }
  688. return {
  689. vertices: vertices.buffer,
  690. encodedNormals: encodedNormals,
  691. indices: indicesTypedArray.buffer,
  692. minimumHeight: minimumHeight,
  693. maximumHeight: maximumHeight,
  694. westIndices: westIndices,
  695. southIndices: southIndices,
  696. eastIndices: eastIndices,
  697. northIndices: northIndices,
  698. boundingSphere: boundingSphere,
  699. orientedBoundingBox: orientedBoundingBox,
  700. horizonOcclusionPoint: horizonOcclusionPoint,
  701. };
  702. }
  703. function Vertex() {
  704. this.vertexBuffer = undefined;
  705. this.index = undefined;
  706. this.first = undefined;
  707. this.second = undefined;
  708. this.ratio = undefined;
  709. }
  710. Vertex.prototype.clone = function (result) {
  711. if (!when.defined(result)) {
  712. result = new Vertex();
  713. }
  714. result.uBuffer = this.uBuffer;
  715. result.vBuffer = this.vBuffer;
  716. result.heightBuffer = this.heightBuffer;
  717. result.normalBuffer = this.normalBuffer;
  718. result.index = this.index;
  719. result.first = this.first;
  720. result.second = this.second;
  721. result.ratio = this.ratio;
  722. return result;
  723. };
  724. Vertex.prototype.initializeIndexed = function (
  725. uBuffer,
  726. vBuffer,
  727. heightBuffer,
  728. normalBuffer,
  729. index
  730. ) {
  731. this.uBuffer = uBuffer;
  732. this.vBuffer = vBuffer;
  733. this.heightBuffer = heightBuffer;
  734. this.normalBuffer = normalBuffer;
  735. this.index = index;
  736. this.first = undefined;
  737. this.second = undefined;
  738. this.ratio = undefined;
  739. };
  740. Vertex.prototype.initializeFromClipResult = function (
  741. clipResult,
  742. index,
  743. vertices
  744. ) {
  745. var nextIndex = index + 1;
  746. if (clipResult[index] !== -1) {
  747. vertices[clipResult[index]].clone(this);
  748. } else {
  749. this.vertexBuffer = undefined;
  750. this.index = undefined;
  751. this.first = vertices[clipResult[nextIndex]];
  752. ++nextIndex;
  753. this.second = vertices[clipResult[nextIndex]];
  754. ++nextIndex;
  755. this.ratio = clipResult[nextIndex];
  756. ++nextIndex;
  757. }
  758. return nextIndex;
  759. };
  760. Vertex.prototype.getKey = function () {
  761. if (this.isIndexed()) {
  762. return this.index;
  763. }
  764. return JSON.stringify({
  765. first: this.first.getKey(),
  766. second: this.second.getKey(),
  767. ratio: this.ratio,
  768. });
  769. };
  770. Vertex.prototype.isIndexed = function () {
  771. return when.defined(this.index);
  772. };
  773. Vertex.prototype.getH = function () {
  774. if (when.defined(this.index)) {
  775. return this.heightBuffer[this.index];
  776. }
  777. return _Math.CesiumMath.lerp(this.first.getH(), this.second.getH(), this.ratio);
  778. };
  779. Vertex.prototype.getU = function () {
  780. if (when.defined(this.index)) {
  781. return this.uBuffer[this.index];
  782. }
  783. return _Math.CesiumMath.lerp(this.first.getU(), this.second.getU(), this.ratio);
  784. };
  785. Vertex.prototype.getV = function () {
  786. if (when.defined(this.index)) {
  787. return this.vBuffer[this.index];
  788. }
  789. return _Math.CesiumMath.lerp(this.first.getV(), this.second.getV(), this.ratio);
  790. };
  791. var encodedScratch = new Cartesian2.Cartesian2();
  792. // An upsampled triangle may be clipped twice before it is assigned an index
  793. // In this case, we need a buffer to handle the recursion of getNormalX() and getNormalY().
  794. var depth = -1;
  795. var cartesianScratch1 = [new Cartesian2.Cartesian3(), new Cartesian2.Cartesian3()];
  796. var cartesianScratch2 = [new Cartesian2.Cartesian3(), new Cartesian2.Cartesian3()];
  797. function lerpOctEncodedNormal(vertex, result) {
  798. ++depth;
  799. var first = cartesianScratch1[depth];
  800. var second = cartesianScratch2[depth];
  801. first = AttributeCompression.AttributeCompression.octDecode(
  802. vertex.first.getNormalX(),
  803. vertex.first.getNormalY(),
  804. first
  805. );
  806. second = AttributeCompression.AttributeCompression.octDecode(
  807. vertex.second.getNormalX(),
  808. vertex.second.getNormalY(),
  809. second
  810. );
  811. cartesian3Scratch = Cartesian2.Cartesian3.lerp(
  812. first,
  813. second,
  814. vertex.ratio,
  815. cartesian3Scratch
  816. );
  817. Cartesian2.Cartesian3.normalize(cartesian3Scratch, cartesian3Scratch);
  818. AttributeCompression.AttributeCompression.octEncode(cartesian3Scratch, result);
  819. --depth;
  820. return result;
  821. }
  822. Vertex.prototype.getNormalX = function () {
  823. if (when.defined(this.index)) {
  824. return this.normalBuffer[this.index * 2];
  825. }
  826. encodedScratch = lerpOctEncodedNormal(this, encodedScratch);
  827. return encodedScratch.x;
  828. };
  829. Vertex.prototype.getNormalY = function () {
  830. if (when.defined(this.index)) {
  831. return this.normalBuffer[this.index * 2 + 1];
  832. }
  833. encodedScratch = lerpOctEncodedNormal(this, encodedScratch);
  834. return encodedScratch.y;
  835. };
  836. var polygonVertices = [];
  837. polygonVertices.push(new Vertex());
  838. polygonVertices.push(new Vertex());
  839. polygonVertices.push(new Vertex());
  840. polygonVertices.push(new Vertex());
  841. function addClippedPolygon(
  842. uBuffer,
  843. vBuffer,
  844. heightBuffer,
  845. normalBuffer,
  846. indices,
  847. vertexMap,
  848. clipped,
  849. triangleVertices,
  850. hasVertexNormals
  851. ) {
  852. if (clipped.length === 0) {
  853. return;
  854. }
  855. var numVertices = 0;
  856. var clippedIndex = 0;
  857. while (clippedIndex < clipped.length) {
  858. clippedIndex = polygonVertices[numVertices++].initializeFromClipResult(
  859. clipped,
  860. clippedIndex,
  861. triangleVertices
  862. );
  863. }
  864. for (var i = 0; i < numVertices; ++i) {
  865. var polygonVertex = polygonVertices[i];
  866. if (!polygonVertex.isIndexed()) {
  867. var key = polygonVertex.getKey();
  868. if (when.defined(vertexMap[key])) {
  869. polygonVertex.newIndex = vertexMap[key];
  870. } else {
  871. var newIndex = uBuffer.length;
  872. uBuffer.push(polygonVertex.getU());
  873. vBuffer.push(polygonVertex.getV());
  874. heightBuffer.push(polygonVertex.getH());
  875. if (hasVertexNormals) {
  876. normalBuffer.push(polygonVertex.getNormalX());
  877. normalBuffer.push(polygonVertex.getNormalY());
  878. }
  879. polygonVertex.newIndex = newIndex;
  880. vertexMap[key] = newIndex;
  881. }
  882. } else {
  883. polygonVertex.newIndex = vertexMap[polygonVertex.index];
  884. polygonVertex.uBuffer = uBuffer;
  885. polygonVertex.vBuffer = vBuffer;
  886. polygonVertex.heightBuffer = heightBuffer;
  887. if (hasVertexNormals) {
  888. polygonVertex.normalBuffer = normalBuffer;
  889. }
  890. }
  891. }
  892. if (numVertices === 3) {
  893. // A triangle.
  894. indices.push(polygonVertices[0].newIndex);
  895. indices.push(polygonVertices[1].newIndex);
  896. indices.push(polygonVertices[2].newIndex);
  897. } else if (numVertices === 4) {
  898. // A quad - two triangles.
  899. indices.push(polygonVertices[0].newIndex);
  900. indices.push(polygonVertices[1].newIndex);
  901. indices.push(polygonVertices[2].newIndex);
  902. indices.push(polygonVertices[0].newIndex);
  903. indices.push(polygonVertices[2].newIndex);
  904. indices.push(polygonVertices[3].newIndex);
  905. }
  906. }
  907. var upsampleQuantizedTerrainMesh$1 = createTaskProcessorWorker(upsampleQuantizedTerrainMesh);
  908. return upsampleQuantizedTerrainMesh$1;
  909. });
  910. //# sourceMappingURL=upsampleQuantizedTerrainMesh.js.map