createGroundPolylineGeometry.js 74 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151
  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', './GeometryAttribute-e9a8b203', './EncodedCartesian3-6c97231d', './IntersectionTests-afc38163', './Plane-fa30fc46', './WebMercatorProjection-054890ef', './arrayRemoveDuplicates-0263f42c', './ArcType-dc1c5aee', './EllipsoidRhumbLine-5f1492e5', './EllipsoidGeodesic-0f41968b'], function (when, Check, _Math, Cartesian2, Transforms, RuntimeError, WebGLConstants, ComponentDatatype, GeometryAttribute, EncodedCartesian3, IntersectionTests, Plane, WebMercatorProjection, arrayRemoveDuplicates, ArcType, EllipsoidRhumbLine, EllipsoidGeodesic) { 'use strict';
  24. /**
  25. * A tiling scheme for geometry referenced to a simple {@link GeographicProjection} where
  26. * longitude and latitude are directly mapped to X and Y. This projection is commonly
  27. * known as geographic, equirectangular, equidistant cylindrical, or plate carrée.
  28. *
  29. * @alias GeographicTilingScheme
  30. * @constructor
  31. *
  32. * @param {Object} [options] Object with the following properties:
  33. * @param {Ellipsoid} [options.ellipsoid=Ellipsoid.WGS84] The ellipsoid whose surface is being tiled. Defaults to
  34. * the WGS84 ellipsoid.
  35. * @param {Rectangle} [options.rectangle=Rectangle.MAX_VALUE] The rectangle, in radians, covered by the tiling scheme.
  36. * @param {Number} [options.numberOfLevelZeroTilesX=2] The number of tiles in the X direction at level zero of
  37. * the tile tree.
  38. * @param {Number} [options.numberOfLevelZeroTilesY=1] The number of tiles in the Y direction at level zero of
  39. * the tile tree.
  40. */
  41. function GeographicTilingScheme(options) {
  42. options = when.defaultValue(options, when.defaultValue.EMPTY_OBJECT);
  43. this._ellipsoid = when.defaultValue(options.ellipsoid, Cartesian2.Ellipsoid.WGS84);
  44. this._rectangle = when.defaultValue(options.rectangle, Cartesian2.Rectangle.MAX_VALUE);
  45. this._projection = new Transforms.GeographicProjection(this._ellipsoid);
  46. this._numberOfLevelZeroTilesX = when.defaultValue(
  47. options.numberOfLevelZeroTilesX,
  48. 2
  49. );
  50. this._numberOfLevelZeroTilesY = when.defaultValue(
  51. options.numberOfLevelZeroTilesY,
  52. 1
  53. );
  54. }
  55. Object.defineProperties(GeographicTilingScheme.prototype, {
  56. /**
  57. * Gets the ellipsoid that is tiled by this tiling scheme.
  58. * @memberof GeographicTilingScheme.prototype
  59. * @type {Ellipsoid}
  60. */
  61. ellipsoid: {
  62. get: function () {
  63. return this._ellipsoid;
  64. },
  65. },
  66. /**
  67. * Gets the rectangle, in radians, covered by this tiling scheme.
  68. * @memberof GeographicTilingScheme.prototype
  69. * @type {Rectangle}
  70. */
  71. rectangle: {
  72. get: function () {
  73. return this._rectangle;
  74. },
  75. },
  76. /**
  77. * Gets the map projection used by this tiling scheme.
  78. * @memberof GeographicTilingScheme.prototype
  79. * @type {MapProjection}
  80. */
  81. projection: {
  82. get: function () {
  83. return this._projection;
  84. },
  85. },
  86. });
  87. /**
  88. * Gets the total number of tiles in the X direction at a specified level-of-detail.
  89. *
  90. * @param {Number} level The level-of-detail.
  91. * @returns {Number} The number of tiles in the X direction at the given level.
  92. */
  93. GeographicTilingScheme.prototype.getNumberOfXTilesAtLevel = function (level) {
  94. return this._numberOfLevelZeroTilesX << level;
  95. };
  96. /**
  97. * Gets the total number of tiles in the Y direction at a specified level-of-detail.
  98. *
  99. * @param {Number} level The level-of-detail.
  100. * @returns {Number} The number of tiles in the Y direction at the given level.
  101. */
  102. GeographicTilingScheme.prototype.getNumberOfYTilesAtLevel = function (level) {
  103. return this._numberOfLevelZeroTilesY << level;
  104. };
  105. /**
  106. * Transforms a rectangle specified in geodetic radians to the native coordinate system
  107. * of this tiling scheme.
  108. *
  109. * @param {Rectangle} rectangle The rectangle to transform.
  110. * @param {Rectangle} [result] The instance to which to copy the result, or undefined if a new instance
  111. * should be created.
  112. * @returns {Rectangle} The specified 'result', or a new object containing the native rectangle if 'result'
  113. * is undefined.
  114. */
  115. GeographicTilingScheme.prototype.rectangleToNativeRectangle = function (
  116. rectangle,
  117. result
  118. ) {
  119. //>>includeStart('debug', pragmas.debug);
  120. Check.Check.defined("rectangle", rectangle);
  121. //>>includeEnd('debug');
  122. var west = _Math.CesiumMath.toDegrees(rectangle.west);
  123. var south = _Math.CesiumMath.toDegrees(rectangle.south);
  124. var east = _Math.CesiumMath.toDegrees(rectangle.east);
  125. var north = _Math.CesiumMath.toDegrees(rectangle.north);
  126. if (!when.defined(result)) {
  127. return new Cartesian2.Rectangle(west, south, east, north);
  128. }
  129. result.west = west;
  130. result.south = south;
  131. result.east = east;
  132. result.north = north;
  133. return result;
  134. };
  135. /**
  136. * Converts tile x, y coordinates and level to a rectangle expressed in the native coordinates
  137. * of the tiling scheme.
  138. *
  139. * @param {Number} x The integer x coordinate of the tile.
  140. * @param {Number} y The integer y coordinate of the tile.
  141. * @param {Number} level The tile level-of-detail. Zero is the least detailed.
  142. * @param {Object} [result] The instance to which to copy the result, or undefined if a new instance
  143. * should be created.
  144. * @returns {Rectangle} The specified 'result', or a new object containing the rectangle
  145. * if 'result' is undefined.
  146. */
  147. GeographicTilingScheme.prototype.tileXYToNativeRectangle = function (
  148. x,
  149. y,
  150. level,
  151. result
  152. ) {
  153. var rectangleRadians = this.tileXYToRectangle(x, y, level, result);
  154. rectangleRadians.west = _Math.CesiumMath.toDegrees(rectangleRadians.west);
  155. rectangleRadians.south = _Math.CesiumMath.toDegrees(rectangleRadians.south);
  156. rectangleRadians.east = _Math.CesiumMath.toDegrees(rectangleRadians.east);
  157. rectangleRadians.north = _Math.CesiumMath.toDegrees(rectangleRadians.north);
  158. return rectangleRadians;
  159. };
  160. /**
  161. * Converts tile x, y coordinates and level to a cartographic rectangle in radians.
  162. *
  163. * @param {Number} x The integer x coordinate of the tile.
  164. * @param {Number} y The integer y coordinate of the tile.
  165. * @param {Number} level The tile level-of-detail. Zero is the least detailed.
  166. * @param {Object} [result] The instance to which to copy the result, or undefined if a new instance
  167. * should be created.
  168. * @returns {Rectangle} The specified 'result', or a new object containing the rectangle
  169. * if 'result' is undefined.
  170. */
  171. GeographicTilingScheme.prototype.tileXYToRectangle = function (
  172. x,
  173. y,
  174. level,
  175. result
  176. ) {
  177. var rectangle = this._rectangle;
  178. var xTiles = this.getNumberOfXTilesAtLevel(level);
  179. var yTiles = this.getNumberOfYTilesAtLevel(level);
  180. var xTileWidth = rectangle.width / xTiles;
  181. var west = x * xTileWidth + rectangle.west;
  182. var east = (x + 1) * xTileWidth + rectangle.west;
  183. var yTileHeight = rectangle.height / yTiles;
  184. var north = rectangle.north - y * yTileHeight;
  185. var south = rectangle.north - (y + 1) * yTileHeight;
  186. if (!when.defined(result)) {
  187. result = new Cartesian2.Rectangle(west, south, east, north);
  188. }
  189. result.west = west;
  190. result.south = south;
  191. result.east = east;
  192. result.north = north;
  193. return result;
  194. };
  195. /**
  196. * Calculates the tile x, y coordinates of the tile containing
  197. * a given cartographic position.
  198. *
  199. * @param {Cartographic} position The position.
  200. * @param {Number} level The tile level-of-detail. Zero is the least detailed.
  201. * @param {Cartesian2} [result] The instance to which to copy the result, or undefined if a new instance
  202. * should be created.
  203. * @returns {Cartesian2} The specified 'result', or a new object containing the tile x, y coordinates
  204. * if 'result' is undefined.
  205. */
  206. GeographicTilingScheme.prototype.positionToTileXY = function (
  207. position,
  208. level,
  209. result
  210. ) {
  211. var rectangle = this._rectangle;
  212. if (!Cartesian2.Rectangle.contains(rectangle, position)) {
  213. // outside the bounds of the tiling scheme
  214. return undefined;
  215. }
  216. var xTiles = this.getNumberOfXTilesAtLevel(level);
  217. var yTiles = this.getNumberOfYTilesAtLevel(level);
  218. var xTileWidth = rectangle.width / xTiles;
  219. var yTileHeight = rectangle.height / yTiles;
  220. var longitude = position.longitude;
  221. if (rectangle.east < rectangle.west) {
  222. longitude += _Math.CesiumMath.TWO_PI;
  223. }
  224. var xTileCoordinate = ((longitude - rectangle.west) / xTileWidth) | 0;
  225. if (xTileCoordinate >= xTiles) {
  226. xTileCoordinate = xTiles - 1;
  227. }
  228. var yTileCoordinate =
  229. ((rectangle.north - position.latitude) / yTileHeight) | 0;
  230. if (yTileCoordinate >= yTiles) {
  231. yTileCoordinate = yTiles - 1;
  232. }
  233. if (!when.defined(result)) {
  234. return new Cartesian2.Cartesian2(xTileCoordinate, yTileCoordinate);
  235. }
  236. result.x = xTileCoordinate;
  237. result.y = yTileCoordinate;
  238. return result;
  239. };
  240. var scratchDiagonalCartesianNE = new Cartesian2.Cartesian3();
  241. var scratchDiagonalCartesianSW = new Cartesian2.Cartesian3();
  242. var scratchDiagonalCartographic = new Cartesian2.Cartographic();
  243. var scratchCenterCartesian = new Cartesian2.Cartesian3();
  244. var scratchSurfaceCartesian = new Cartesian2.Cartesian3();
  245. var scratchBoundingSphere = new Transforms.BoundingSphere();
  246. var tilingScheme = new GeographicTilingScheme();
  247. var scratchCorners = [
  248. new Cartesian2.Cartographic(),
  249. new Cartesian2.Cartographic(),
  250. new Cartesian2.Cartographic(),
  251. new Cartesian2.Cartographic(),
  252. ];
  253. var scratchTileXY = new Cartesian2.Cartesian2();
  254. /**
  255. * A collection of functions for approximating terrain height
  256. * @private
  257. */
  258. var ApproximateTerrainHeights = {};
  259. /**
  260. * Initializes the minimum and maximum terrain heights
  261. * @return {Promise<void>}
  262. */
  263. ApproximateTerrainHeights.initialize = function () {
  264. var initPromise = ApproximateTerrainHeights._initPromise;
  265. if (when.defined(initPromise)) {
  266. return initPromise;
  267. }
  268. initPromise = Transforms.Resource.fetchJson(
  269. Transforms.buildModuleUrl("Assets/approximateTerrainHeights.json")
  270. ).then(function (json) {
  271. ApproximateTerrainHeights._terrainHeights = json;
  272. });
  273. ApproximateTerrainHeights._initPromise = initPromise;
  274. return initPromise;
  275. };
  276. /**
  277. * Computes the minimum and maximum terrain heights for a given rectangle
  278. * @param {Rectangle} rectangle The bounding rectangle
  279. * @param {Ellipsoid} [ellipsoid=Ellipsoid.WGS84] The ellipsoid
  280. * @return {{minimumTerrainHeight: Number, maximumTerrainHeight: Number}}
  281. */
  282. ApproximateTerrainHeights.getMinimumMaximumHeights = function (
  283. rectangle,
  284. ellipsoid
  285. ) {
  286. //>>includeStart('debug', pragmas.debug);
  287. Check.Check.defined("rectangle", rectangle);
  288. if (!when.defined(ApproximateTerrainHeights._terrainHeights)) {
  289. throw new Check.DeveloperError(
  290. "You must call ApproximateTerrainHeights.initialize and wait for the promise to resolve before using this function"
  291. );
  292. }
  293. //>>includeEnd('debug');
  294. ellipsoid = when.defaultValue(ellipsoid, Cartesian2.Ellipsoid.WGS84);
  295. var xyLevel = getTileXYLevel(rectangle);
  296. // Get the terrain min/max for that tile
  297. var minTerrainHeight = ApproximateTerrainHeights._defaultMinTerrainHeight;
  298. var maxTerrainHeight = ApproximateTerrainHeights._defaultMaxTerrainHeight;
  299. if (when.defined(xyLevel)) {
  300. var key = xyLevel.level + "-" + xyLevel.x + "-" + xyLevel.y;
  301. var heights = ApproximateTerrainHeights._terrainHeights[key];
  302. if (when.defined(heights)) {
  303. minTerrainHeight = heights[0];
  304. maxTerrainHeight = heights[1];
  305. }
  306. // Compute min by taking the center of the NE->SW diagonal and finding distance to the surface
  307. ellipsoid.cartographicToCartesian(
  308. Cartesian2.Rectangle.northeast(rectangle, scratchDiagonalCartographic),
  309. scratchDiagonalCartesianNE
  310. );
  311. ellipsoid.cartographicToCartesian(
  312. Cartesian2.Rectangle.southwest(rectangle, scratchDiagonalCartographic),
  313. scratchDiagonalCartesianSW
  314. );
  315. Cartesian2.Cartesian3.midpoint(
  316. scratchDiagonalCartesianSW,
  317. scratchDiagonalCartesianNE,
  318. scratchCenterCartesian
  319. );
  320. var surfacePosition = ellipsoid.scaleToGeodeticSurface(
  321. scratchCenterCartesian,
  322. scratchSurfaceCartesian
  323. );
  324. if (when.defined(surfacePosition)) {
  325. var distance = Cartesian2.Cartesian3.distance(
  326. scratchCenterCartesian,
  327. surfacePosition
  328. );
  329. minTerrainHeight = Math.min(minTerrainHeight, -distance);
  330. } else {
  331. minTerrainHeight = ApproximateTerrainHeights._defaultMinTerrainHeight;
  332. }
  333. }
  334. minTerrainHeight = Math.max(
  335. ApproximateTerrainHeights._defaultMinTerrainHeight,
  336. minTerrainHeight
  337. );
  338. return {
  339. minimumTerrainHeight: minTerrainHeight,
  340. maximumTerrainHeight: maxTerrainHeight,
  341. };
  342. };
  343. /**
  344. * Computes the bounding sphere based on the tile heights in the rectangle
  345. * @param {Rectangle} rectangle The bounding rectangle
  346. * @param {Ellipsoid} [ellipsoid=Ellipsoid.WGS84] The ellipsoid
  347. * @return {BoundingSphere} The result bounding sphere
  348. */
  349. ApproximateTerrainHeights.getBoundingSphere = function (rectangle, ellipsoid) {
  350. //>>includeStart('debug', pragmas.debug);
  351. Check.Check.defined("rectangle", rectangle);
  352. if (!when.defined(ApproximateTerrainHeights._terrainHeights)) {
  353. throw new Check.DeveloperError(
  354. "You must call ApproximateTerrainHeights.initialize and wait for the promise to resolve before using this function"
  355. );
  356. }
  357. //>>includeEnd('debug');
  358. ellipsoid = when.defaultValue(ellipsoid, Cartesian2.Ellipsoid.WGS84);
  359. var xyLevel = getTileXYLevel(rectangle);
  360. // Get the terrain max for that tile
  361. var maxTerrainHeight = ApproximateTerrainHeights._defaultMaxTerrainHeight;
  362. if (when.defined(xyLevel)) {
  363. var key = xyLevel.level + "-" + xyLevel.x + "-" + xyLevel.y;
  364. var heights = ApproximateTerrainHeights._terrainHeights[key];
  365. if (when.defined(heights)) {
  366. maxTerrainHeight = heights[1];
  367. }
  368. }
  369. var result = Transforms.BoundingSphere.fromRectangle3D(rectangle, ellipsoid, 0.0);
  370. Transforms.BoundingSphere.fromRectangle3D(
  371. rectangle,
  372. ellipsoid,
  373. maxTerrainHeight,
  374. scratchBoundingSphere
  375. );
  376. return Transforms.BoundingSphere.union(result, scratchBoundingSphere, result);
  377. };
  378. function getTileXYLevel(rectangle) {
  379. Cartesian2.Cartographic.fromRadians(
  380. rectangle.east,
  381. rectangle.north,
  382. 0.0,
  383. scratchCorners[0]
  384. );
  385. Cartesian2.Cartographic.fromRadians(
  386. rectangle.west,
  387. rectangle.north,
  388. 0.0,
  389. scratchCorners[1]
  390. );
  391. Cartesian2.Cartographic.fromRadians(
  392. rectangle.east,
  393. rectangle.south,
  394. 0.0,
  395. scratchCorners[2]
  396. );
  397. Cartesian2.Cartographic.fromRadians(
  398. rectangle.west,
  399. rectangle.south,
  400. 0.0,
  401. scratchCorners[3]
  402. );
  403. // Determine which tile the bounding rectangle is in
  404. var lastLevelX = 0,
  405. lastLevelY = 0;
  406. var currentX = 0,
  407. currentY = 0;
  408. var maxLevel = ApproximateTerrainHeights._terrainHeightsMaxLevel;
  409. var i;
  410. for (i = 0; i <= maxLevel; ++i) {
  411. var failed = false;
  412. for (var j = 0; j < 4; ++j) {
  413. var corner = scratchCorners[j];
  414. tilingScheme.positionToTileXY(corner, i, scratchTileXY);
  415. if (j === 0) {
  416. currentX = scratchTileXY.x;
  417. currentY = scratchTileXY.y;
  418. } else if (currentX !== scratchTileXY.x || currentY !== scratchTileXY.y) {
  419. failed = true;
  420. break;
  421. }
  422. }
  423. if (failed) {
  424. break;
  425. }
  426. lastLevelX = currentX;
  427. lastLevelY = currentY;
  428. }
  429. if (i === 0) {
  430. return undefined;
  431. }
  432. return {
  433. x: lastLevelX,
  434. y: lastLevelY,
  435. level: i > maxLevel ? maxLevel : i - 1,
  436. };
  437. }
  438. ApproximateTerrainHeights._terrainHeightsMaxLevel = 6;
  439. ApproximateTerrainHeights._defaultMaxTerrainHeight = 9000.0;
  440. ApproximateTerrainHeights._defaultMinTerrainHeight = -100000.0;
  441. ApproximateTerrainHeights._terrainHeights = undefined;
  442. ApproximateTerrainHeights._initPromise = undefined;
  443. Object.defineProperties(ApproximateTerrainHeights, {
  444. /**
  445. * Determines if the terrain heights are initialized and ready to use. To initialize the terrain heights,
  446. * call {@link ApproximateTerrainHeights#initialize} and wait for the returned promise to resolve.
  447. * @type {Boolean}
  448. * @readonly
  449. * @memberof ApproximateTerrainHeights
  450. */
  451. initialized: {
  452. get: function () {
  453. return when.defined(ApproximateTerrainHeights._terrainHeights);
  454. },
  455. },
  456. });
  457. var PROJECTIONS = [Transforms.GeographicProjection, WebMercatorProjection.WebMercatorProjection];
  458. var PROJECTION_COUNT = PROJECTIONS.length;
  459. var MITER_BREAK_SMALL = Math.cos(_Math.CesiumMath.toRadians(30.0));
  460. var MITER_BREAK_LARGE = Math.cos(_Math.CesiumMath.toRadians(150.0));
  461. // Initial heights for constructing the wall.
  462. // Keeping WALL_INITIAL_MIN_HEIGHT near the ellipsoid surface helps
  463. // prevent precision problems with planes in the shader.
  464. // Putting the start point of a plane at ApproximateTerrainHeights._defaultMinTerrainHeight,
  465. // which is a highly conservative bound, usually puts the plane origin several thousands
  466. // of meters away from the actual terrain, causing floating point problems when checking
  467. // fragments on terrain against the plane.
  468. // Ellipsoid height is generally much closer.
  469. // The initial max height is arbitrary.
  470. // Both heights are corrected using ApproximateTerrainHeights for computing the actual volume geometry.
  471. var WALL_INITIAL_MIN_HEIGHT = 0.0;
  472. var WALL_INITIAL_MAX_HEIGHT = 1000.0;
  473. /**
  474. * A description of a polyline on terrain or 3D Tiles. Only to be used with {@link GroundPolylinePrimitive}.
  475. *
  476. * @alias GroundPolylineGeometry
  477. * @constructor
  478. *
  479. * @param {Object} options Options with the following properties:
  480. * @param {Cartesian3[]} options.positions An array of {@link Cartesian3} defining the polyline's points. Heights above the ellipsoid will be ignored.
  481. * @param {Number} [options.width=1.0] The screen space width in pixels.
  482. * @param {Number} [options.granularity=9999.0] The distance interval in meters used for interpolating options.points. Defaults to 9999.0 meters. Zero indicates no interpolation.
  483. * @param {Boolean} [options.loop=false] Whether during geometry creation a line segment will be added between the last and first line positions to make this Polyline a loop.
  484. * @param {ArcType} [options.arcType=ArcType.GEODESIC] The type of line the polyline segments must follow. Valid options are {@link ArcType.GEODESIC} and {@link ArcType.RHUMB}.
  485. *
  486. * @exception {DeveloperError} At least two positions are required.
  487. *
  488. * @see GroundPolylinePrimitive
  489. *
  490. * @example
  491. * var positions = Cesium.Cartesian3.fromDegreesArray([
  492. * -112.1340164450331, 36.05494287836128,
  493. * -112.08821010582645, 36.097804071380715,
  494. * -112.13296079730024, 36.168769146801104
  495. * ]);
  496. *
  497. * var geometry = new Cesium.GroundPolylineGeometry({
  498. * positions : positions
  499. * });
  500. */
  501. function GroundPolylineGeometry(options) {
  502. options = when.defaultValue(options, when.defaultValue.EMPTY_OBJECT);
  503. var positions = options.positions;
  504. //>>includeStart('debug', pragmas.debug);
  505. if (!when.defined(positions) || positions.length < 2) {
  506. throw new Check.DeveloperError("At least two positions are required.");
  507. }
  508. if (
  509. when.defined(options.arcType) &&
  510. options.arcType !== ArcType.ArcType.GEODESIC &&
  511. options.arcType !== ArcType.ArcType.RHUMB
  512. ) {
  513. throw new Check.DeveloperError(
  514. "Valid options for arcType are ArcType.GEODESIC and ArcType.RHUMB."
  515. );
  516. }
  517. //>>includeEnd('debug');
  518. /**
  519. * The screen space width in pixels.
  520. * @type {Number}
  521. */
  522. this.width = when.defaultValue(options.width, 1.0); // Doesn't get packed, not necessary for computing geometry.
  523. this._positions = positions;
  524. /**
  525. * The distance interval used for interpolating options.points. Zero indicates no interpolation.
  526. * Default of 9999.0 allows centimeter accuracy with 32 bit floating point.
  527. * @type {Boolean}
  528. * @default 9999.0
  529. */
  530. this.granularity = when.defaultValue(options.granularity, 9999.0);
  531. /**
  532. * Whether during geometry creation a line segment will be added between the last and first line positions to make this Polyline a loop.
  533. * If the geometry has two positions this parameter will be ignored.
  534. * @type {Boolean}
  535. * @default false
  536. */
  537. this.loop = when.defaultValue(options.loop, false);
  538. /**
  539. * The type of path the polyline must follow. Valid options are {@link ArcType.GEODESIC} and {@link ArcType.RHUMB}.
  540. * @type {ArcType}
  541. * @default ArcType.GEODESIC
  542. */
  543. this.arcType = when.defaultValue(options.arcType, ArcType.ArcType.GEODESIC);
  544. this._ellipsoid = Cartesian2.Ellipsoid.WGS84;
  545. // MapProjections can't be packed, so store the index to a known MapProjection.
  546. this._projectionIndex = 0;
  547. this._workerName = "createGroundPolylineGeometry";
  548. // Used by GroundPolylinePrimitive to signal worker that scenemode is 3D only.
  549. this._scene3DOnly = false;
  550. }
  551. Object.defineProperties(GroundPolylineGeometry.prototype, {
  552. /**
  553. * The number of elements used to pack the object into an array.
  554. * @memberof GroundPolylineGeometry.prototype
  555. * @type {Number}
  556. * @readonly
  557. * @private
  558. */
  559. packedLength: {
  560. get: function () {
  561. return (
  562. 1.0 +
  563. this._positions.length * 3 +
  564. 1.0 +
  565. 1.0 +
  566. 1.0 +
  567. Cartesian2.Ellipsoid.packedLength +
  568. 1.0 +
  569. 1.0
  570. );
  571. },
  572. },
  573. });
  574. /**
  575. * Set the GroundPolylineGeometry's projection and ellipsoid.
  576. * Used by GroundPolylinePrimitive to signal scene information to the geometry for generating 2D attributes.
  577. *
  578. * @param {GroundPolylineGeometry} groundPolylineGeometry GroundPolylinGeometry describing a polyline on terrain or 3D Tiles.
  579. * @param {Projection} mapProjection A MapProjection used for projecting cartographic coordinates to 2D.
  580. * @private
  581. */
  582. GroundPolylineGeometry.setProjectionAndEllipsoid = function (
  583. groundPolylineGeometry,
  584. mapProjection
  585. ) {
  586. var projectionIndex = 0;
  587. for (var i = 0; i < PROJECTION_COUNT; i++) {
  588. if (mapProjection instanceof PROJECTIONS[i]) {
  589. projectionIndex = i;
  590. break;
  591. }
  592. }
  593. groundPolylineGeometry._projectionIndex = projectionIndex;
  594. groundPolylineGeometry._ellipsoid = mapProjection.ellipsoid;
  595. };
  596. var cart3Scratch1 = new Cartesian2.Cartesian3();
  597. var cart3Scratch2 = new Cartesian2.Cartesian3();
  598. var cart3Scratch3 = new Cartesian2.Cartesian3();
  599. function computeRightNormal(start, end, maxHeight, ellipsoid, result) {
  600. var startBottom = getPosition(ellipsoid, start, 0.0, cart3Scratch1);
  601. var startTop = getPosition(ellipsoid, start, maxHeight, cart3Scratch2);
  602. var endBottom = getPosition(ellipsoid, end, 0.0, cart3Scratch3);
  603. var up = direction(startTop, startBottom, cart3Scratch2);
  604. var forward = direction(endBottom, startBottom, cart3Scratch3);
  605. Cartesian2.Cartesian3.cross(forward, up, result);
  606. return Cartesian2.Cartesian3.normalize(result, result);
  607. }
  608. var interpolatedCartographicScratch = new Cartesian2.Cartographic();
  609. var interpolatedBottomScratch = new Cartesian2.Cartesian3();
  610. var interpolatedTopScratch = new Cartesian2.Cartesian3();
  611. var interpolatedNormalScratch = new Cartesian2.Cartesian3();
  612. function interpolateSegment(
  613. start,
  614. end,
  615. minHeight,
  616. maxHeight,
  617. granularity,
  618. arcType,
  619. ellipsoid,
  620. normalsArray,
  621. bottomPositionsArray,
  622. topPositionsArray,
  623. cartographicsArray
  624. ) {
  625. if (granularity === 0.0) {
  626. return;
  627. }
  628. var ellipsoidLine;
  629. if (arcType === ArcType.ArcType.GEODESIC) {
  630. ellipsoidLine = new EllipsoidGeodesic.EllipsoidGeodesic(start, end, ellipsoid);
  631. } else if (arcType === ArcType.ArcType.RHUMB) {
  632. ellipsoidLine = new EllipsoidRhumbLine.EllipsoidRhumbLine(start, end, ellipsoid);
  633. }
  634. var surfaceDistance = ellipsoidLine.surfaceDistance;
  635. if (surfaceDistance < granularity) {
  636. return;
  637. }
  638. // Compute rightwards normal applicable at all interpolated points
  639. var interpolatedNormal = computeRightNormal(
  640. start,
  641. end,
  642. maxHeight,
  643. ellipsoid,
  644. interpolatedNormalScratch
  645. );
  646. var segments = Math.ceil(surfaceDistance / granularity);
  647. var interpointDistance = surfaceDistance / segments;
  648. var distanceFromStart = interpointDistance;
  649. var pointsToAdd = segments - 1;
  650. var packIndex = normalsArray.length;
  651. for (var i = 0; i < pointsToAdd; i++) {
  652. var interpolatedCartographic = ellipsoidLine.interpolateUsingSurfaceDistance(
  653. distanceFromStart,
  654. interpolatedCartographicScratch
  655. );
  656. var interpolatedBottom = getPosition(
  657. ellipsoid,
  658. interpolatedCartographic,
  659. minHeight,
  660. interpolatedBottomScratch
  661. );
  662. var interpolatedTop = getPosition(
  663. ellipsoid,
  664. interpolatedCartographic,
  665. maxHeight,
  666. interpolatedTopScratch
  667. );
  668. Cartesian2.Cartesian3.pack(interpolatedNormal, normalsArray, packIndex);
  669. Cartesian2.Cartesian3.pack(interpolatedBottom, bottomPositionsArray, packIndex);
  670. Cartesian2.Cartesian3.pack(interpolatedTop, topPositionsArray, packIndex);
  671. cartographicsArray.push(interpolatedCartographic.latitude);
  672. cartographicsArray.push(interpolatedCartographic.longitude);
  673. packIndex += 3;
  674. distanceFromStart += interpointDistance;
  675. }
  676. }
  677. var heightlessCartographicScratch = new Cartesian2.Cartographic();
  678. function getPosition(ellipsoid, cartographic, height, result) {
  679. Cartesian2.Cartographic.clone(cartographic, heightlessCartographicScratch);
  680. heightlessCartographicScratch.height = height;
  681. return Cartesian2.Cartographic.toCartesian(
  682. heightlessCartographicScratch,
  683. ellipsoid,
  684. result
  685. );
  686. }
  687. /**
  688. * Stores the provided instance into the provided array.
  689. *
  690. * @param {PolygonGeometry} value The value to pack.
  691. * @param {Number[]} array The array to pack into.
  692. * @param {Number} [startingIndex=0] The index into the array at which to start packing the elements.
  693. *
  694. * @returns {Number[]} The array that was packed into
  695. */
  696. GroundPolylineGeometry.pack = function (value, array, startingIndex) {
  697. //>>includeStart('debug', pragmas.debug);
  698. Check.Check.typeOf.object("value", value);
  699. Check.Check.defined("array", array);
  700. //>>includeEnd('debug');
  701. var index = when.defaultValue(startingIndex, 0);
  702. var positions = value._positions;
  703. var positionsLength = positions.length;
  704. array[index++] = positionsLength;
  705. for (var i = 0; i < positionsLength; ++i) {
  706. var cartesian = positions[i];
  707. Cartesian2.Cartesian3.pack(cartesian, array, index);
  708. index += 3;
  709. }
  710. array[index++] = value.granularity;
  711. array[index++] = value.loop ? 1.0 : 0.0;
  712. array[index++] = value.arcType;
  713. Cartesian2.Ellipsoid.pack(value._ellipsoid, array, index);
  714. index += Cartesian2.Ellipsoid.packedLength;
  715. array[index++] = value._projectionIndex;
  716. array[index++] = value._scene3DOnly ? 1.0 : 0.0;
  717. return array;
  718. };
  719. /**
  720. * Retrieves an instance from a packed array.
  721. *
  722. * @param {Number[]} array The packed array.
  723. * @param {Number} [startingIndex=0] The starting index of the element to be unpacked.
  724. * @param {PolygonGeometry} [result] The object into which to store the result.
  725. */
  726. GroundPolylineGeometry.unpack = function (array, startingIndex, result) {
  727. //>>includeStart('debug', pragmas.debug);
  728. Check.Check.defined("array", array);
  729. //>>includeEnd('debug');
  730. var index = when.defaultValue(startingIndex, 0);
  731. var positionsLength = array[index++];
  732. var positions = new Array(positionsLength);
  733. for (var i = 0; i < positionsLength; i++) {
  734. positions[i] = Cartesian2.Cartesian3.unpack(array, index);
  735. index += 3;
  736. }
  737. var granularity = array[index++];
  738. var loop = array[index++] === 1.0;
  739. var arcType = array[index++];
  740. var ellipsoid = Cartesian2.Ellipsoid.unpack(array, index);
  741. index += Cartesian2.Ellipsoid.packedLength;
  742. var projectionIndex = array[index++];
  743. var scene3DOnly = array[index++] === 1.0;
  744. if (!when.defined(result)) {
  745. result = new GroundPolylineGeometry({
  746. positions: positions,
  747. });
  748. }
  749. result._positions = positions;
  750. result.granularity = granularity;
  751. result.loop = loop;
  752. result.arcType = arcType;
  753. result._ellipsoid = ellipsoid;
  754. result._projectionIndex = projectionIndex;
  755. result._scene3DOnly = scene3DOnly;
  756. return result;
  757. };
  758. function direction(target, origin, result) {
  759. Cartesian2.Cartesian3.subtract(target, origin, result);
  760. Cartesian2.Cartesian3.normalize(result, result);
  761. return result;
  762. }
  763. function tangentDirection(target, origin, up, result) {
  764. result = direction(target, origin, result);
  765. // orthogonalize
  766. result = Cartesian2.Cartesian3.cross(result, up, result);
  767. result = Cartesian2.Cartesian3.normalize(result, result);
  768. result = Cartesian2.Cartesian3.cross(up, result, result);
  769. return result;
  770. }
  771. var toPreviousScratch = new Cartesian2.Cartesian3();
  772. var toNextScratch = new Cartesian2.Cartesian3();
  773. var forwardScratch = new Cartesian2.Cartesian3();
  774. var vertexUpScratch = new Cartesian2.Cartesian3();
  775. var cosine90 = 0.0;
  776. var cosine180 = -1.0;
  777. function computeVertexMiterNormal(
  778. previousBottom,
  779. vertexBottom,
  780. vertexTop,
  781. nextBottom,
  782. result
  783. ) {
  784. var up = direction(vertexTop, vertexBottom, vertexUpScratch);
  785. // Compute vectors pointing towards neighboring points but tangent to this point on the ellipsoid
  786. var toPrevious = tangentDirection(
  787. previousBottom,
  788. vertexBottom,
  789. up,
  790. toPreviousScratch
  791. );
  792. var toNext = tangentDirection(nextBottom, vertexBottom, up, toNextScratch);
  793. // Check if tangents are almost opposite - if so, no need to miter.
  794. if (
  795. _Math.CesiumMath.equalsEpsilon(
  796. Cartesian2.Cartesian3.dot(toPrevious, toNext),
  797. cosine180,
  798. _Math.CesiumMath.EPSILON5
  799. )
  800. ) {
  801. result = Cartesian2.Cartesian3.cross(up, toPrevious, result);
  802. result = Cartesian2.Cartesian3.normalize(result, result);
  803. return result;
  804. }
  805. // Average directions to previous and to next in the plane of Up
  806. result = Cartesian2.Cartesian3.add(toNext, toPrevious, result);
  807. result = Cartesian2.Cartesian3.normalize(result, result);
  808. // Flip the normal if it isn't pointing roughly bound right (aka if forward is pointing more "backwards")
  809. var forward = Cartesian2.Cartesian3.cross(up, result, forwardScratch);
  810. if (Cartesian2.Cartesian3.dot(toNext, forward) < cosine90) {
  811. result = Cartesian2.Cartesian3.negate(result, result);
  812. }
  813. return result;
  814. }
  815. var XZ_PLANE = Plane.Plane.fromPointNormal(Cartesian2.Cartesian3.ZERO, Cartesian2.Cartesian3.UNIT_Y);
  816. var previousBottomScratch = new Cartesian2.Cartesian3();
  817. var vertexBottomScratch = new Cartesian2.Cartesian3();
  818. var vertexTopScratch = new Cartesian2.Cartesian3();
  819. var nextBottomScratch = new Cartesian2.Cartesian3();
  820. var vertexNormalScratch = new Cartesian2.Cartesian3();
  821. var intersectionScratch = new Cartesian2.Cartesian3();
  822. var cartographicScratch0 = new Cartesian2.Cartographic();
  823. var cartographicScratch1 = new Cartesian2.Cartographic();
  824. var cartographicIntersectionScratch = new Cartesian2.Cartographic();
  825. /**
  826. * Computes shadow volumes for the ground polyline, consisting of its vertices, indices, and a bounding sphere.
  827. * Vertices are "fat," packing all the data needed in each volume to describe a line on terrain or 3D Tiles.
  828. * Should not be called independent of {@link GroundPolylinePrimitive}.
  829. *
  830. * @param {GroundPolylineGeometry} groundPolylineGeometry
  831. * @private
  832. */
  833. GroundPolylineGeometry.createGeometry = function (groundPolylineGeometry) {
  834. var compute2dAttributes = !groundPolylineGeometry._scene3DOnly;
  835. var loop = groundPolylineGeometry.loop;
  836. var ellipsoid = groundPolylineGeometry._ellipsoid;
  837. var granularity = groundPolylineGeometry.granularity;
  838. var arcType = groundPolylineGeometry.arcType;
  839. var projection = new PROJECTIONS[groundPolylineGeometry._projectionIndex](
  840. ellipsoid
  841. );
  842. var minHeight = WALL_INITIAL_MIN_HEIGHT;
  843. var maxHeight = WALL_INITIAL_MAX_HEIGHT;
  844. var index;
  845. var i;
  846. var positions = groundPolylineGeometry._positions;
  847. var positionsLength = positions.length;
  848. if (positionsLength === 2) {
  849. loop = false;
  850. }
  851. // Split positions across the IDL and the Prime Meridian as well.
  852. // Split across prime meridian because very large geometries crossing the Prime Meridian but not the IDL
  853. // may get split by the plane of IDL + Prime Meridian.
  854. var p0;
  855. var p1;
  856. var c0;
  857. var c1;
  858. var rhumbLine = new EllipsoidRhumbLine.EllipsoidRhumbLine(undefined, undefined, ellipsoid);
  859. var intersection;
  860. var intersectionCartographic;
  861. var intersectionLongitude;
  862. var splitPositions = [positions[0]];
  863. for (i = 0; i < positionsLength - 1; i++) {
  864. p0 = positions[i];
  865. p1 = positions[i + 1];
  866. intersection = IntersectionTests.IntersectionTests.lineSegmentPlane(
  867. p0,
  868. p1,
  869. XZ_PLANE,
  870. intersectionScratch
  871. );
  872. if (
  873. when.defined(intersection) &&
  874. !Cartesian2.Cartesian3.equalsEpsilon(intersection, p0, _Math.CesiumMath.EPSILON7) &&
  875. !Cartesian2.Cartesian3.equalsEpsilon(intersection, p1, _Math.CesiumMath.EPSILON7)
  876. ) {
  877. if (groundPolylineGeometry.arcType === ArcType.ArcType.GEODESIC) {
  878. splitPositions.push(Cartesian2.Cartesian3.clone(intersection));
  879. } else if (groundPolylineGeometry.arcType === ArcType.ArcType.RHUMB) {
  880. intersectionLongitude = ellipsoid.cartesianToCartographic(
  881. intersection,
  882. cartographicScratch0
  883. ).longitude;
  884. c0 = ellipsoid.cartesianToCartographic(p0, cartographicScratch0);
  885. c1 = ellipsoid.cartesianToCartographic(p1, cartographicScratch1);
  886. rhumbLine.setEndPoints(c0, c1);
  887. intersectionCartographic = rhumbLine.findIntersectionWithLongitude(
  888. intersectionLongitude,
  889. cartographicIntersectionScratch
  890. );
  891. intersection = ellipsoid.cartographicToCartesian(
  892. intersectionCartographic,
  893. intersectionScratch
  894. );
  895. if (
  896. when.defined(intersection) &&
  897. !Cartesian2.Cartesian3.equalsEpsilon(intersection, p0, _Math.CesiumMath.EPSILON7) &&
  898. !Cartesian2.Cartesian3.equalsEpsilon(intersection, p1, _Math.CesiumMath.EPSILON7)
  899. ) {
  900. splitPositions.push(Cartesian2.Cartesian3.clone(intersection));
  901. }
  902. }
  903. }
  904. splitPositions.push(p1);
  905. }
  906. if (loop) {
  907. p0 = positions[positionsLength - 1];
  908. p1 = positions[0];
  909. intersection = IntersectionTests.IntersectionTests.lineSegmentPlane(
  910. p0,
  911. p1,
  912. XZ_PLANE,
  913. intersectionScratch
  914. );
  915. if (
  916. when.defined(intersection) &&
  917. !Cartesian2.Cartesian3.equalsEpsilon(intersection, p0, _Math.CesiumMath.EPSILON7) &&
  918. !Cartesian2.Cartesian3.equalsEpsilon(intersection, p1, _Math.CesiumMath.EPSILON7)
  919. ) {
  920. if (groundPolylineGeometry.arcType === ArcType.ArcType.GEODESIC) {
  921. splitPositions.push(Cartesian2.Cartesian3.clone(intersection));
  922. } else if (groundPolylineGeometry.arcType === ArcType.ArcType.RHUMB) {
  923. intersectionLongitude = ellipsoid.cartesianToCartographic(
  924. intersection,
  925. cartographicScratch0
  926. ).longitude;
  927. c0 = ellipsoid.cartesianToCartographic(p0, cartographicScratch0);
  928. c1 = ellipsoid.cartesianToCartographic(p1, cartographicScratch1);
  929. rhumbLine.setEndPoints(c0, c1);
  930. intersectionCartographic = rhumbLine.findIntersectionWithLongitude(
  931. intersectionLongitude,
  932. cartographicIntersectionScratch
  933. );
  934. intersection = ellipsoid.cartographicToCartesian(
  935. intersectionCartographic,
  936. intersectionScratch
  937. );
  938. if (
  939. when.defined(intersection) &&
  940. !Cartesian2.Cartesian3.equalsEpsilon(intersection, p0, _Math.CesiumMath.EPSILON7) &&
  941. !Cartesian2.Cartesian3.equalsEpsilon(intersection, p1, _Math.CesiumMath.EPSILON7)
  942. ) {
  943. splitPositions.push(Cartesian2.Cartesian3.clone(intersection));
  944. }
  945. }
  946. }
  947. }
  948. var cartographicsLength = splitPositions.length;
  949. var cartographics = new Array(cartographicsLength);
  950. for (i = 0; i < cartographicsLength; i++) {
  951. var cartographic = Cartesian2.Cartographic.fromCartesian(splitPositions[i], ellipsoid);
  952. cartographic.height = 0.0;
  953. cartographics[i] = cartographic;
  954. }
  955. cartographics = arrayRemoveDuplicates.arrayRemoveDuplicates(
  956. cartographics,
  957. Cartesian2.Cartographic.equalsEpsilon
  958. );
  959. cartographicsLength = cartographics.length;
  960. if (cartographicsLength < 2) {
  961. return undefined;
  962. }
  963. /**** Build heap-side arrays for positions, interpolated cartographics, and normals from which to compute vertices ****/
  964. // We build a "wall" and then decompose it into separately connected component "volumes" because we need a lot
  965. // of information about the wall. Also, this simplifies interpolation.
  966. // Convention: "next" and "end" are locally forward to each segment of the wall,
  967. // and we are computing normals pointing towards the local right side of the vertices in each segment.
  968. var cartographicsArray = [];
  969. var normalsArray = [];
  970. var bottomPositionsArray = [];
  971. var topPositionsArray = [];
  972. var previousBottom = previousBottomScratch;
  973. var vertexBottom = vertexBottomScratch;
  974. var vertexTop = vertexTopScratch;
  975. var nextBottom = nextBottomScratch;
  976. var vertexNormal = vertexNormalScratch;
  977. // First point - either loop or attach a "perpendicular" normal
  978. var startCartographic = cartographics[0];
  979. var nextCartographic = cartographics[1];
  980. var prestartCartographic = cartographics[cartographicsLength - 1];
  981. previousBottom = getPosition(
  982. ellipsoid,
  983. prestartCartographic,
  984. minHeight,
  985. previousBottom
  986. );
  987. nextBottom = getPosition(ellipsoid, nextCartographic, minHeight, nextBottom);
  988. vertexBottom = getPosition(
  989. ellipsoid,
  990. startCartographic,
  991. minHeight,
  992. vertexBottom
  993. );
  994. vertexTop = getPosition(ellipsoid, startCartographic, maxHeight, vertexTop);
  995. if (loop) {
  996. vertexNormal = computeVertexMiterNormal(
  997. previousBottom,
  998. vertexBottom,
  999. vertexTop,
  1000. nextBottom,
  1001. vertexNormal
  1002. );
  1003. } else {
  1004. vertexNormal = computeRightNormal(
  1005. startCartographic,
  1006. nextCartographic,
  1007. maxHeight,
  1008. ellipsoid,
  1009. vertexNormal
  1010. );
  1011. }
  1012. Cartesian2.Cartesian3.pack(vertexNormal, normalsArray, 0);
  1013. Cartesian2.Cartesian3.pack(vertexBottom, bottomPositionsArray, 0);
  1014. Cartesian2.Cartesian3.pack(vertexTop, topPositionsArray, 0);
  1015. cartographicsArray.push(startCartographic.latitude);
  1016. cartographicsArray.push(startCartographic.longitude);
  1017. interpolateSegment(
  1018. startCartographic,
  1019. nextCartographic,
  1020. minHeight,
  1021. maxHeight,
  1022. granularity,
  1023. arcType,
  1024. ellipsoid,
  1025. normalsArray,
  1026. bottomPositionsArray,
  1027. topPositionsArray,
  1028. cartographicsArray
  1029. );
  1030. // All inbetween points
  1031. for (i = 1; i < cartographicsLength - 1; ++i) {
  1032. previousBottom = Cartesian2.Cartesian3.clone(vertexBottom, previousBottom);
  1033. vertexBottom = Cartesian2.Cartesian3.clone(nextBottom, vertexBottom);
  1034. var vertexCartographic = cartographics[i];
  1035. getPosition(ellipsoid, vertexCartographic, maxHeight, vertexTop);
  1036. getPosition(ellipsoid, cartographics[i + 1], minHeight, nextBottom);
  1037. computeVertexMiterNormal(
  1038. previousBottom,
  1039. vertexBottom,
  1040. vertexTop,
  1041. nextBottom,
  1042. vertexNormal
  1043. );
  1044. index = normalsArray.length;
  1045. Cartesian2.Cartesian3.pack(vertexNormal, normalsArray, index);
  1046. Cartesian2.Cartesian3.pack(vertexBottom, bottomPositionsArray, index);
  1047. Cartesian2.Cartesian3.pack(vertexTop, topPositionsArray, index);
  1048. cartographicsArray.push(vertexCartographic.latitude);
  1049. cartographicsArray.push(vertexCartographic.longitude);
  1050. interpolateSegment(
  1051. cartographics[i],
  1052. cartographics[i + 1],
  1053. minHeight,
  1054. maxHeight,
  1055. granularity,
  1056. arcType,
  1057. ellipsoid,
  1058. normalsArray,
  1059. bottomPositionsArray,
  1060. topPositionsArray,
  1061. cartographicsArray
  1062. );
  1063. }
  1064. // Last point - either loop or attach a normal "perpendicular" to the wall.
  1065. var endCartographic = cartographics[cartographicsLength - 1];
  1066. var preEndCartographic = cartographics[cartographicsLength - 2];
  1067. vertexBottom = getPosition(
  1068. ellipsoid,
  1069. endCartographic,
  1070. minHeight,
  1071. vertexBottom
  1072. );
  1073. vertexTop = getPosition(ellipsoid, endCartographic, maxHeight, vertexTop);
  1074. if (loop) {
  1075. var postEndCartographic = cartographics[0];
  1076. previousBottom = getPosition(
  1077. ellipsoid,
  1078. preEndCartographic,
  1079. minHeight,
  1080. previousBottom
  1081. );
  1082. nextBottom = getPosition(
  1083. ellipsoid,
  1084. postEndCartographic,
  1085. minHeight,
  1086. nextBottom
  1087. );
  1088. vertexNormal = computeVertexMiterNormal(
  1089. previousBottom,
  1090. vertexBottom,
  1091. vertexTop,
  1092. nextBottom,
  1093. vertexNormal
  1094. );
  1095. } else {
  1096. vertexNormal = computeRightNormal(
  1097. preEndCartographic,
  1098. endCartographic,
  1099. maxHeight,
  1100. ellipsoid,
  1101. vertexNormal
  1102. );
  1103. }
  1104. index = normalsArray.length;
  1105. Cartesian2.Cartesian3.pack(vertexNormal, normalsArray, index);
  1106. Cartesian2.Cartesian3.pack(vertexBottom, bottomPositionsArray, index);
  1107. Cartesian2.Cartesian3.pack(vertexTop, topPositionsArray, index);
  1108. cartographicsArray.push(endCartographic.latitude);
  1109. cartographicsArray.push(endCartographic.longitude);
  1110. if (loop) {
  1111. interpolateSegment(
  1112. endCartographic,
  1113. startCartographic,
  1114. minHeight,
  1115. maxHeight,
  1116. granularity,
  1117. arcType,
  1118. ellipsoid,
  1119. normalsArray,
  1120. bottomPositionsArray,
  1121. topPositionsArray,
  1122. cartographicsArray
  1123. );
  1124. index = normalsArray.length;
  1125. for (i = 0; i < 3; ++i) {
  1126. normalsArray[index + i] = normalsArray[i];
  1127. bottomPositionsArray[index + i] = bottomPositionsArray[i];
  1128. topPositionsArray[index + i] = topPositionsArray[i];
  1129. }
  1130. cartographicsArray.push(startCartographic.latitude);
  1131. cartographicsArray.push(startCartographic.longitude);
  1132. }
  1133. return generateGeometryAttributes(
  1134. loop,
  1135. projection,
  1136. bottomPositionsArray,
  1137. topPositionsArray,
  1138. normalsArray,
  1139. cartographicsArray,
  1140. compute2dAttributes
  1141. );
  1142. };
  1143. // If the end normal angle is too steep compared to the direction of the line segment,
  1144. // "break" the miter by rotating the normal 90 degrees around the "up" direction at the point
  1145. // For ultra precision we would want to project into a plane, but in practice this is sufficient.
  1146. var lineDirectionScratch = new Cartesian2.Cartesian3();
  1147. var matrix3Scratch = new Transforms.Matrix3();
  1148. var quaternionScratch = new Transforms.Quaternion();
  1149. function breakMiter(endGeometryNormal, startBottom, endBottom, endTop) {
  1150. var lineDirection = direction(endBottom, startBottom, lineDirectionScratch);
  1151. var dot = Cartesian2.Cartesian3.dot(lineDirection, endGeometryNormal);
  1152. if (dot > MITER_BREAK_SMALL || dot < MITER_BREAK_LARGE) {
  1153. var vertexUp = direction(endTop, endBottom, vertexUpScratch);
  1154. var angle =
  1155. dot < MITER_BREAK_LARGE
  1156. ? _Math.CesiumMath.PI_OVER_TWO
  1157. : -_Math.CesiumMath.PI_OVER_TWO;
  1158. var quaternion = Transforms.Quaternion.fromAxisAngle(
  1159. vertexUp,
  1160. angle,
  1161. quaternionScratch
  1162. );
  1163. var rotationMatrix = Transforms.Matrix3.fromQuaternion(quaternion, matrix3Scratch);
  1164. Transforms.Matrix3.multiplyByVector(
  1165. rotationMatrix,
  1166. endGeometryNormal,
  1167. endGeometryNormal
  1168. );
  1169. return true;
  1170. }
  1171. return false;
  1172. }
  1173. var endPosCartographicScratch = new Cartesian2.Cartographic();
  1174. var normalStartpointScratch = new Cartesian2.Cartesian3();
  1175. var normalEndpointScratch = new Cartesian2.Cartesian3();
  1176. function projectNormal(
  1177. projection,
  1178. cartographic,
  1179. normal,
  1180. projectedPosition,
  1181. result
  1182. ) {
  1183. var position = Cartesian2.Cartographic.toCartesian(
  1184. cartographic,
  1185. projection._ellipsoid,
  1186. normalStartpointScratch
  1187. );
  1188. var normalEndpoint = Cartesian2.Cartesian3.add(position, normal, normalEndpointScratch);
  1189. var flipNormal = false;
  1190. var ellipsoid = projection._ellipsoid;
  1191. var normalEndpointCartographic = ellipsoid.cartesianToCartographic(
  1192. normalEndpoint,
  1193. endPosCartographicScratch
  1194. );
  1195. // If normal crosses the IDL, go the other way and flip the result.
  1196. // In practice this almost never happens because the cartographic start
  1197. // and end points of each segment are "nudged" to be on the same side
  1198. // of the IDL and slightly away from the IDL.
  1199. if (
  1200. Math.abs(cartographic.longitude - normalEndpointCartographic.longitude) >
  1201. _Math.CesiumMath.PI_OVER_TWO
  1202. ) {
  1203. flipNormal = true;
  1204. normalEndpoint = Cartesian2.Cartesian3.subtract(
  1205. position,
  1206. normal,
  1207. normalEndpointScratch
  1208. );
  1209. normalEndpointCartographic = ellipsoid.cartesianToCartographic(
  1210. normalEndpoint,
  1211. endPosCartographicScratch
  1212. );
  1213. }
  1214. normalEndpointCartographic.height = 0.0;
  1215. var normalEndpointProjected = projection.project(
  1216. normalEndpointCartographic,
  1217. result
  1218. );
  1219. result = Cartesian2.Cartesian3.subtract(
  1220. normalEndpointProjected,
  1221. projectedPosition,
  1222. result
  1223. );
  1224. result.z = 0.0;
  1225. result = Cartesian2.Cartesian3.normalize(result, result);
  1226. if (flipNormal) {
  1227. Cartesian2.Cartesian3.negate(result, result);
  1228. }
  1229. return result;
  1230. }
  1231. var adjustHeightNormalScratch = new Cartesian2.Cartesian3();
  1232. var adjustHeightOffsetScratch = new Cartesian2.Cartesian3();
  1233. function adjustHeights(
  1234. bottom,
  1235. top,
  1236. minHeight,
  1237. maxHeight,
  1238. adjustHeightBottom,
  1239. adjustHeightTop
  1240. ) {
  1241. // bottom and top should be at WALL_INITIAL_MIN_HEIGHT and WALL_INITIAL_MAX_HEIGHT, respectively
  1242. var adjustHeightNormal = Cartesian2.Cartesian3.subtract(
  1243. top,
  1244. bottom,
  1245. adjustHeightNormalScratch
  1246. );
  1247. Cartesian2.Cartesian3.normalize(adjustHeightNormal, adjustHeightNormal);
  1248. var distanceForBottom = minHeight - WALL_INITIAL_MIN_HEIGHT;
  1249. var adjustHeightOffset = Cartesian2.Cartesian3.multiplyByScalar(
  1250. adjustHeightNormal,
  1251. distanceForBottom,
  1252. adjustHeightOffsetScratch
  1253. );
  1254. Cartesian2.Cartesian3.add(bottom, adjustHeightOffset, adjustHeightBottom);
  1255. var distanceForTop = maxHeight - WALL_INITIAL_MAX_HEIGHT;
  1256. adjustHeightOffset = Cartesian2.Cartesian3.multiplyByScalar(
  1257. adjustHeightNormal,
  1258. distanceForTop,
  1259. adjustHeightOffsetScratch
  1260. );
  1261. Cartesian2.Cartesian3.add(top, adjustHeightOffset, adjustHeightTop);
  1262. }
  1263. var nudgeDirectionScratch = new Cartesian2.Cartesian3();
  1264. function nudgeXZ(start, end) {
  1265. var startToXZdistance = Plane.Plane.getPointDistance(XZ_PLANE, start);
  1266. var endToXZdistance = Plane.Plane.getPointDistance(XZ_PLANE, end);
  1267. var offset = nudgeDirectionScratch;
  1268. // Larger epsilon than what's used in GeometryPipeline, a centimeter in world space
  1269. if (_Math.CesiumMath.equalsEpsilon(startToXZdistance, 0.0, _Math.CesiumMath.EPSILON2)) {
  1270. offset = direction(end, start, offset);
  1271. Cartesian2.Cartesian3.multiplyByScalar(offset, _Math.CesiumMath.EPSILON2, offset);
  1272. Cartesian2.Cartesian3.add(start, offset, start);
  1273. } else if (
  1274. _Math.CesiumMath.equalsEpsilon(endToXZdistance, 0.0, _Math.CesiumMath.EPSILON2)
  1275. ) {
  1276. offset = direction(start, end, offset);
  1277. Cartesian2.Cartesian3.multiplyByScalar(offset, _Math.CesiumMath.EPSILON2, offset);
  1278. Cartesian2.Cartesian3.add(end, offset, end);
  1279. }
  1280. }
  1281. // "Nudge" cartographic coordinates so start and end are on the same side of the IDL.
  1282. // Nudge amounts are tiny, basically just an IDL flip.
  1283. // Only used for 2D/CV.
  1284. function nudgeCartographic(start, end) {
  1285. var absStartLon = Math.abs(start.longitude);
  1286. var absEndLon = Math.abs(end.longitude);
  1287. if (
  1288. _Math.CesiumMath.equalsEpsilon(absStartLon, _Math.CesiumMath.PI, _Math.CesiumMath.EPSILON11)
  1289. ) {
  1290. var endSign = _Math.CesiumMath.sign(end.longitude);
  1291. start.longitude = endSign * (absStartLon - _Math.CesiumMath.EPSILON11);
  1292. return 1;
  1293. } else if (
  1294. _Math.CesiumMath.equalsEpsilon(absEndLon, _Math.CesiumMath.PI, _Math.CesiumMath.EPSILON11)
  1295. ) {
  1296. var startSign = _Math.CesiumMath.sign(start.longitude);
  1297. end.longitude = startSign * (absEndLon - _Math.CesiumMath.EPSILON11);
  1298. return 2;
  1299. }
  1300. return 0;
  1301. }
  1302. var startCartographicScratch = new Cartesian2.Cartographic();
  1303. var endCartographicScratch = new Cartesian2.Cartographic();
  1304. var segmentStartTopScratch = new Cartesian2.Cartesian3();
  1305. var segmentEndTopScratch = new Cartesian2.Cartesian3();
  1306. var segmentStartBottomScratch = new Cartesian2.Cartesian3();
  1307. var segmentEndBottomScratch = new Cartesian2.Cartesian3();
  1308. var segmentStartNormalScratch = new Cartesian2.Cartesian3();
  1309. var segmentEndNormalScratch = new Cartesian2.Cartesian3();
  1310. var getHeightCartographics = [startCartographicScratch, endCartographicScratch];
  1311. var getHeightRectangleScratch = new Cartesian2.Rectangle();
  1312. var adjustHeightStartTopScratch = new Cartesian2.Cartesian3();
  1313. var adjustHeightEndTopScratch = new Cartesian2.Cartesian3();
  1314. var adjustHeightStartBottomScratch = new Cartesian2.Cartesian3();
  1315. var adjustHeightEndBottomScratch = new Cartesian2.Cartesian3();
  1316. var segmentStart2DScratch = new Cartesian2.Cartesian3();
  1317. var segmentEnd2DScratch = new Cartesian2.Cartesian3();
  1318. var segmentStartNormal2DScratch = new Cartesian2.Cartesian3();
  1319. var segmentEndNormal2DScratch = new Cartesian2.Cartesian3();
  1320. var offsetScratch = new Cartesian2.Cartesian3();
  1321. var startUpScratch = new Cartesian2.Cartesian3();
  1322. var endUpScratch = new Cartesian2.Cartesian3();
  1323. var rightScratch = new Cartesian2.Cartesian3();
  1324. var startPlaneNormalScratch = new Cartesian2.Cartesian3();
  1325. var endPlaneNormalScratch = new Cartesian2.Cartesian3();
  1326. var encodeScratch = new EncodedCartesian3.EncodedCartesian3();
  1327. var encodeScratch2D = new EncodedCartesian3.EncodedCartesian3();
  1328. var forwardOffset2DScratch = new Cartesian2.Cartesian3();
  1329. var right2DScratch = new Cartesian2.Cartesian3();
  1330. var normalNudgeScratch = new Cartesian2.Cartesian3();
  1331. var scratchBoundingSpheres = [new Transforms.BoundingSphere(), new Transforms.BoundingSphere()];
  1332. // Winding order is reversed so each segment's volume is inside-out
  1333. var REFERENCE_INDICES = [
  1334. 0,
  1335. 2,
  1336. 1,
  1337. 0,
  1338. 3,
  1339. 2, // right
  1340. 0,
  1341. 7,
  1342. 3,
  1343. 0,
  1344. 4,
  1345. 7, // start
  1346. 0,
  1347. 5,
  1348. 4,
  1349. 0,
  1350. 1,
  1351. 5, // bottom
  1352. 5,
  1353. 7,
  1354. 4,
  1355. 5,
  1356. 6,
  1357. 7, // left
  1358. 5,
  1359. 2,
  1360. 6,
  1361. 5,
  1362. 1,
  1363. 2, // end
  1364. 3,
  1365. 6,
  1366. 2,
  1367. 3,
  1368. 7,
  1369. 6, // top
  1370. ];
  1371. var REFERENCE_INDICES_LENGTH = REFERENCE_INDICES.length;
  1372. // Decompose the "wall" into a series of shadow volumes.
  1373. // Each shadow volume's vertices encode a description of the line it contains,
  1374. // including mitering planes at the end points, a plane along the line itself,
  1375. // and attributes for computing length-wise texture coordinates.
  1376. function generateGeometryAttributes(
  1377. loop,
  1378. projection,
  1379. bottomPositionsArray,
  1380. topPositionsArray,
  1381. normalsArray,
  1382. cartographicsArray,
  1383. compute2dAttributes
  1384. ) {
  1385. var i;
  1386. var index;
  1387. var ellipsoid = projection._ellipsoid;
  1388. // Each segment will have 8 vertices
  1389. var segmentCount = bottomPositionsArray.length / 3 - 1;
  1390. var vertexCount = segmentCount * 8;
  1391. var arraySizeVec4 = vertexCount * 4;
  1392. var indexCount = segmentCount * 36;
  1393. var indices =
  1394. vertexCount > 65535
  1395. ? new Uint32Array(indexCount)
  1396. : new Uint16Array(indexCount);
  1397. var positionsArray = new Float64Array(vertexCount * 3);
  1398. var startHiAndForwardOffsetX = new Float32Array(arraySizeVec4);
  1399. var startLoAndForwardOffsetY = new Float32Array(arraySizeVec4);
  1400. var startNormalAndForwardOffsetZ = new Float32Array(arraySizeVec4);
  1401. var endNormalAndTextureCoordinateNormalizationX = new Float32Array(
  1402. arraySizeVec4
  1403. );
  1404. var rightNormalAndTextureCoordinateNormalizationY = new Float32Array(
  1405. arraySizeVec4
  1406. );
  1407. var startHiLo2D;
  1408. var offsetAndRight2D;
  1409. var startEndNormals2D;
  1410. var texcoordNormalization2D;
  1411. if (compute2dAttributes) {
  1412. startHiLo2D = new Float32Array(arraySizeVec4);
  1413. offsetAndRight2D = new Float32Array(arraySizeVec4);
  1414. startEndNormals2D = new Float32Array(arraySizeVec4);
  1415. texcoordNormalization2D = new Float32Array(vertexCount * 2);
  1416. }
  1417. /*** Compute total lengths for texture coordinate normalization ***/
  1418. // 2D
  1419. var cartographicsLength = cartographicsArray.length / 2;
  1420. var length2D = 0.0;
  1421. var startCartographic = startCartographicScratch;
  1422. startCartographic.height = 0.0;
  1423. var endCartographic = endCartographicScratch;
  1424. endCartographic.height = 0.0;
  1425. var segmentStartCartesian = segmentStartTopScratch;
  1426. var segmentEndCartesian = segmentEndTopScratch;
  1427. if (compute2dAttributes) {
  1428. index = 0;
  1429. for (i = 1; i < cartographicsLength; i++) {
  1430. // Don't clone anything from previous segment b/c possible IDL touch
  1431. startCartographic.latitude = cartographicsArray[index];
  1432. startCartographic.longitude = cartographicsArray[index + 1];
  1433. endCartographic.latitude = cartographicsArray[index + 2];
  1434. endCartographic.longitude = cartographicsArray[index + 3];
  1435. segmentStartCartesian = projection.project(
  1436. startCartographic,
  1437. segmentStartCartesian
  1438. );
  1439. segmentEndCartesian = projection.project(
  1440. endCartographic,
  1441. segmentEndCartesian
  1442. );
  1443. length2D += Cartesian2.Cartesian3.distance(
  1444. segmentStartCartesian,
  1445. segmentEndCartesian
  1446. );
  1447. index += 2;
  1448. }
  1449. }
  1450. // 3D
  1451. var positionsLength = topPositionsArray.length / 3;
  1452. segmentEndCartesian = Cartesian2.Cartesian3.unpack(
  1453. topPositionsArray,
  1454. 0,
  1455. segmentEndCartesian
  1456. );
  1457. var length3D = 0.0;
  1458. index = 3;
  1459. for (i = 1; i < positionsLength; i++) {
  1460. segmentStartCartesian = Cartesian2.Cartesian3.clone(
  1461. segmentEndCartesian,
  1462. segmentStartCartesian
  1463. );
  1464. segmentEndCartesian = Cartesian2.Cartesian3.unpack(
  1465. topPositionsArray,
  1466. index,
  1467. segmentEndCartesian
  1468. );
  1469. length3D += Cartesian2.Cartesian3.distance(segmentStartCartesian, segmentEndCartesian);
  1470. index += 3;
  1471. }
  1472. /*** Generate segments ***/
  1473. var j;
  1474. index = 3;
  1475. var cartographicsIndex = 0;
  1476. var vec2sWriteIndex = 0;
  1477. var vec3sWriteIndex = 0;
  1478. var vec4sWriteIndex = 0;
  1479. var miterBroken = false;
  1480. var endBottom = Cartesian2.Cartesian3.unpack(
  1481. bottomPositionsArray,
  1482. 0,
  1483. segmentEndBottomScratch
  1484. );
  1485. var endTop = Cartesian2.Cartesian3.unpack(topPositionsArray, 0, segmentEndTopScratch);
  1486. var endGeometryNormal = Cartesian2.Cartesian3.unpack(
  1487. normalsArray,
  1488. 0,
  1489. segmentEndNormalScratch
  1490. );
  1491. if (loop) {
  1492. var preEndBottom = Cartesian2.Cartesian3.unpack(
  1493. bottomPositionsArray,
  1494. bottomPositionsArray.length - 6,
  1495. segmentStartBottomScratch
  1496. );
  1497. if (breakMiter(endGeometryNormal, preEndBottom, endBottom, endTop)) {
  1498. // Miter broken as if for the last point in the loop, needs to be inverted for first point (clone of endBottom)
  1499. endGeometryNormal = Cartesian2.Cartesian3.negate(
  1500. endGeometryNormal,
  1501. endGeometryNormal
  1502. );
  1503. }
  1504. }
  1505. var lengthSoFar3D = 0.0;
  1506. var lengthSoFar2D = 0.0;
  1507. // For translating bounding volume
  1508. var sumHeights = 0.0;
  1509. for (i = 0; i < segmentCount; i++) {
  1510. var startBottom = Cartesian2.Cartesian3.clone(endBottom, segmentStartBottomScratch);
  1511. var startTop = Cartesian2.Cartesian3.clone(endTop, segmentStartTopScratch);
  1512. var startGeometryNormal = Cartesian2.Cartesian3.clone(
  1513. endGeometryNormal,
  1514. segmentStartNormalScratch
  1515. );
  1516. if (miterBroken) {
  1517. startGeometryNormal = Cartesian2.Cartesian3.negate(
  1518. startGeometryNormal,
  1519. startGeometryNormal
  1520. );
  1521. }
  1522. endBottom = Cartesian2.Cartesian3.unpack(
  1523. bottomPositionsArray,
  1524. index,
  1525. segmentEndBottomScratch
  1526. );
  1527. endTop = Cartesian2.Cartesian3.unpack(topPositionsArray, index, segmentEndTopScratch);
  1528. endGeometryNormal = Cartesian2.Cartesian3.unpack(
  1529. normalsArray,
  1530. index,
  1531. segmentEndNormalScratch
  1532. );
  1533. miterBroken = breakMiter(endGeometryNormal, startBottom, endBottom, endTop);
  1534. // 2D - don't clone anything from previous segment b/c possible IDL touch
  1535. startCartographic.latitude = cartographicsArray[cartographicsIndex];
  1536. startCartographic.longitude = cartographicsArray[cartographicsIndex + 1];
  1537. endCartographic.latitude = cartographicsArray[cartographicsIndex + 2];
  1538. endCartographic.longitude = cartographicsArray[cartographicsIndex + 3];
  1539. var start2D;
  1540. var end2D;
  1541. var startGeometryNormal2D;
  1542. var endGeometryNormal2D;
  1543. if (compute2dAttributes) {
  1544. var nudgeResult = nudgeCartographic(startCartographic, endCartographic);
  1545. start2D = projection.project(startCartographic, segmentStart2DScratch);
  1546. end2D = projection.project(endCartographic, segmentEnd2DScratch);
  1547. var direction2D = direction(end2D, start2D, forwardOffset2DScratch);
  1548. direction2D.y = Math.abs(direction2D.y);
  1549. startGeometryNormal2D = segmentStartNormal2DScratch;
  1550. endGeometryNormal2D = segmentEndNormal2DScratch;
  1551. if (
  1552. nudgeResult === 0 ||
  1553. Cartesian2.Cartesian3.dot(direction2D, Cartesian2.Cartesian3.UNIT_Y) > MITER_BREAK_SMALL
  1554. ) {
  1555. // No nudge - project the original normal
  1556. // Or, if the line's angle relative to the IDL is very acute,
  1557. // in which case snapping will produce oddly shaped volumes.
  1558. startGeometryNormal2D = projectNormal(
  1559. projection,
  1560. startCartographic,
  1561. startGeometryNormal,
  1562. start2D,
  1563. segmentStartNormal2DScratch
  1564. );
  1565. endGeometryNormal2D = projectNormal(
  1566. projection,
  1567. endCartographic,
  1568. endGeometryNormal,
  1569. end2D,
  1570. segmentEndNormal2DScratch
  1571. );
  1572. } else if (nudgeResult === 1) {
  1573. // Start is close to IDL - snap start normal to align with IDL
  1574. endGeometryNormal2D = projectNormal(
  1575. projection,
  1576. endCartographic,
  1577. endGeometryNormal,
  1578. end2D,
  1579. segmentEndNormal2DScratch
  1580. );
  1581. startGeometryNormal2D.x = 0.0;
  1582. // If start longitude is negative and end longitude is less negative, relative right is unit -Y
  1583. // If start longitude is positive and end longitude is less positive, relative right is unit +Y
  1584. startGeometryNormal2D.y = _Math.CesiumMath.sign(
  1585. startCartographic.longitude - Math.abs(endCartographic.longitude)
  1586. );
  1587. startGeometryNormal2D.z = 0.0;
  1588. } else {
  1589. // End is close to IDL - snap end normal to align with IDL
  1590. startGeometryNormal2D = projectNormal(
  1591. projection,
  1592. startCartographic,
  1593. startGeometryNormal,
  1594. start2D,
  1595. segmentStartNormal2DScratch
  1596. );
  1597. endGeometryNormal2D.x = 0.0;
  1598. // If end longitude is negative and start longitude is less negative, relative right is unit Y
  1599. // If end longitude is positive and start longitude is less positive, relative right is unit -Y
  1600. endGeometryNormal2D.y = _Math.CesiumMath.sign(
  1601. startCartographic.longitude - endCartographic.longitude
  1602. );
  1603. endGeometryNormal2D.z = 0.0;
  1604. }
  1605. }
  1606. /****************************************
  1607. * Geometry descriptors of a "line on terrain,"
  1608. * as opposed to the "shadow volume used to draw
  1609. * the line on terrain":
  1610. * - position of start + offset to end
  1611. * - start, end, and right-facing planes
  1612. * - encoded texture coordinate offsets
  1613. ****************************************/
  1614. /* 3D */
  1615. var segmentLength3D = Cartesian2.Cartesian3.distance(startTop, endTop);
  1616. var encodedStart = EncodedCartesian3.EncodedCartesian3.fromCartesian(
  1617. startBottom,
  1618. encodeScratch
  1619. );
  1620. var forwardOffset = Cartesian2.Cartesian3.subtract(
  1621. endBottom,
  1622. startBottom,
  1623. offsetScratch
  1624. );
  1625. var forward = Cartesian2.Cartesian3.normalize(forwardOffset, rightScratch);
  1626. var startUp = Cartesian2.Cartesian3.subtract(startTop, startBottom, startUpScratch);
  1627. startUp = Cartesian2.Cartesian3.normalize(startUp, startUp);
  1628. var rightNormal = Cartesian2.Cartesian3.cross(forward, startUp, rightScratch);
  1629. rightNormal = Cartesian2.Cartesian3.normalize(rightNormal, rightNormal);
  1630. var startPlaneNormal = Cartesian2.Cartesian3.cross(
  1631. startUp,
  1632. startGeometryNormal,
  1633. startPlaneNormalScratch
  1634. );
  1635. startPlaneNormal = Cartesian2.Cartesian3.normalize(startPlaneNormal, startPlaneNormal);
  1636. var endUp = Cartesian2.Cartesian3.subtract(endTop, endBottom, endUpScratch);
  1637. endUp = Cartesian2.Cartesian3.normalize(endUp, endUp);
  1638. var endPlaneNormal = Cartesian2.Cartesian3.cross(
  1639. endGeometryNormal,
  1640. endUp,
  1641. endPlaneNormalScratch
  1642. );
  1643. endPlaneNormal = Cartesian2.Cartesian3.normalize(endPlaneNormal, endPlaneNormal);
  1644. var texcoordNormalization3DX = segmentLength3D / length3D;
  1645. var texcoordNormalization3DY = lengthSoFar3D / length3D;
  1646. /* 2D */
  1647. var segmentLength2D = 0.0;
  1648. var encodedStart2D;
  1649. var forwardOffset2D;
  1650. var right2D;
  1651. var texcoordNormalization2DX = 0.0;
  1652. var texcoordNormalization2DY = 0.0;
  1653. if (compute2dAttributes) {
  1654. segmentLength2D = Cartesian2.Cartesian3.distance(start2D, end2D);
  1655. encodedStart2D = EncodedCartesian3.EncodedCartesian3.fromCartesian(
  1656. start2D,
  1657. encodeScratch2D
  1658. );
  1659. forwardOffset2D = Cartesian2.Cartesian3.subtract(
  1660. end2D,
  1661. start2D,
  1662. forwardOffset2DScratch
  1663. );
  1664. // Right direction is just forward direction rotated by -90 degrees around Z
  1665. // Similarly with plane normals
  1666. right2D = Cartesian2.Cartesian3.normalize(forwardOffset2D, right2DScratch);
  1667. var swap = right2D.x;
  1668. right2D.x = right2D.y;
  1669. right2D.y = -swap;
  1670. texcoordNormalization2DX = segmentLength2D / length2D;
  1671. texcoordNormalization2DY = lengthSoFar2D / length2D;
  1672. }
  1673. /** Pack **/
  1674. for (j = 0; j < 8; j++) {
  1675. var vec4Index = vec4sWriteIndex + j * 4;
  1676. var vec2Index = vec2sWriteIndex + j * 2;
  1677. var wIndex = vec4Index + 3;
  1678. // Encode sidedness of vertex relative to right plane in texture coordinate normalization X,
  1679. // whether vertex is top or bottom of volume in sign/magnitude of normalization Y.
  1680. var rightPlaneSide = j < 4 ? 1.0 : -1.0;
  1681. var topBottomSide = j === 2 || j === 3 || j === 6 || j === 7 ? 1.0 : -1.0;
  1682. // 3D
  1683. Cartesian2.Cartesian3.pack(encodedStart.high, startHiAndForwardOffsetX, vec4Index);
  1684. startHiAndForwardOffsetX[wIndex] = forwardOffset.x;
  1685. Cartesian2.Cartesian3.pack(encodedStart.low, startLoAndForwardOffsetY, vec4Index);
  1686. startLoAndForwardOffsetY[wIndex] = forwardOffset.y;
  1687. Cartesian2.Cartesian3.pack(
  1688. startPlaneNormal,
  1689. startNormalAndForwardOffsetZ,
  1690. vec4Index
  1691. );
  1692. startNormalAndForwardOffsetZ[wIndex] = forwardOffset.z;
  1693. Cartesian2.Cartesian3.pack(
  1694. endPlaneNormal,
  1695. endNormalAndTextureCoordinateNormalizationX,
  1696. vec4Index
  1697. );
  1698. endNormalAndTextureCoordinateNormalizationX[wIndex] =
  1699. texcoordNormalization3DX * rightPlaneSide;
  1700. Cartesian2.Cartesian3.pack(
  1701. rightNormal,
  1702. rightNormalAndTextureCoordinateNormalizationY,
  1703. vec4Index
  1704. );
  1705. var texcoordNormalization = texcoordNormalization3DY * topBottomSide;
  1706. if (texcoordNormalization === 0.0 && topBottomSide < 0.0) {
  1707. texcoordNormalization = 9.0; // some value greater than 1.0
  1708. }
  1709. rightNormalAndTextureCoordinateNormalizationY[
  1710. wIndex
  1711. ] = texcoordNormalization;
  1712. // 2D
  1713. if (compute2dAttributes) {
  1714. startHiLo2D[vec4Index] = encodedStart2D.high.x;
  1715. startHiLo2D[vec4Index + 1] = encodedStart2D.high.y;
  1716. startHiLo2D[vec4Index + 2] = encodedStart2D.low.x;
  1717. startHiLo2D[vec4Index + 3] = encodedStart2D.low.y;
  1718. startEndNormals2D[vec4Index] = -startGeometryNormal2D.y;
  1719. startEndNormals2D[vec4Index + 1] = startGeometryNormal2D.x;
  1720. startEndNormals2D[vec4Index + 2] = endGeometryNormal2D.y;
  1721. startEndNormals2D[vec4Index + 3] = -endGeometryNormal2D.x;
  1722. offsetAndRight2D[vec4Index] = forwardOffset2D.x;
  1723. offsetAndRight2D[vec4Index + 1] = forwardOffset2D.y;
  1724. offsetAndRight2D[vec4Index + 2] = right2D.x;
  1725. offsetAndRight2D[vec4Index + 3] = right2D.y;
  1726. texcoordNormalization2D[vec2Index] =
  1727. texcoordNormalization2DX * rightPlaneSide;
  1728. texcoordNormalization = texcoordNormalization2DY * topBottomSide;
  1729. if (texcoordNormalization === 0.0 && topBottomSide < 0.0) {
  1730. texcoordNormalization = 9.0; // some value greater than 1.0
  1731. }
  1732. texcoordNormalization2D[vec2Index + 1] = texcoordNormalization;
  1733. }
  1734. }
  1735. // Adjust height of volume in 3D
  1736. var adjustHeightStartBottom = adjustHeightStartBottomScratch;
  1737. var adjustHeightEndBottom = adjustHeightEndBottomScratch;
  1738. var adjustHeightStartTop = adjustHeightStartTopScratch;
  1739. var adjustHeightEndTop = adjustHeightEndTopScratch;
  1740. var getHeightsRectangle = Cartesian2.Rectangle.fromCartographicArray(
  1741. getHeightCartographics,
  1742. getHeightRectangleScratch
  1743. );
  1744. var minMaxHeights = ApproximateTerrainHeights.getMinimumMaximumHeights(
  1745. getHeightsRectangle,
  1746. ellipsoid
  1747. );
  1748. var minHeight = minMaxHeights.minimumTerrainHeight;
  1749. var maxHeight = minMaxHeights.maximumTerrainHeight;
  1750. sumHeights += minHeight;
  1751. sumHeights += maxHeight;
  1752. adjustHeights(
  1753. startBottom,
  1754. startTop,
  1755. minHeight,
  1756. maxHeight,
  1757. adjustHeightStartBottom,
  1758. adjustHeightStartTop
  1759. );
  1760. adjustHeights(
  1761. endBottom,
  1762. endTop,
  1763. minHeight,
  1764. maxHeight,
  1765. adjustHeightEndBottom,
  1766. adjustHeightEndTop
  1767. );
  1768. // Nudge the positions away from the "polyline" a little bit to prevent errors in GeometryPipeline
  1769. var normalNudge = Cartesian2.Cartesian3.multiplyByScalar(
  1770. rightNormal,
  1771. _Math.CesiumMath.EPSILON5,
  1772. normalNudgeScratch
  1773. );
  1774. Cartesian2.Cartesian3.add(
  1775. adjustHeightStartBottom,
  1776. normalNudge,
  1777. adjustHeightStartBottom
  1778. );
  1779. Cartesian2.Cartesian3.add(adjustHeightEndBottom, normalNudge, adjustHeightEndBottom);
  1780. Cartesian2.Cartesian3.add(adjustHeightStartTop, normalNudge, adjustHeightStartTop);
  1781. Cartesian2.Cartesian3.add(adjustHeightEndTop, normalNudge, adjustHeightEndTop);
  1782. // If the segment is very close to the XZ plane, nudge the vertices slightly to avoid touching it.
  1783. nudgeXZ(adjustHeightStartBottom, adjustHeightEndBottom);
  1784. nudgeXZ(adjustHeightStartTop, adjustHeightEndTop);
  1785. Cartesian2.Cartesian3.pack(adjustHeightStartBottom, positionsArray, vec3sWriteIndex);
  1786. Cartesian2.Cartesian3.pack(adjustHeightEndBottom, positionsArray, vec3sWriteIndex + 3);
  1787. Cartesian2.Cartesian3.pack(adjustHeightEndTop, positionsArray, vec3sWriteIndex + 6);
  1788. Cartesian2.Cartesian3.pack(adjustHeightStartTop, positionsArray, vec3sWriteIndex + 9);
  1789. normalNudge = Cartesian2.Cartesian3.multiplyByScalar(
  1790. rightNormal,
  1791. -2.0 * _Math.CesiumMath.EPSILON5,
  1792. normalNudgeScratch
  1793. );
  1794. Cartesian2.Cartesian3.add(
  1795. adjustHeightStartBottom,
  1796. normalNudge,
  1797. adjustHeightStartBottom
  1798. );
  1799. Cartesian2.Cartesian3.add(adjustHeightEndBottom, normalNudge, adjustHeightEndBottom);
  1800. Cartesian2.Cartesian3.add(adjustHeightStartTop, normalNudge, adjustHeightStartTop);
  1801. Cartesian2.Cartesian3.add(adjustHeightEndTop, normalNudge, adjustHeightEndTop);
  1802. nudgeXZ(adjustHeightStartBottom, adjustHeightEndBottom);
  1803. nudgeXZ(adjustHeightStartTop, adjustHeightEndTop);
  1804. Cartesian2.Cartesian3.pack(
  1805. adjustHeightStartBottom,
  1806. positionsArray,
  1807. vec3sWriteIndex + 12
  1808. );
  1809. Cartesian2.Cartesian3.pack(
  1810. adjustHeightEndBottom,
  1811. positionsArray,
  1812. vec3sWriteIndex + 15
  1813. );
  1814. Cartesian2.Cartesian3.pack(adjustHeightEndTop, positionsArray, vec3sWriteIndex + 18);
  1815. Cartesian2.Cartesian3.pack(adjustHeightStartTop, positionsArray, vec3sWriteIndex + 21);
  1816. cartographicsIndex += 2;
  1817. index += 3;
  1818. vec2sWriteIndex += 16;
  1819. vec3sWriteIndex += 24;
  1820. vec4sWriteIndex += 32;
  1821. lengthSoFar3D += segmentLength3D;
  1822. lengthSoFar2D += segmentLength2D;
  1823. }
  1824. index = 0;
  1825. var indexOffset = 0;
  1826. for (i = 0; i < segmentCount; i++) {
  1827. for (j = 0; j < REFERENCE_INDICES_LENGTH; j++) {
  1828. indices[index + j] = REFERENCE_INDICES[j] + indexOffset;
  1829. }
  1830. indexOffset += 8;
  1831. index += REFERENCE_INDICES_LENGTH;
  1832. }
  1833. var boundingSpheres = scratchBoundingSpheres;
  1834. Transforms.BoundingSphere.fromVertices(
  1835. bottomPositionsArray,
  1836. Cartesian2.Cartesian3.ZERO,
  1837. 3,
  1838. boundingSpheres[0]
  1839. );
  1840. Transforms.BoundingSphere.fromVertices(
  1841. topPositionsArray,
  1842. Cartesian2.Cartesian3.ZERO,
  1843. 3,
  1844. boundingSpheres[1]
  1845. );
  1846. var boundingSphere = Transforms.BoundingSphere.fromBoundingSpheres(boundingSpheres);
  1847. // Adjust bounding sphere height and radius to cover more of the volume
  1848. boundingSphere.radius += sumHeights / (segmentCount * 2.0);
  1849. var attributes = {
  1850. position: new GeometryAttribute.GeometryAttribute({
  1851. componentDatatype: ComponentDatatype.ComponentDatatype.DOUBLE,
  1852. componentsPerAttribute: 3,
  1853. normalize: false,
  1854. values: positionsArray,
  1855. }),
  1856. startHiAndForwardOffsetX: getVec4GeometryAttribute(
  1857. startHiAndForwardOffsetX
  1858. ),
  1859. startLoAndForwardOffsetY: getVec4GeometryAttribute(
  1860. startLoAndForwardOffsetY
  1861. ),
  1862. startNormalAndForwardOffsetZ: getVec4GeometryAttribute(
  1863. startNormalAndForwardOffsetZ
  1864. ),
  1865. endNormalAndTextureCoordinateNormalizationX: getVec4GeometryAttribute(
  1866. endNormalAndTextureCoordinateNormalizationX
  1867. ),
  1868. rightNormalAndTextureCoordinateNormalizationY: getVec4GeometryAttribute(
  1869. rightNormalAndTextureCoordinateNormalizationY
  1870. ),
  1871. };
  1872. if (compute2dAttributes) {
  1873. attributes.startHiLo2D = getVec4GeometryAttribute(startHiLo2D);
  1874. attributes.offsetAndRight2D = getVec4GeometryAttribute(offsetAndRight2D);
  1875. attributes.startEndNormals2D = getVec4GeometryAttribute(startEndNormals2D);
  1876. attributes.texcoordNormalization2D = new GeometryAttribute.GeometryAttribute({
  1877. componentDatatype: ComponentDatatype.ComponentDatatype.FLOAT,
  1878. componentsPerAttribute: 2,
  1879. normalize: false,
  1880. values: texcoordNormalization2D,
  1881. });
  1882. }
  1883. return new GeometryAttribute.Geometry({
  1884. attributes: attributes,
  1885. indices: indices,
  1886. boundingSphere: boundingSphere,
  1887. });
  1888. }
  1889. function getVec4GeometryAttribute(typedArray) {
  1890. return new GeometryAttribute.GeometryAttribute({
  1891. componentDatatype: ComponentDatatype.ComponentDatatype.FLOAT,
  1892. componentsPerAttribute: 4,
  1893. normalize: false,
  1894. values: typedArray,
  1895. });
  1896. }
  1897. /**
  1898. * Approximates an ellipsoid-tangent vector in 2D by projecting the end point into 2D.
  1899. * Exposed for testing.
  1900. *
  1901. * @param {MapProjection} projection Map Projection for projecting coordinates to 2D.
  1902. * @param {Cartographic} cartographic The cartographic origin point of the normal.
  1903. * Used to check if the normal crosses the IDL during projection.
  1904. * @param {Cartesian3} normal The normal in 3D.
  1905. * @param {Cartesian3} projectedPosition The projected origin point of the normal in 2D.
  1906. * @param {Cartesian3} result Result parameter on which to store the projected normal.
  1907. * @private
  1908. */
  1909. GroundPolylineGeometry._projectNormal = projectNormal;
  1910. function createGroundPolylineGeometry(groundPolylineGeometry, offset) {
  1911. return ApproximateTerrainHeights.initialize().then(function () {
  1912. if (when.defined(offset)) {
  1913. groundPolylineGeometry = GroundPolylineGeometry.unpack(
  1914. groundPolylineGeometry,
  1915. offset
  1916. );
  1917. }
  1918. return GroundPolylineGeometry.createGeometry(groundPolylineGeometry);
  1919. });
  1920. }
  1921. return createGroundPolylineGeometry;
  1922. });
  1923. //# sourceMappingURL=createGroundPolylineGeometry.js.map