OrientedBoundingBox-8897f474.js 38 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046
  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', './Transforms-8be64844', './Plane-fa30fc46', './EllipsoidTangentPlane-ce6e380f'], function (exports, when, Check, _Math, Cartesian2, Transforms, Plane, EllipsoidTangentPlane) { 'use strict';
  24. /**
  25. * Creates an instance of an OrientedBoundingBox.
  26. * An OrientedBoundingBox of some object is a closed and convex cuboid. It can provide a tighter bounding volume than {@link BoundingSphere} or {@link AxisAlignedBoundingBox} in many cases.
  27. * @alias OrientedBoundingBox
  28. * @constructor
  29. *
  30. * @param {Cartesian3} [center=Cartesian3.ZERO] The center of the box.
  31. * @param {Matrix3} [halfAxes=Matrix3.ZERO] The three orthogonal half-axes of the bounding box.
  32. * Equivalently, the transformation matrix, to rotate and scale a 0x0x0
  33. * cube centered at the origin.
  34. *
  35. *
  36. * @example
  37. * // Create an OrientedBoundingBox using a transformation matrix, a position where the box will be translated, and a scale.
  38. * var center = new Cesium.Cartesian3(1.0, 0.0, 0.0);
  39. * var halfAxes = Cesium.Matrix3.fromScale(new Cesium.Cartesian3(1.0, 3.0, 2.0), new Cesium.Matrix3());
  40. *
  41. * var obb = new Cesium.OrientedBoundingBox(center, halfAxes);
  42. *
  43. * @see BoundingSphere
  44. * @see BoundingRectangle
  45. */
  46. function OrientedBoundingBox(center, halfAxes) {
  47. /**
  48. * The center of the box.
  49. * @type {Cartesian3}
  50. * @default {@link Cartesian3.ZERO}
  51. */
  52. this.center = Cartesian2.Cartesian3.clone(when.defaultValue(center, Cartesian2.Cartesian3.ZERO));
  53. /**
  54. * The transformation matrix, to rotate the box to the right position.
  55. * @type {Matrix3}
  56. * @default {@link Matrix3.ZERO}
  57. */
  58. this.halfAxes = Transforms.Matrix3.clone(when.defaultValue(halfAxes, Transforms.Matrix3.ZERO));
  59. }
  60. /**
  61. * The number of elements used to pack the object into an array.
  62. * @type {Number}
  63. */
  64. OrientedBoundingBox.packedLength =
  65. Cartesian2.Cartesian3.packedLength + Transforms.Matrix3.packedLength;
  66. /**
  67. * Stores the provided instance into the provided array.
  68. *
  69. * @param {OrientedBoundingBox} value The value to pack.
  70. * @param {Number[]} array The array to pack into.
  71. * @param {Number} [startingIndex=0] The index into the array at which to start packing the elements.
  72. *
  73. * @returns {Number[]} The array that was packed into
  74. */
  75. OrientedBoundingBox.pack = function (value, array, startingIndex) {
  76. //>>includeStart('debug', pragmas.debug);
  77. Check.Check.typeOf.object("value", value);
  78. Check.Check.defined("array", array);
  79. //>>includeEnd('debug');
  80. startingIndex = when.defaultValue(startingIndex, 0);
  81. Cartesian2.Cartesian3.pack(value.center, array, startingIndex);
  82. Transforms.Matrix3.pack(value.halfAxes, array, startingIndex + Cartesian2.Cartesian3.packedLength);
  83. return array;
  84. };
  85. /**
  86. * Retrieves an instance from a packed array.
  87. *
  88. * @param {Number[]} array The packed array.
  89. * @param {Number} [startingIndex=0] The starting index of the element to be unpacked.
  90. * @param {OrientedBoundingBox} [result] The object into which to store the result.
  91. * @returns {OrientedBoundingBox} The modified result parameter or a new OrientedBoundingBox instance if one was not provided.
  92. */
  93. OrientedBoundingBox.unpack = function (array, startingIndex, result) {
  94. //>>includeStart('debug', pragmas.debug);
  95. Check.Check.defined("array", array);
  96. //>>includeEnd('debug');
  97. startingIndex = when.defaultValue(startingIndex, 0);
  98. if (!when.defined(result)) {
  99. result = new OrientedBoundingBox();
  100. }
  101. Cartesian2.Cartesian3.unpack(array, startingIndex, result.center);
  102. Transforms.Matrix3.unpack(
  103. array,
  104. startingIndex + Cartesian2.Cartesian3.packedLength,
  105. result.halfAxes
  106. );
  107. return result;
  108. };
  109. var scratchCartesian1 = new Cartesian2.Cartesian3();
  110. var scratchCartesian2 = new Cartesian2.Cartesian3();
  111. var scratchCartesian3 = new Cartesian2.Cartesian3();
  112. var scratchCartesian4 = new Cartesian2.Cartesian3();
  113. var scratchCartesian5 = new Cartesian2.Cartesian3();
  114. var scratchCartesian6 = new Cartesian2.Cartesian3();
  115. var scratchCovarianceResult = new Transforms.Matrix3();
  116. var scratchEigenResult = {
  117. unitary: new Transforms.Matrix3(),
  118. diagonal: new Transforms.Matrix3(),
  119. };
  120. /**
  121. * Computes an instance of an OrientedBoundingBox of the given positions.
  122. * This is an implementation of Stefan Gottschalk's Collision Queries using Oriented Bounding Boxes solution (PHD thesis).
  123. * Reference: http://gamma.cs.unc.edu/users/gottschalk/main.pdf
  124. *
  125. * @param {Cartesian3[]} [positions] List of {@link Cartesian3} points that the bounding box will enclose.
  126. * @param {OrientedBoundingBox} [result] The object onto which to store the result.
  127. * @returns {OrientedBoundingBox} The modified result parameter or a new OrientedBoundingBox instance if one was not provided.
  128. *
  129. * @example
  130. * // Compute an object oriented bounding box enclosing two points.
  131. * var box = Cesium.OrientedBoundingBox.fromPoints([new Cesium.Cartesian3(2, 0, 0), new Cesium.Cartesian3(-2, 0, 0)]);
  132. */
  133. OrientedBoundingBox.fromPoints = function (positions, result) {
  134. if (!when.defined(result)) {
  135. result = new OrientedBoundingBox();
  136. }
  137. if (!when.defined(positions) || positions.length === 0) {
  138. result.halfAxes = Transforms.Matrix3.ZERO;
  139. result.center = Cartesian2.Cartesian3.ZERO;
  140. return result;
  141. }
  142. var i;
  143. var length = positions.length;
  144. var meanPoint = Cartesian2.Cartesian3.clone(positions[0], scratchCartesian1);
  145. for (i = 1; i < length; i++) {
  146. Cartesian2.Cartesian3.add(meanPoint, positions[i], meanPoint);
  147. }
  148. var invLength = 1.0 / length;
  149. Cartesian2.Cartesian3.multiplyByScalar(meanPoint, invLength, meanPoint);
  150. var exx = 0.0;
  151. var exy = 0.0;
  152. var exz = 0.0;
  153. var eyy = 0.0;
  154. var eyz = 0.0;
  155. var ezz = 0.0;
  156. var p;
  157. for (i = 0; i < length; i++) {
  158. p = Cartesian2.Cartesian3.subtract(positions[i], meanPoint, scratchCartesian2);
  159. exx += p.x * p.x;
  160. exy += p.x * p.y;
  161. exz += p.x * p.z;
  162. eyy += p.y * p.y;
  163. eyz += p.y * p.z;
  164. ezz += p.z * p.z;
  165. }
  166. exx *= invLength;
  167. exy *= invLength;
  168. exz *= invLength;
  169. eyy *= invLength;
  170. eyz *= invLength;
  171. ezz *= invLength;
  172. var covarianceMatrix = scratchCovarianceResult;
  173. covarianceMatrix[0] = exx;
  174. covarianceMatrix[1] = exy;
  175. covarianceMatrix[2] = exz;
  176. covarianceMatrix[3] = exy;
  177. covarianceMatrix[4] = eyy;
  178. covarianceMatrix[5] = eyz;
  179. covarianceMatrix[6] = exz;
  180. covarianceMatrix[7] = eyz;
  181. covarianceMatrix[8] = ezz;
  182. var eigenDecomposition = Transforms.Matrix3.computeEigenDecomposition(
  183. covarianceMatrix,
  184. scratchEigenResult
  185. );
  186. var rotation = Transforms.Matrix3.clone(eigenDecomposition.unitary, result.halfAxes);
  187. var v1 = Transforms.Matrix3.getColumn(rotation, 0, scratchCartesian4);
  188. var v2 = Transforms.Matrix3.getColumn(rotation, 1, scratchCartesian5);
  189. var v3 = Transforms.Matrix3.getColumn(rotation, 2, scratchCartesian6);
  190. var u1 = -Number.MAX_VALUE;
  191. var u2 = -Number.MAX_VALUE;
  192. var u3 = -Number.MAX_VALUE;
  193. var l1 = Number.MAX_VALUE;
  194. var l2 = Number.MAX_VALUE;
  195. var l3 = Number.MAX_VALUE;
  196. for (i = 0; i < length; i++) {
  197. p = positions[i];
  198. u1 = Math.max(Cartesian2.Cartesian3.dot(v1, p), u1);
  199. u2 = Math.max(Cartesian2.Cartesian3.dot(v2, p), u2);
  200. u3 = Math.max(Cartesian2.Cartesian3.dot(v3, p), u3);
  201. l1 = Math.min(Cartesian2.Cartesian3.dot(v1, p), l1);
  202. l2 = Math.min(Cartesian2.Cartesian3.dot(v2, p), l2);
  203. l3 = Math.min(Cartesian2.Cartesian3.dot(v3, p), l3);
  204. }
  205. v1 = Cartesian2.Cartesian3.multiplyByScalar(v1, 0.5 * (l1 + u1), v1);
  206. v2 = Cartesian2.Cartesian3.multiplyByScalar(v2, 0.5 * (l2 + u2), v2);
  207. v3 = Cartesian2.Cartesian3.multiplyByScalar(v3, 0.5 * (l3 + u3), v3);
  208. var center = Cartesian2.Cartesian3.add(v1, v2, result.center);
  209. Cartesian2.Cartesian3.add(center, v3, center);
  210. var scale = scratchCartesian3;
  211. scale.x = u1 - l1;
  212. scale.y = u2 - l2;
  213. scale.z = u3 - l3;
  214. Cartesian2.Cartesian3.multiplyByScalar(scale, 0.5, scale);
  215. Transforms.Matrix3.multiplyByScale(result.halfAxes, scale, result.halfAxes);
  216. return result;
  217. };
  218. var scratchOffset = new Cartesian2.Cartesian3();
  219. var scratchScale = new Cartesian2.Cartesian3();
  220. function fromPlaneExtents(
  221. planeOrigin,
  222. planeXAxis,
  223. planeYAxis,
  224. planeZAxis,
  225. minimumX,
  226. maximumX,
  227. minimumY,
  228. maximumY,
  229. minimumZ,
  230. maximumZ,
  231. result
  232. ) {
  233. //>>includeStart('debug', pragmas.debug);
  234. if (
  235. !when.defined(minimumX) ||
  236. !when.defined(maximumX) ||
  237. !when.defined(minimumY) ||
  238. !when.defined(maximumY) ||
  239. !when.defined(minimumZ) ||
  240. !when.defined(maximumZ)
  241. ) {
  242. throw new Check.DeveloperError(
  243. "all extents (minimum/maximum X/Y/Z) are required."
  244. );
  245. }
  246. //>>includeEnd('debug');
  247. if (!when.defined(result)) {
  248. result = new OrientedBoundingBox();
  249. }
  250. var halfAxes = result.halfAxes;
  251. Transforms.Matrix3.setColumn(halfAxes, 0, planeXAxis, halfAxes);
  252. Transforms.Matrix3.setColumn(halfAxes, 1, planeYAxis, halfAxes);
  253. Transforms.Matrix3.setColumn(halfAxes, 2, planeZAxis, halfAxes);
  254. var centerOffset = scratchOffset;
  255. centerOffset.x = (minimumX + maximumX) / 2.0;
  256. centerOffset.y = (minimumY + maximumY) / 2.0;
  257. centerOffset.z = (minimumZ + maximumZ) / 2.0;
  258. var scale = scratchScale;
  259. scale.x = (maximumX - minimumX) / 2.0;
  260. scale.y = (maximumY - minimumY) / 2.0;
  261. scale.z = (maximumZ - minimumZ) / 2.0;
  262. var center = result.center;
  263. centerOffset = Transforms.Matrix3.multiplyByVector(halfAxes, centerOffset, centerOffset);
  264. Cartesian2.Cartesian3.add(planeOrigin, centerOffset, center);
  265. Transforms.Matrix3.multiplyByScale(halfAxes, scale, halfAxes);
  266. return result;
  267. }
  268. var scratchRectangleCenterCartographic = new Cartesian2.Cartographic();
  269. var scratchRectangleCenter = new Cartesian2.Cartesian3();
  270. var scratchPerimeterCartographicNC = new Cartesian2.Cartographic();
  271. var scratchPerimeterCartographicNW = new Cartesian2.Cartographic();
  272. var scratchPerimeterCartographicCW = new Cartesian2.Cartographic();
  273. var scratchPerimeterCartographicSW = new Cartesian2.Cartographic();
  274. var scratchPerimeterCartographicSC = new Cartesian2.Cartographic();
  275. var scratchPerimeterCartesianNC = new Cartesian2.Cartesian3();
  276. var scratchPerimeterCartesianNW = new Cartesian2.Cartesian3();
  277. var scratchPerimeterCartesianCW = new Cartesian2.Cartesian3();
  278. var scratchPerimeterCartesianSW = new Cartesian2.Cartesian3();
  279. var scratchPerimeterCartesianSC = new Cartesian2.Cartesian3();
  280. var scratchPerimeterProjectedNC = new Cartesian2.Cartesian2();
  281. var scratchPerimeterProjectedNW = new Cartesian2.Cartesian2();
  282. var scratchPerimeterProjectedCW = new Cartesian2.Cartesian2();
  283. var scratchPerimeterProjectedSW = new Cartesian2.Cartesian2();
  284. var scratchPerimeterProjectedSC = new Cartesian2.Cartesian2();
  285. var scratchPlaneOrigin = new Cartesian2.Cartesian3();
  286. var scratchPlaneNormal = new Cartesian2.Cartesian3();
  287. var scratchPlaneXAxis = new Cartesian2.Cartesian3();
  288. var scratchHorizonCartesian = new Cartesian2.Cartesian3();
  289. var scratchHorizonProjected = new Cartesian2.Cartesian2();
  290. var scratchMaxY = new Cartesian2.Cartesian3();
  291. var scratchMinY = new Cartesian2.Cartesian3();
  292. var scratchZ = new Cartesian2.Cartesian3();
  293. var scratchPlane = new Plane.Plane(Cartesian2.Cartesian3.UNIT_X, 0.0);
  294. /**
  295. * Computes an OrientedBoundingBox that bounds a {@link Rectangle} on the surface of an {@link Ellipsoid}.
  296. * There are no guarantees about the orientation of the bounding box.
  297. *
  298. * @param {Rectangle} rectangle The cartographic rectangle on the surface of the ellipsoid.
  299. * @param {Number} [minimumHeight=0.0] The minimum height (elevation) within the tile.
  300. * @param {Number} [maximumHeight=0.0] The maximum height (elevation) within the tile.
  301. * @param {Ellipsoid} [ellipsoid=Ellipsoid.WGS84] The ellipsoid on which the rectangle is defined.
  302. * @param {OrientedBoundingBox} [result] The object onto which to store the result.
  303. * @returns {OrientedBoundingBox} The modified result parameter or a new OrientedBoundingBox instance if none was provided.
  304. *
  305. * @exception {DeveloperError} rectangle.width must be between 0 and pi.
  306. * @exception {DeveloperError} rectangle.height must be between 0 and pi.
  307. * @exception {DeveloperError} ellipsoid must be an ellipsoid of revolution (<code>radii.x == radii.y</code>)
  308. */
  309. OrientedBoundingBox.fromRectangle = function (
  310. rectangle,
  311. minimumHeight,
  312. maximumHeight,
  313. ellipsoid,
  314. result
  315. ) {
  316. //>>includeStart('debug', pragmas.debug);
  317. if (!when.defined(rectangle)) {
  318. throw new Check.DeveloperError("rectangle is required");
  319. }
  320. if (rectangle.width < 0.0 || rectangle.width > _Math.CesiumMath.TWO_PI) {
  321. throw new Check.DeveloperError("Rectangle width must be between 0 and 2*pi");
  322. }
  323. if (rectangle.height < 0.0 || rectangle.height > _Math.CesiumMath.PI) {
  324. throw new Check.DeveloperError("Rectangle height must be between 0 and pi");
  325. }
  326. if (
  327. when.defined(ellipsoid) &&
  328. !_Math.CesiumMath.equalsEpsilon(
  329. ellipsoid.radii.x,
  330. ellipsoid.radii.y,
  331. _Math.CesiumMath.EPSILON15
  332. )
  333. ) {
  334. throw new Check.DeveloperError(
  335. "Ellipsoid must be an ellipsoid of revolution (radii.x == radii.y)"
  336. );
  337. }
  338. //>>includeEnd('debug');
  339. minimumHeight = when.defaultValue(minimumHeight, 0.0);
  340. maximumHeight = when.defaultValue(maximumHeight, 0.0);
  341. ellipsoid = when.defaultValue(ellipsoid, Cartesian2.Ellipsoid.WGS84);
  342. var minX, maxX, minY, maxY, minZ, maxZ, plane;
  343. if (rectangle.width <= _Math.CesiumMath.PI) {
  344. // The bounding box will be aligned with the tangent plane at the center of the rectangle.
  345. var tangentPointCartographic = Cartesian2.Rectangle.center(
  346. rectangle,
  347. scratchRectangleCenterCartographic
  348. );
  349. var tangentPoint = ellipsoid.cartographicToCartesian(
  350. tangentPointCartographic,
  351. scratchRectangleCenter
  352. );
  353. var tangentPlane = new EllipsoidTangentPlane.EllipsoidTangentPlane(tangentPoint, ellipsoid);
  354. plane = tangentPlane.plane;
  355. // If the rectangle spans the equator, CW is instead aligned with the equator (because it sticks out the farthest at the equator).
  356. var lonCenter = tangentPointCartographic.longitude;
  357. var latCenter =
  358. rectangle.south < 0.0 && rectangle.north > 0.0
  359. ? 0.0
  360. : tangentPointCartographic.latitude;
  361. // Compute XY extents using the rectangle at maximum height
  362. var perimeterCartographicNC = Cartesian2.Cartographic.fromRadians(
  363. lonCenter,
  364. rectangle.north,
  365. maximumHeight,
  366. scratchPerimeterCartographicNC
  367. );
  368. var perimeterCartographicNW = Cartesian2.Cartographic.fromRadians(
  369. rectangle.west,
  370. rectangle.north,
  371. maximumHeight,
  372. scratchPerimeterCartographicNW
  373. );
  374. var perimeterCartographicCW = Cartesian2.Cartographic.fromRadians(
  375. rectangle.west,
  376. latCenter,
  377. maximumHeight,
  378. scratchPerimeterCartographicCW
  379. );
  380. var perimeterCartographicSW = Cartesian2.Cartographic.fromRadians(
  381. rectangle.west,
  382. rectangle.south,
  383. maximumHeight,
  384. scratchPerimeterCartographicSW
  385. );
  386. var perimeterCartographicSC = Cartesian2.Cartographic.fromRadians(
  387. lonCenter,
  388. rectangle.south,
  389. maximumHeight,
  390. scratchPerimeterCartographicSC
  391. );
  392. var perimeterCartesianNC = ellipsoid.cartographicToCartesian(
  393. perimeterCartographicNC,
  394. scratchPerimeterCartesianNC
  395. );
  396. var perimeterCartesianNW = ellipsoid.cartographicToCartesian(
  397. perimeterCartographicNW,
  398. scratchPerimeterCartesianNW
  399. );
  400. var perimeterCartesianCW = ellipsoid.cartographicToCartesian(
  401. perimeterCartographicCW,
  402. scratchPerimeterCartesianCW
  403. );
  404. var perimeterCartesianSW = ellipsoid.cartographicToCartesian(
  405. perimeterCartographicSW,
  406. scratchPerimeterCartesianSW
  407. );
  408. var perimeterCartesianSC = ellipsoid.cartographicToCartesian(
  409. perimeterCartographicSC,
  410. scratchPerimeterCartesianSC
  411. );
  412. var perimeterProjectedNC = tangentPlane.projectPointToNearestOnPlane(
  413. perimeterCartesianNC,
  414. scratchPerimeterProjectedNC
  415. );
  416. var perimeterProjectedNW = tangentPlane.projectPointToNearestOnPlane(
  417. perimeterCartesianNW,
  418. scratchPerimeterProjectedNW
  419. );
  420. var perimeterProjectedCW = tangentPlane.projectPointToNearestOnPlane(
  421. perimeterCartesianCW,
  422. scratchPerimeterProjectedCW
  423. );
  424. var perimeterProjectedSW = tangentPlane.projectPointToNearestOnPlane(
  425. perimeterCartesianSW,
  426. scratchPerimeterProjectedSW
  427. );
  428. var perimeterProjectedSC = tangentPlane.projectPointToNearestOnPlane(
  429. perimeterCartesianSC,
  430. scratchPerimeterProjectedSC
  431. );
  432. minX = Math.min(
  433. perimeterProjectedNW.x,
  434. perimeterProjectedCW.x,
  435. perimeterProjectedSW.x
  436. );
  437. maxX = -minX; // symmetrical
  438. maxY = Math.max(perimeterProjectedNW.y, perimeterProjectedNC.y);
  439. minY = Math.min(perimeterProjectedSW.y, perimeterProjectedSC.y);
  440. // Compute minimum Z using the rectangle at minimum height, since it will be deeper than the maximum height
  441. perimeterCartographicNW.height = perimeterCartographicSW.height = minimumHeight;
  442. perimeterCartesianNW = ellipsoid.cartographicToCartesian(
  443. perimeterCartographicNW,
  444. scratchPerimeterCartesianNW
  445. );
  446. perimeterCartesianSW = ellipsoid.cartographicToCartesian(
  447. perimeterCartographicSW,
  448. scratchPerimeterCartesianSW
  449. );
  450. minZ = Math.min(
  451. Plane.Plane.getPointDistance(plane, perimeterCartesianNW),
  452. Plane.Plane.getPointDistance(plane, perimeterCartesianSW)
  453. );
  454. maxZ = maximumHeight; // Since the tangent plane touches the surface at height = 0, this is okay
  455. return fromPlaneExtents(
  456. tangentPlane.origin,
  457. tangentPlane.xAxis,
  458. tangentPlane.yAxis,
  459. tangentPlane.zAxis,
  460. minX,
  461. maxX,
  462. minY,
  463. maxY,
  464. minZ,
  465. maxZ,
  466. result
  467. );
  468. }
  469. // Handle the case where rectangle width is greater than PI (wraps around more than half the ellipsoid).
  470. var fullyAboveEquator = rectangle.south > 0.0;
  471. var fullyBelowEquator = rectangle.north < 0.0;
  472. var latitudeNearestToEquator = fullyAboveEquator
  473. ? rectangle.south
  474. : fullyBelowEquator
  475. ? rectangle.north
  476. : 0.0;
  477. var centerLongitude = Cartesian2.Rectangle.center(
  478. rectangle,
  479. scratchRectangleCenterCartographic
  480. ).longitude;
  481. // Plane is located at the rectangle's center longitude and the rectangle's latitude that is closest to the equator. It rotates around the Z axis.
  482. // This results in a better fit than the obb approach for smaller rectangles, which orients with the rectangle's center normal.
  483. var planeOrigin = Cartesian2.Cartesian3.fromRadians(
  484. centerLongitude,
  485. latitudeNearestToEquator,
  486. maximumHeight,
  487. ellipsoid,
  488. scratchPlaneOrigin
  489. );
  490. planeOrigin.z = 0.0; // center the plane on the equator to simpify plane normal calculation
  491. var isPole =
  492. Math.abs(planeOrigin.x) < _Math.CesiumMath.EPSILON10 &&
  493. Math.abs(planeOrigin.y) < _Math.CesiumMath.EPSILON10;
  494. var planeNormal = !isPole
  495. ? Cartesian2.Cartesian3.normalize(planeOrigin, scratchPlaneNormal)
  496. : Cartesian2.Cartesian3.UNIT_X;
  497. var planeYAxis = Cartesian2.Cartesian3.UNIT_Z;
  498. var planeXAxis = Cartesian2.Cartesian3.cross(planeNormal, planeYAxis, scratchPlaneXAxis);
  499. plane = Plane.Plane.fromPointNormal(planeOrigin, planeNormal, scratchPlane);
  500. // Get the horizon point relative to the center. This will be the farthest extent in the plane's X dimension.
  501. var horizonCartesian = Cartesian2.Cartesian3.fromRadians(
  502. centerLongitude + _Math.CesiumMath.PI_OVER_TWO,
  503. latitudeNearestToEquator,
  504. maximumHeight,
  505. ellipsoid,
  506. scratchHorizonCartesian
  507. );
  508. maxX = Cartesian2.Cartesian3.dot(
  509. Plane.Plane.projectPointOntoPlane(
  510. plane,
  511. horizonCartesian,
  512. scratchHorizonProjected
  513. ),
  514. planeXAxis
  515. );
  516. minX = -maxX; // symmetrical
  517. // Get the min and max Y, using the height that will give the largest extent
  518. maxY = Cartesian2.Cartesian3.fromRadians(
  519. 0.0,
  520. rectangle.north,
  521. fullyBelowEquator ? minimumHeight : maximumHeight,
  522. ellipsoid,
  523. scratchMaxY
  524. ).z;
  525. minY = Cartesian2.Cartesian3.fromRadians(
  526. 0.0,
  527. rectangle.south,
  528. fullyAboveEquator ? minimumHeight : maximumHeight,
  529. ellipsoid,
  530. scratchMinY
  531. ).z;
  532. var farZ = Cartesian2.Cartesian3.fromRadians(
  533. rectangle.east,
  534. latitudeNearestToEquator,
  535. maximumHeight,
  536. ellipsoid,
  537. scratchZ
  538. );
  539. minZ = Plane.Plane.getPointDistance(plane, farZ);
  540. maxZ = 0.0; // plane origin starts at maxZ already
  541. // min and max are local to the plane axes
  542. return fromPlaneExtents(
  543. planeOrigin,
  544. planeXAxis,
  545. planeYAxis,
  546. planeNormal,
  547. minX,
  548. maxX,
  549. minY,
  550. maxY,
  551. minZ,
  552. maxZ,
  553. result
  554. );
  555. };
  556. /**
  557. * Duplicates a OrientedBoundingBox instance.
  558. *
  559. * @param {OrientedBoundingBox} box The bounding box to duplicate.
  560. * @param {OrientedBoundingBox} [result] The object onto which to store the result.
  561. * @returns {OrientedBoundingBox} The modified result parameter or a new OrientedBoundingBox instance if none was provided. (Returns undefined if box is undefined)
  562. */
  563. OrientedBoundingBox.clone = function (box, result) {
  564. if (!when.defined(box)) {
  565. return undefined;
  566. }
  567. if (!when.defined(result)) {
  568. return new OrientedBoundingBox(box.center, box.halfAxes);
  569. }
  570. Cartesian2.Cartesian3.clone(box.center, result.center);
  571. Transforms.Matrix3.clone(box.halfAxes, result.halfAxes);
  572. return result;
  573. };
  574. /**
  575. * Determines which side of a plane the oriented bounding box is located.
  576. *
  577. * @param {OrientedBoundingBox} box The oriented bounding box to test.
  578. * @param {Plane} plane The plane to test against.
  579. * @returns {Intersect} {@link Intersect.INSIDE} if the entire box is on the side of the plane
  580. * the normal is pointing, {@link Intersect.OUTSIDE} if the entire box is
  581. * on the opposite side, and {@link Intersect.INTERSECTING} if the box
  582. * intersects the plane.
  583. */
  584. OrientedBoundingBox.intersectPlane = function (box, plane) {
  585. //>>includeStart('debug', pragmas.debug);
  586. if (!when.defined(box)) {
  587. throw new Check.DeveloperError("box is required.");
  588. }
  589. if (!when.defined(plane)) {
  590. throw new Check.DeveloperError("plane is required.");
  591. }
  592. //>>includeEnd('debug');
  593. var center = box.center;
  594. var normal = plane.normal;
  595. var halfAxes = box.halfAxes;
  596. var normalX = normal.x,
  597. normalY = normal.y,
  598. normalZ = normal.z;
  599. // plane is used as if it is its normal; the first three components are assumed to be normalized
  600. var radEffective =
  601. Math.abs(
  602. normalX * halfAxes[Transforms.Matrix3.COLUMN0ROW0] +
  603. normalY * halfAxes[Transforms.Matrix3.COLUMN0ROW1] +
  604. normalZ * halfAxes[Transforms.Matrix3.COLUMN0ROW2]
  605. ) +
  606. Math.abs(
  607. normalX * halfAxes[Transforms.Matrix3.COLUMN1ROW0] +
  608. normalY * halfAxes[Transforms.Matrix3.COLUMN1ROW1] +
  609. normalZ * halfAxes[Transforms.Matrix3.COLUMN1ROW2]
  610. ) +
  611. Math.abs(
  612. normalX * halfAxes[Transforms.Matrix3.COLUMN2ROW0] +
  613. normalY * halfAxes[Transforms.Matrix3.COLUMN2ROW1] +
  614. normalZ * halfAxes[Transforms.Matrix3.COLUMN2ROW2]
  615. );
  616. var distanceToPlane = Cartesian2.Cartesian3.dot(normal, center) + plane.distance;
  617. if (distanceToPlane <= -radEffective) {
  618. // The entire box is on the negative side of the plane normal
  619. return Transforms.Intersect.OUTSIDE;
  620. } else if (distanceToPlane >= radEffective) {
  621. // The entire box is on the positive side of the plane normal
  622. return Transforms.Intersect.INSIDE;
  623. }
  624. return Transforms.Intersect.INTERSECTING;
  625. };
  626. var scratchCartesianU = new Cartesian2.Cartesian3();
  627. var scratchCartesianV = new Cartesian2.Cartesian3();
  628. var scratchCartesianW = new Cartesian2.Cartesian3();
  629. var scratchPPrime = new Cartesian2.Cartesian3();
  630. /**
  631. * Computes the estimated distance squared from the closest point on a bounding box to a point.
  632. *
  633. * @param {OrientedBoundingBox} box The box.
  634. * @param {Cartesian3} cartesian The point
  635. * @returns {Number} The estimated distance squared from the bounding sphere to the point.
  636. *
  637. * @example
  638. * // Sort bounding boxes from back to front
  639. * boxes.sort(function(a, b) {
  640. * return Cesium.OrientedBoundingBox.distanceSquaredTo(b, camera.positionWC) - Cesium.OrientedBoundingBox.distanceSquaredTo(a, camera.positionWC);
  641. * });
  642. */
  643. OrientedBoundingBox.distanceSquaredTo = function (box, cartesian) {
  644. // See Geometric Tools for Computer Graphics 10.4.2
  645. //>>includeStart('debug', pragmas.debug);
  646. if (!when.defined(box)) {
  647. throw new Check.DeveloperError("box is required.");
  648. }
  649. if (!when.defined(cartesian)) {
  650. throw new Check.DeveloperError("cartesian is required.");
  651. }
  652. //>>includeEnd('debug');
  653. var offset = Cartesian2.Cartesian3.subtract(cartesian, box.center, scratchOffset);
  654. var halfAxes = box.halfAxes;
  655. var u = Transforms.Matrix3.getColumn(halfAxes, 0, scratchCartesianU);
  656. var v = Transforms.Matrix3.getColumn(halfAxes, 1, scratchCartesianV);
  657. var w = Transforms.Matrix3.getColumn(halfAxes, 2, scratchCartesianW);
  658. var uHalf = Cartesian2.Cartesian3.magnitude(u);
  659. var vHalf = Cartesian2.Cartesian3.magnitude(v);
  660. var wHalf = Cartesian2.Cartesian3.magnitude(w);
  661. Cartesian2.Cartesian3.normalize(u, u);
  662. Cartesian2.Cartesian3.normalize(v, v);
  663. Cartesian2.Cartesian3.normalize(w, w);
  664. var pPrime = scratchPPrime;
  665. pPrime.x = Cartesian2.Cartesian3.dot(offset, u);
  666. pPrime.y = Cartesian2.Cartesian3.dot(offset, v);
  667. pPrime.z = Cartesian2.Cartesian3.dot(offset, w);
  668. var distanceSquared = 0.0;
  669. var d;
  670. if (pPrime.x < -uHalf) {
  671. d = pPrime.x + uHalf;
  672. distanceSquared += d * d;
  673. } else if (pPrime.x > uHalf) {
  674. d = pPrime.x - uHalf;
  675. distanceSquared += d * d;
  676. }
  677. if (pPrime.y < -vHalf) {
  678. d = pPrime.y + vHalf;
  679. distanceSquared += d * d;
  680. } else if (pPrime.y > vHalf) {
  681. d = pPrime.y - vHalf;
  682. distanceSquared += d * d;
  683. }
  684. if (pPrime.z < -wHalf) {
  685. d = pPrime.z + wHalf;
  686. distanceSquared += d * d;
  687. } else if (pPrime.z > wHalf) {
  688. d = pPrime.z - wHalf;
  689. distanceSquared += d * d;
  690. }
  691. return distanceSquared;
  692. };
  693. var scratchCorner = new Cartesian2.Cartesian3();
  694. var scratchToCenter = new Cartesian2.Cartesian3();
  695. /**
  696. * The distances calculated by the vector from the center of the bounding box to position projected onto direction.
  697. * <br>
  698. * If you imagine the infinite number of planes with normal direction, this computes the smallest distance to the
  699. * closest and farthest planes from position that intersect the bounding box.
  700. *
  701. * @param {OrientedBoundingBox} box The bounding box to calculate the distance to.
  702. * @param {Cartesian3} position The position to calculate the distance from.
  703. * @param {Cartesian3} direction The direction from position.
  704. * @param {Interval} [result] A Interval to store the nearest and farthest distances.
  705. * @returns {Interval} The nearest and farthest distances on the bounding box from position in direction.
  706. */
  707. OrientedBoundingBox.computePlaneDistances = function (
  708. box,
  709. position,
  710. direction,
  711. result
  712. ) {
  713. //>>includeStart('debug', pragmas.debug);
  714. if (!when.defined(box)) {
  715. throw new Check.DeveloperError("box is required.");
  716. }
  717. if (!when.defined(position)) {
  718. throw new Check.DeveloperError("position is required.");
  719. }
  720. if (!when.defined(direction)) {
  721. throw new Check.DeveloperError("direction is required.");
  722. }
  723. //>>includeEnd('debug');
  724. if (!when.defined(result)) {
  725. result = new Transforms.Interval();
  726. }
  727. var minDist = Number.POSITIVE_INFINITY;
  728. var maxDist = Number.NEGATIVE_INFINITY;
  729. var center = box.center;
  730. var halfAxes = box.halfAxes;
  731. var u = Transforms.Matrix3.getColumn(halfAxes, 0, scratchCartesianU);
  732. var v = Transforms.Matrix3.getColumn(halfAxes, 1, scratchCartesianV);
  733. var w = Transforms.Matrix3.getColumn(halfAxes, 2, scratchCartesianW);
  734. // project first corner
  735. var corner = Cartesian2.Cartesian3.add(u, v, scratchCorner);
  736. Cartesian2.Cartesian3.add(corner, w, corner);
  737. Cartesian2.Cartesian3.add(corner, center, corner);
  738. var toCenter = Cartesian2.Cartesian3.subtract(corner, position, scratchToCenter);
  739. var mag = Cartesian2.Cartesian3.dot(direction, toCenter);
  740. minDist = Math.min(mag, minDist);
  741. maxDist = Math.max(mag, maxDist);
  742. // project second corner
  743. Cartesian2.Cartesian3.add(center, u, corner);
  744. Cartesian2.Cartesian3.add(corner, v, corner);
  745. Cartesian2.Cartesian3.subtract(corner, w, corner);
  746. Cartesian2.Cartesian3.subtract(corner, position, toCenter);
  747. mag = Cartesian2.Cartesian3.dot(direction, toCenter);
  748. minDist = Math.min(mag, minDist);
  749. maxDist = Math.max(mag, maxDist);
  750. // project third corner
  751. Cartesian2.Cartesian3.add(center, u, corner);
  752. Cartesian2.Cartesian3.subtract(corner, v, corner);
  753. Cartesian2.Cartesian3.add(corner, w, corner);
  754. Cartesian2.Cartesian3.subtract(corner, position, toCenter);
  755. mag = Cartesian2.Cartesian3.dot(direction, toCenter);
  756. minDist = Math.min(mag, minDist);
  757. maxDist = Math.max(mag, maxDist);
  758. // project fourth corner
  759. Cartesian2.Cartesian3.add(center, u, corner);
  760. Cartesian2.Cartesian3.subtract(corner, v, corner);
  761. Cartesian2.Cartesian3.subtract(corner, w, corner);
  762. Cartesian2.Cartesian3.subtract(corner, position, toCenter);
  763. mag = Cartesian2.Cartesian3.dot(direction, toCenter);
  764. minDist = Math.min(mag, minDist);
  765. maxDist = Math.max(mag, maxDist);
  766. // project fifth corner
  767. Cartesian2.Cartesian3.subtract(center, u, corner);
  768. Cartesian2.Cartesian3.add(corner, v, corner);
  769. Cartesian2.Cartesian3.add(corner, w, corner);
  770. Cartesian2.Cartesian3.subtract(corner, position, toCenter);
  771. mag = Cartesian2.Cartesian3.dot(direction, toCenter);
  772. minDist = Math.min(mag, minDist);
  773. maxDist = Math.max(mag, maxDist);
  774. // project sixth corner
  775. Cartesian2.Cartesian3.subtract(center, u, corner);
  776. Cartesian2.Cartesian3.add(corner, v, corner);
  777. Cartesian2.Cartesian3.subtract(corner, w, corner);
  778. Cartesian2.Cartesian3.subtract(corner, position, toCenter);
  779. mag = Cartesian2.Cartesian3.dot(direction, toCenter);
  780. minDist = Math.min(mag, minDist);
  781. maxDist = Math.max(mag, maxDist);
  782. // project seventh corner
  783. Cartesian2.Cartesian3.subtract(center, u, corner);
  784. Cartesian2.Cartesian3.subtract(corner, v, corner);
  785. Cartesian2.Cartesian3.add(corner, w, corner);
  786. Cartesian2.Cartesian3.subtract(corner, position, toCenter);
  787. mag = Cartesian2.Cartesian3.dot(direction, toCenter);
  788. minDist = Math.min(mag, minDist);
  789. maxDist = Math.max(mag, maxDist);
  790. // project eighth corner
  791. Cartesian2.Cartesian3.subtract(center, u, corner);
  792. Cartesian2.Cartesian3.subtract(corner, v, corner);
  793. Cartesian2.Cartesian3.subtract(corner, w, corner);
  794. Cartesian2.Cartesian3.subtract(corner, position, toCenter);
  795. mag = Cartesian2.Cartesian3.dot(direction, toCenter);
  796. minDist = Math.min(mag, minDist);
  797. maxDist = Math.max(mag, maxDist);
  798. result.start = minDist;
  799. result.stop = maxDist;
  800. return result;
  801. };
  802. var scratchBoundingSphere = new Transforms.BoundingSphere();
  803. /**
  804. * Determines whether or not a bounding box is hidden from view by the occluder.
  805. *
  806. * @param {OrientedBoundingBox} box The bounding box surrounding the occludee object.
  807. * @param {Occluder} occluder The occluder.
  808. * @returns {Boolean} <code>true</code> if the box is not visible; otherwise <code>false</code>.
  809. */
  810. OrientedBoundingBox.isOccluded = function (box, occluder) {
  811. //>>includeStart('debug', pragmas.debug);
  812. if (!when.defined(box)) {
  813. throw new Check.DeveloperError("box is required.");
  814. }
  815. if (!when.defined(occluder)) {
  816. throw new Check.DeveloperError("occluder is required.");
  817. }
  818. //>>includeEnd('debug');
  819. var sphere = Transforms.BoundingSphere.fromOrientedBoundingBox(
  820. box,
  821. scratchBoundingSphere
  822. );
  823. return !occluder.isBoundingSphereVisible(sphere);
  824. };
  825. /**
  826. * Determines which side of a plane the oriented bounding box is located.
  827. *
  828. * @param {Plane} plane The plane to test against.
  829. * @returns {Intersect} {@link Intersect.INSIDE} if the entire box is on the side of the plane
  830. * the normal is pointing, {@link Intersect.OUTSIDE} if the entire box is
  831. * on the opposite side, and {@link Intersect.INTERSECTING} if the box
  832. * intersects the plane.
  833. */
  834. OrientedBoundingBox.prototype.intersectPlane = function (plane) {
  835. return OrientedBoundingBox.intersectPlane(this, plane);
  836. };
  837. /**
  838. * Computes the estimated distance squared from the closest point on a bounding box to a point.
  839. *
  840. * @param {Cartesian3} cartesian The point
  841. * @returns {Number} The estimated distance squared from the bounding sphere to the point.
  842. *
  843. * @example
  844. * // Sort bounding boxes from back to front
  845. * boxes.sort(function(a, b) {
  846. * return b.distanceSquaredTo(camera.positionWC) - a.distanceSquaredTo(camera.positionWC);
  847. * });
  848. */
  849. OrientedBoundingBox.prototype.distanceSquaredTo = function (cartesian) {
  850. return OrientedBoundingBox.distanceSquaredTo(this, cartesian);
  851. };
  852. /**
  853. * The distances calculated by the vector from the center of the bounding box to position projected onto direction.
  854. * <br>
  855. * If you imagine the infinite number of planes with normal direction, this computes the smallest distance to the
  856. * closest and farthest planes from position that intersect the bounding box.
  857. *
  858. * @param {Cartesian3} position The position to calculate the distance from.
  859. * @param {Cartesian3} direction The direction from position.
  860. * @param {Interval} [result] A Interval to store the nearest and farthest distances.
  861. * @returns {Interval} The nearest and farthest distances on the bounding box from position in direction.
  862. */
  863. OrientedBoundingBox.prototype.computePlaneDistances = function (
  864. position,
  865. direction,
  866. result
  867. ) {
  868. return OrientedBoundingBox.computePlaneDistances(
  869. this,
  870. position,
  871. direction,
  872. result
  873. );
  874. };
  875. /**
  876. * Determines whether or not a bounding box is hidden from view by the occluder.
  877. *
  878. * @param {Occluder} occluder The occluder.
  879. * @returns {Boolean} <code>true</code> if the sphere is not visible; otherwise <code>false</code>.
  880. */
  881. OrientedBoundingBox.prototype.isOccluded = function (occluder) {
  882. return OrientedBoundingBox.isOccluded(this, occluder);
  883. };
  884. /**
  885. * Compares the provided OrientedBoundingBox componentwise and returns
  886. * <code>true</code> if they are equal, <code>false</code> otherwise.
  887. *
  888. * @param {OrientedBoundingBox} left The first OrientedBoundingBox.
  889. * @param {OrientedBoundingBox} right The second OrientedBoundingBox.
  890. * @returns {Boolean} <code>true</code> if left and right are equal, <code>false</code> otherwise.
  891. */
  892. OrientedBoundingBox.equals = function (left, right) {
  893. return (
  894. left === right ||
  895. (when.defined(left) &&
  896. when.defined(right) &&
  897. Cartesian2.Cartesian3.equals(left.center, right.center) &&
  898. Transforms.Matrix3.equals(left.halfAxes, right.halfAxes))
  899. );
  900. };
  901. /**
  902. * Duplicates this OrientedBoundingBox instance.
  903. *
  904. * @param {OrientedBoundingBox} [result] The object onto which to store the result.
  905. * @returns {OrientedBoundingBox} The modified result parameter or a new OrientedBoundingBox instance if one was not provided.
  906. */
  907. OrientedBoundingBox.prototype.clone = function (result) {
  908. return OrientedBoundingBox.clone(this, result);
  909. };
  910. /**
  911. * Compares this OrientedBoundingBox against the provided OrientedBoundingBox componentwise and returns
  912. * <code>true</code> if they are equal, <code>false</code> otherwise.
  913. *
  914. * @param {OrientedBoundingBox} [right] The right hand side OrientedBoundingBox.
  915. * @returns {Boolean} <code>true</code> if they are equal, <code>false</code> otherwise.
  916. */
  917. OrientedBoundingBox.prototype.equals = function (right) {
  918. return OrientedBoundingBox.equals(this, right);
  919. };
  920. exports.OrientedBoundingBox = OrientedBoundingBox;
  921. });
  922. //# sourceMappingURL=OrientedBoundingBox-8897f474.js.map