EllipsoidRhumbLine-5f1492e5.js 23 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750
  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(['exports', './when-54c2dc71', './Check-6c0211bc', './Math-1124a290', './Cartesian2-33d2657c'], function (exports, when, Check, _Math, Cartesian2) { 'use strict';
  24. function calculateM(ellipticity, major, latitude) {
  25. if (ellipticity === 0.0) {
  26. // sphere
  27. return major * latitude;
  28. }
  29. var e2 = ellipticity * ellipticity;
  30. var e4 = e2 * e2;
  31. var e6 = e4 * e2;
  32. var e8 = e6 * e2;
  33. var e10 = e8 * e2;
  34. var e12 = e10 * e2;
  35. var phi = latitude;
  36. var sin2Phi = Math.sin(2 * phi);
  37. var sin4Phi = Math.sin(4 * phi);
  38. var sin6Phi = Math.sin(6 * phi);
  39. var sin8Phi = Math.sin(8 * phi);
  40. var sin10Phi = Math.sin(10 * phi);
  41. var sin12Phi = Math.sin(12 * phi);
  42. return (
  43. major *
  44. ((1 -
  45. e2 / 4 -
  46. (3 * e4) / 64 -
  47. (5 * e6) / 256 -
  48. (175 * e8) / 16384 -
  49. (441 * e10) / 65536 -
  50. (4851 * e12) / 1048576) *
  51. phi -
  52. ((3 * e2) / 8 +
  53. (3 * e4) / 32 +
  54. (45 * e6) / 1024 +
  55. (105 * e8) / 4096 +
  56. (2205 * e10) / 131072 +
  57. (6237 * e12) / 524288) *
  58. sin2Phi +
  59. ((15 * e4) / 256 +
  60. (45 * e6) / 1024 +
  61. (525 * e8) / 16384 +
  62. (1575 * e10) / 65536 +
  63. (155925 * e12) / 8388608) *
  64. sin4Phi -
  65. ((35 * e6) / 3072 +
  66. (175 * e8) / 12288 +
  67. (3675 * e10) / 262144 +
  68. (13475 * e12) / 1048576) *
  69. sin6Phi +
  70. ((315 * e8) / 131072 + (2205 * e10) / 524288 + (43659 * e12) / 8388608) *
  71. sin8Phi -
  72. ((693 * e10) / 1310720 + (6237 * e12) / 5242880) * sin10Phi +
  73. ((1001 * e12) / 8388608) * sin12Phi)
  74. );
  75. }
  76. function calculateInverseM(M, ellipticity, major) {
  77. var d = M / major;
  78. if (ellipticity === 0.0) {
  79. // sphere
  80. return d;
  81. }
  82. var d2 = d * d;
  83. var d3 = d2 * d;
  84. var d4 = d3 * d;
  85. var e = ellipticity;
  86. var e2 = e * e;
  87. var e4 = e2 * e2;
  88. var e6 = e4 * e2;
  89. var e8 = e6 * e2;
  90. var e10 = e8 * e2;
  91. var e12 = e10 * e2;
  92. var sin2D = Math.sin(2 * d);
  93. var cos2D = Math.cos(2 * d);
  94. var sin4D = Math.sin(4 * d);
  95. var cos4D = Math.cos(4 * d);
  96. var sin6D = Math.sin(6 * d);
  97. var cos6D = Math.cos(6 * d);
  98. var sin8D = Math.sin(8 * d);
  99. var cos8D = Math.cos(8 * d);
  100. var sin10D = Math.sin(10 * d);
  101. var cos10D = Math.cos(10 * d);
  102. var sin12D = Math.sin(12 * d);
  103. return (
  104. d +
  105. (d * e2) / 4 +
  106. (7 * d * e4) / 64 +
  107. (15 * d * e6) / 256 +
  108. (579 * d * e8) / 16384 +
  109. (1515 * d * e10) / 65536 +
  110. (16837 * d * e12) / 1048576 +
  111. ((3 * d * e4) / 16 +
  112. (45 * d * e6) / 256 -
  113. (d * (32 * d2 - 561) * e8) / 4096 -
  114. (d * (232 * d2 - 1677) * e10) / 16384 +
  115. (d * (399985 - 90560 * d2 + 512 * d4) * e12) / 5242880) *
  116. cos2D +
  117. ((21 * d * e6) / 256 +
  118. (483 * d * e8) / 4096 -
  119. (d * (224 * d2 - 1969) * e10) / 16384 -
  120. (d * (33152 * d2 - 112599) * e12) / 1048576) *
  121. cos4D +
  122. ((151 * d * e8) / 4096 +
  123. (4681 * d * e10) / 65536 +
  124. (1479 * d * e12) / 16384 -
  125. (453 * d3 * e12) / 32768) *
  126. cos6D +
  127. ((1097 * d * e10) / 65536 + (42783 * d * e12) / 1048576) * cos8D +
  128. ((8011 * d * e12) / 1048576) * cos10D +
  129. ((3 * e2) / 8 +
  130. (3 * e4) / 16 +
  131. (213 * e6) / 2048 -
  132. (3 * d2 * e6) / 64 +
  133. (255 * e8) / 4096 -
  134. (33 * d2 * e8) / 512 +
  135. (20861 * e10) / 524288 -
  136. (33 * d2 * e10) / 512 +
  137. (d4 * e10) / 1024 +
  138. (28273 * e12) / 1048576 -
  139. (471 * d2 * e12) / 8192 +
  140. (9 * d4 * e12) / 4096) *
  141. sin2D +
  142. ((21 * e4) / 256 +
  143. (21 * e6) / 256 +
  144. (533 * e8) / 8192 -
  145. (21 * d2 * e8) / 512 +
  146. (197 * e10) / 4096 -
  147. (315 * d2 * e10) / 4096 +
  148. (584039 * e12) / 16777216 -
  149. (12517 * d2 * e12) / 131072 +
  150. (7 * d4 * e12) / 2048) *
  151. sin4D +
  152. ((151 * e6) / 6144 +
  153. (151 * e8) / 4096 +
  154. (5019 * e10) / 131072 -
  155. (453 * d2 * e10) / 16384 +
  156. (26965 * e12) / 786432 -
  157. (8607 * d2 * e12) / 131072) *
  158. sin6D +
  159. ((1097 * e8) / 131072 +
  160. (1097 * e10) / 65536 +
  161. (225797 * e12) / 10485760 -
  162. (1097 * d2 * e12) / 65536) *
  163. sin8D +
  164. ((8011 * e10) / 2621440 + (8011 * e12) / 1048576) * sin10D +
  165. ((293393 * e12) / 251658240) * sin12D
  166. );
  167. }
  168. function calculateSigma(ellipticity, latitude) {
  169. if (ellipticity === 0.0) {
  170. // sphere
  171. return Math.log(Math.tan(0.5 * (_Math.CesiumMath.PI_OVER_TWO + latitude)));
  172. }
  173. var eSinL = ellipticity * Math.sin(latitude);
  174. return (
  175. Math.log(Math.tan(0.5 * (_Math.CesiumMath.PI_OVER_TWO + latitude))) -
  176. (ellipticity / 2.0) * Math.log((1 + eSinL) / (1 - eSinL))
  177. );
  178. }
  179. function calculateHeading(
  180. ellipsoidRhumbLine,
  181. firstLongitude,
  182. firstLatitude,
  183. secondLongitude,
  184. secondLatitude
  185. ) {
  186. var sigma1 = calculateSigma(ellipsoidRhumbLine._ellipticity, firstLatitude);
  187. var sigma2 = calculateSigma(ellipsoidRhumbLine._ellipticity, secondLatitude);
  188. return Math.atan2(
  189. _Math.CesiumMath.negativePiToPi(secondLongitude - firstLongitude),
  190. sigma2 - sigma1
  191. );
  192. }
  193. function calculateArcLength(
  194. ellipsoidRhumbLine,
  195. major,
  196. minor,
  197. firstLongitude,
  198. firstLatitude,
  199. secondLongitude,
  200. secondLatitude
  201. ) {
  202. var heading = ellipsoidRhumbLine._heading;
  203. var deltaLongitude = secondLongitude - firstLongitude;
  204. var distance = 0.0;
  205. //Check to see if the rhumb line has constant latitude
  206. //This equation will diverge if heading gets close to 90 degrees
  207. if (
  208. _Math.CesiumMath.equalsEpsilon(
  209. Math.abs(heading),
  210. _Math.CesiumMath.PI_OVER_TWO,
  211. _Math.CesiumMath.EPSILON8
  212. )
  213. ) {
  214. //If heading is close to 90 degrees
  215. if (major === minor) {
  216. distance =
  217. major *
  218. Math.cos(firstLatitude) *
  219. _Math.CesiumMath.negativePiToPi(deltaLongitude);
  220. } else {
  221. var sinPhi = Math.sin(firstLatitude);
  222. distance =
  223. (major *
  224. Math.cos(firstLatitude) *
  225. _Math.CesiumMath.negativePiToPi(deltaLongitude)) /
  226. Math.sqrt(1 - ellipsoidRhumbLine._ellipticitySquared * sinPhi * sinPhi);
  227. }
  228. } else {
  229. var M1 = calculateM(ellipsoidRhumbLine._ellipticity, major, firstLatitude);
  230. var M2 = calculateM(ellipsoidRhumbLine._ellipticity, major, secondLatitude);
  231. distance = (M2 - M1) / Math.cos(heading);
  232. }
  233. return Math.abs(distance);
  234. }
  235. var scratchCart1 = new Cartesian2.Cartesian3();
  236. var scratchCart2 = new Cartesian2.Cartesian3();
  237. function computeProperties(ellipsoidRhumbLine, start, end, ellipsoid) {
  238. var firstCartesian = Cartesian2.Cartesian3.normalize(
  239. ellipsoid.cartographicToCartesian(start, scratchCart2),
  240. scratchCart1
  241. );
  242. var lastCartesian = Cartesian2.Cartesian3.normalize(
  243. ellipsoid.cartographicToCartesian(end, scratchCart2),
  244. scratchCart2
  245. );
  246. //>>includeStart('debug', pragmas.debug);
  247. Check.Check.typeOf.number.greaterThanOrEquals(
  248. "value",
  249. Math.abs(
  250. Math.abs(Cartesian2.Cartesian3.angleBetween(firstCartesian, lastCartesian)) - Math.PI
  251. ),
  252. 0.0125
  253. );
  254. //>>includeEnd('debug');
  255. var major = ellipsoid.maximumRadius;
  256. var minor = ellipsoid.minimumRadius;
  257. var majorSquared = major * major;
  258. var minorSquared = minor * minor;
  259. ellipsoidRhumbLine._ellipticitySquared =
  260. (majorSquared - minorSquared) / majorSquared;
  261. ellipsoidRhumbLine._ellipticity = Math.sqrt(
  262. ellipsoidRhumbLine._ellipticitySquared
  263. );
  264. ellipsoidRhumbLine._start = Cartesian2.Cartographic.clone(
  265. start,
  266. ellipsoidRhumbLine._start
  267. );
  268. ellipsoidRhumbLine._start.height = 0;
  269. ellipsoidRhumbLine._end = Cartesian2.Cartographic.clone(end, ellipsoidRhumbLine._end);
  270. ellipsoidRhumbLine._end.height = 0;
  271. ellipsoidRhumbLine._heading = calculateHeading(
  272. ellipsoidRhumbLine,
  273. start.longitude,
  274. start.latitude,
  275. end.longitude,
  276. end.latitude
  277. );
  278. ellipsoidRhumbLine._distance = calculateArcLength(
  279. ellipsoidRhumbLine,
  280. ellipsoid.maximumRadius,
  281. ellipsoid.minimumRadius,
  282. start.longitude,
  283. start.latitude,
  284. end.longitude,
  285. end.latitude
  286. );
  287. }
  288. function interpolateUsingSurfaceDistance(
  289. start,
  290. heading,
  291. distance,
  292. major,
  293. ellipticity,
  294. result
  295. ) {
  296. var ellipticitySquared = ellipticity * ellipticity;
  297. var longitude;
  298. var latitude;
  299. var deltaLongitude;
  300. //Check to see if the rhumb line has constant latitude
  301. //This won't converge if heading is close to 90 degrees
  302. if (
  303. Math.abs(_Math.CesiumMath.PI_OVER_TWO - Math.abs(heading)) > _Math.CesiumMath.EPSILON8
  304. ) {
  305. //Calculate latitude of the second point
  306. var M1 = calculateM(ellipticity, major, start.latitude);
  307. var deltaM = distance * Math.cos(heading);
  308. var M2 = M1 + deltaM;
  309. latitude = calculateInverseM(M2, ellipticity, major);
  310. //Now find the longitude of the second point
  311. var sigma1 = calculateSigma(ellipticity, start.latitude);
  312. var sigma2 = calculateSigma(ellipticity, latitude);
  313. deltaLongitude = Math.tan(heading) * (sigma2 - sigma1);
  314. longitude = _Math.CesiumMath.negativePiToPi(start.longitude + deltaLongitude);
  315. } else {
  316. //If heading is close to 90 degrees
  317. latitude = start.latitude;
  318. var localRad;
  319. if (ellipticity === 0.0) {
  320. // sphere
  321. localRad = major * Math.cos(start.latitude);
  322. } else {
  323. var sinPhi = Math.sin(start.latitude);
  324. localRad =
  325. (major * Math.cos(start.latitude)) /
  326. Math.sqrt(1 - ellipticitySquared * sinPhi * sinPhi);
  327. }
  328. deltaLongitude = distance / localRad;
  329. if (heading > 0.0) {
  330. longitude = _Math.CesiumMath.negativePiToPi(start.longitude + deltaLongitude);
  331. } else {
  332. longitude = _Math.CesiumMath.negativePiToPi(start.longitude - deltaLongitude);
  333. }
  334. }
  335. if (when.defined(result)) {
  336. result.longitude = longitude;
  337. result.latitude = latitude;
  338. result.height = 0;
  339. return result;
  340. }
  341. return new Cartesian2.Cartographic(longitude, latitude, 0);
  342. }
  343. /**
  344. * Initializes a rhumb line on the ellipsoid connecting the two provided planetodetic points.
  345. *
  346. * @alias EllipsoidRhumbLine
  347. * @constructor
  348. *
  349. * @param {Cartographic} [start] The initial planetodetic point on the path.
  350. * @param {Cartographic} [end] The final planetodetic point on the path.
  351. * @param {Ellipsoid} [ellipsoid=Ellipsoid.WGS84] The ellipsoid on which the rhumb line lies.
  352. *
  353. * @exception {DeveloperError} angle between start and end must be at least 0.0125 radians.
  354. */
  355. function EllipsoidRhumbLine(start, end, ellipsoid) {
  356. var e = when.defaultValue(ellipsoid, Cartesian2.Ellipsoid.WGS84);
  357. this._ellipsoid = e;
  358. this._start = new Cartesian2.Cartographic();
  359. this._end = new Cartesian2.Cartographic();
  360. this._heading = undefined;
  361. this._distance = undefined;
  362. this._ellipticity = undefined;
  363. this._ellipticitySquared = undefined;
  364. if (when.defined(start) && when.defined(end)) {
  365. computeProperties(this, start, end, e);
  366. }
  367. }
  368. Object.defineProperties(EllipsoidRhumbLine.prototype, {
  369. /**
  370. * Gets the ellipsoid.
  371. * @memberof EllipsoidRhumbLine.prototype
  372. * @type {Ellipsoid}
  373. * @readonly
  374. */
  375. ellipsoid: {
  376. get: function () {
  377. return this._ellipsoid;
  378. },
  379. },
  380. /**
  381. * Gets the surface distance between the start and end point
  382. * @memberof EllipsoidRhumbLine.prototype
  383. * @type {Number}
  384. * @readonly
  385. */
  386. surfaceDistance: {
  387. get: function () {
  388. //>>includeStart('debug', pragmas.debug);
  389. Check.Check.defined("distance", this._distance);
  390. //>>includeEnd('debug');
  391. return this._distance;
  392. },
  393. },
  394. /**
  395. * Gets the initial planetodetic point on the path.
  396. * @memberof EllipsoidRhumbLine.prototype
  397. * @type {Cartographic}
  398. * @readonly
  399. */
  400. start: {
  401. get: function () {
  402. return this._start;
  403. },
  404. },
  405. /**
  406. * Gets the final planetodetic point on the path.
  407. * @memberof EllipsoidRhumbLine.prototype
  408. * @type {Cartographic}
  409. * @readonly
  410. */
  411. end: {
  412. get: function () {
  413. return this._end;
  414. },
  415. },
  416. /**
  417. * Gets the heading from the start point to the end point.
  418. * @memberof EllipsoidRhumbLine.prototype
  419. * @type {Number}
  420. * @readonly
  421. */
  422. heading: {
  423. get: function () {
  424. //>>includeStart('debug', pragmas.debug);
  425. Check.Check.defined("distance", this._distance);
  426. //>>includeEnd('debug');
  427. return this._heading;
  428. },
  429. },
  430. });
  431. /**
  432. * Create a rhumb line using an initial position with a heading and distance.
  433. *
  434. * @param {Cartographic} start The initial planetodetic point on the path.
  435. * @param {Number} heading The heading in radians.
  436. * @param {Number} distance The rhumb line distance between the start and end point.
  437. * @param {Ellipsoid} [ellipsoid=Ellipsoid.WGS84] The ellipsoid on which the rhumb line lies.
  438. * @param {EllipsoidRhumbLine} [result] The object in which to store the result.
  439. * @returns {EllipsoidRhumbLine} The EllipsoidRhumbLine object.
  440. */
  441. EllipsoidRhumbLine.fromStartHeadingDistance = function (
  442. start,
  443. heading,
  444. distance,
  445. ellipsoid,
  446. result
  447. ) {
  448. //>>includeStart('debug', pragmas.debug);
  449. Check.Check.defined("start", start);
  450. Check.Check.defined("heading", heading);
  451. Check.Check.defined("distance", distance);
  452. Check.Check.typeOf.number.greaterThan("distance", distance, 0.0);
  453. //>>includeEnd('debug');
  454. var e = when.defaultValue(ellipsoid, Cartesian2.Ellipsoid.WGS84);
  455. var major = e.maximumRadius;
  456. var minor = e.minimumRadius;
  457. var majorSquared = major * major;
  458. var minorSquared = minor * minor;
  459. var ellipticity = Math.sqrt((majorSquared - minorSquared) / majorSquared);
  460. heading = _Math.CesiumMath.negativePiToPi(heading);
  461. var end = interpolateUsingSurfaceDistance(
  462. start,
  463. heading,
  464. distance,
  465. e.maximumRadius,
  466. ellipticity
  467. );
  468. if (
  469. !when.defined(result) ||
  470. (when.defined(ellipsoid) && !ellipsoid.equals(result.ellipsoid))
  471. ) {
  472. return new EllipsoidRhumbLine(start, end, e);
  473. }
  474. result.setEndPoints(start, end);
  475. return result;
  476. };
  477. /**
  478. * Sets the start and end points of the rhumb line.
  479. *
  480. * @param {Cartographic} start The initial planetodetic point on the path.
  481. * @param {Cartographic} end The final planetodetic point on the path.
  482. */
  483. EllipsoidRhumbLine.prototype.setEndPoints = function (start, end) {
  484. //>>includeStart('debug', pragmas.debug);
  485. Check.Check.defined("start", start);
  486. Check.Check.defined("end", end);
  487. //>>includeEnd('debug');
  488. computeProperties(this, start, end, this._ellipsoid);
  489. };
  490. /**
  491. * Provides the location of a point at the indicated portion along the rhumb line.
  492. *
  493. * @param {Number} fraction The portion of the distance between the initial and final points.
  494. * @param {Cartographic} [result] The object in which to store the result.
  495. * @returns {Cartographic} The location of the point along the rhumb line.
  496. */
  497. EllipsoidRhumbLine.prototype.interpolateUsingFraction = function (
  498. fraction,
  499. result
  500. ) {
  501. return this.interpolateUsingSurfaceDistance(
  502. fraction * this._distance,
  503. result
  504. );
  505. };
  506. /**
  507. * Provides the location of a point at the indicated distance along the rhumb line.
  508. *
  509. * @param {Number} distance The distance from the inital point to the point of interest along the rhumbLine.
  510. * @param {Cartographic} [result] The object in which to store the result.
  511. * @returns {Cartographic} The location of the point along the rhumb line.
  512. *
  513. * @exception {DeveloperError} start and end must be set before calling function interpolateUsingSurfaceDistance
  514. */
  515. EllipsoidRhumbLine.prototype.interpolateUsingSurfaceDistance = function (
  516. distance,
  517. result
  518. ) {
  519. //>>includeStart('debug', pragmas.debug);
  520. Check.Check.typeOf.number("distance", distance);
  521. if (!when.defined(this._distance) || this._distance === 0.0) {
  522. throw new Check.DeveloperError(
  523. "EllipsoidRhumbLine must have distinct start and end set."
  524. );
  525. }
  526. //>>includeEnd('debug');
  527. return interpolateUsingSurfaceDistance(
  528. this._start,
  529. this._heading,
  530. distance,
  531. this._ellipsoid.maximumRadius,
  532. this._ellipticity,
  533. result
  534. );
  535. };
  536. /**
  537. * Provides the location of a point at the indicated longitude along the rhumb line.
  538. * If the longitude is outside the range of start and end points, the first intersection with the longitude from the start point in the direction of the heading is returned. This follows the spiral property of a rhumb line.
  539. *
  540. * @param {Number} intersectionLongitude The longitude, in radians, at which to find the intersection point from the starting point using the heading.
  541. * @param {Cartographic} [result] The object in which to store the result.
  542. * @returns {Cartographic} The location of the intersection point along the rhumb line, undefined if there is no intersection or infinite intersections.
  543. *
  544. * @exception {DeveloperError} start and end must be set before calling function findIntersectionWithLongitude.
  545. */
  546. EllipsoidRhumbLine.prototype.findIntersectionWithLongitude = function (
  547. intersectionLongitude,
  548. result
  549. ) {
  550. //>>includeStart('debug', pragmas.debug);
  551. Check.Check.typeOf.number("intersectionLongitude", intersectionLongitude);
  552. if (!when.defined(this._distance) || this._distance === 0.0) {
  553. throw new Check.DeveloperError(
  554. "EllipsoidRhumbLine must have distinct start and end set."
  555. );
  556. }
  557. //>>includeEnd('debug');
  558. var ellipticity = this._ellipticity;
  559. var heading = this._heading;
  560. var absHeading = Math.abs(heading);
  561. var start = this._start;
  562. intersectionLongitude = _Math.CesiumMath.negativePiToPi(intersectionLongitude);
  563. if (
  564. _Math.CesiumMath.equalsEpsilon(
  565. Math.abs(intersectionLongitude),
  566. Math.PI,
  567. _Math.CesiumMath.EPSILON14
  568. )
  569. ) {
  570. intersectionLongitude = _Math.CesiumMath.sign(start.longitude) * Math.PI;
  571. }
  572. if (!when.defined(result)) {
  573. result = new Cartesian2.Cartographic();
  574. }
  575. // If heading is -PI/2 or PI/2, this is an E-W rhumb line
  576. // If heading is 0 or PI, this is an N-S rhumb line
  577. if (Math.abs(_Math.CesiumMath.PI_OVER_TWO - absHeading) <= _Math.CesiumMath.EPSILON8) {
  578. result.longitude = intersectionLongitude;
  579. result.latitude = start.latitude;
  580. result.height = 0;
  581. return result;
  582. } else if (
  583. _Math.CesiumMath.equalsEpsilon(
  584. Math.abs(_Math.CesiumMath.PI_OVER_TWO - absHeading),
  585. _Math.CesiumMath.PI_OVER_TWO,
  586. _Math.CesiumMath.EPSILON8
  587. )
  588. ) {
  589. if (
  590. _Math.CesiumMath.equalsEpsilon(
  591. intersectionLongitude,
  592. start.longitude,
  593. _Math.CesiumMath.EPSILON12
  594. )
  595. ) {
  596. return undefined;
  597. }
  598. result.longitude = intersectionLongitude;
  599. result.latitude =
  600. _Math.CesiumMath.PI_OVER_TWO *
  601. _Math.CesiumMath.sign(_Math.CesiumMath.PI_OVER_TWO - heading);
  602. result.height = 0;
  603. return result;
  604. }
  605. // Use iterative solver from Equation 9 from http://edwilliams.org/ellipsoid/ellipsoid.pdf
  606. var phi1 = start.latitude;
  607. var eSinPhi1 = ellipticity * Math.sin(phi1);
  608. var leftComponent =
  609. Math.tan(0.5 * (_Math.CesiumMath.PI_OVER_TWO + phi1)) *
  610. Math.exp((intersectionLongitude - start.longitude) / Math.tan(heading));
  611. var denominator = (1 + eSinPhi1) / (1 - eSinPhi1);
  612. var newPhi = start.latitude;
  613. var phi;
  614. do {
  615. phi = newPhi;
  616. var eSinPhi = ellipticity * Math.sin(phi);
  617. var numerator = (1 + eSinPhi) / (1 - eSinPhi);
  618. newPhi =
  619. 2 *
  620. Math.atan(
  621. leftComponent * Math.pow(numerator / denominator, ellipticity / 2)
  622. ) -
  623. _Math.CesiumMath.PI_OVER_TWO;
  624. } while (!_Math.CesiumMath.equalsEpsilon(newPhi, phi, _Math.CesiumMath.EPSILON12));
  625. result.longitude = intersectionLongitude;
  626. result.latitude = newPhi;
  627. result.height = 0;
  628. return result;
  629. };
  630. /**
  631. * Provides the location of a point at the indicated latitude along the rhumb line.
  632. * If the latitude is outside the range of start and end points, the first intersection with the latitude from that start point in the direction of the heading is returned. This follows the spiral property of a rhumb line.
  633. *
  634. * @param {Number} intersectionLatitude The latitude, in radians, at which to find the intersection point from the starting point using the heading.
  635. * @param {Cartographic} [result] The object in which to store the result.
  636. * @returns {Cartographic} The location of the intersection point along the rhumb line, undefined if there is no intersection or infinite intersections.
  637. *
  638. * @exception {DeveloperError} start and end must be set before calling function findIntersectionWithLongitude.
  639. */
  640. EllipsoidRhumbLine.prototype.findIntersectionWithLatitude = function (
  641. intersectionLatitude,
  642. result
  643. ) {
  644. //>>includeStart('debug', pragmas.debug);
  645. Check.Check.typeOf.number("intersectionLatitude", intersectionLatitude);
  646. if (!when.defined(this._distance) || this._distance === 0.0) {
  647. throw new Check.DeveloperError(
  648. "EllipsoidRhumbLine must have distinct start and end set."
  649. );
  650. }
  651. //>>includeEnd('debug');
  652. var ellipticity = this._ellipticity;
  653. var heading = this._heading;
  654. var start = this._start;
  655. // If start and end have same latitude, return undefined since it's either no intersection or infinite intersections
  656. if (
  657. _Math.CesiumMath.equalsEpsilon(
  658. Math.abs(heading),
  659. _Math.CesiumMath.PI_OVER_TWO,
  660. _Math.CesiumMath.EPSILON8
  661. )
  662. ) {
  663. return;
  664. }
  665. // Can be solved using the same equations from interpolateUsingSurfaceDistance
  666. var sigma1 = calculateSigma(ellipticity, start.latitude);
  667. var sigma2 = calculateSigma(ellipticity, intersectionLatitude);
  668. var deltaLongitude = Math.tan(heading) * (sigma2 - sigma1);
  669. var longitude = _Math.CesiumMath.negativePiToPi(start.longitude + deltaLongitude);
  670. if (when.defined(result)) {
  671. result.longitude = longitude;
  672. result.latitude = intersectionLatitude;
  673. result.height = 0;
  674. return result;
  675. }
  676. return new Cartesian2.Cartographic(longitude, intersectionLatitude, 0);
  677. };
  678. exports.EllipsoidRhumbLine = EllipsoidRhumbLine;
  679. });
  680. //# sourceMappingURL=EllipsoidRhumbLine-5f1492e5.js.map