BoundingBox.cs 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532
  1. // MIT License - Copyright (C) The Mono.Xna Team
  2. // This file is subject to the terms and conditions defined in
  3. // file 'LICENSE.txt', which is part of this source code package.
  4. using System;
  5. using System.Collections.Generic;
  6. using System.Diagnostics;
  7. using System.Runtime.Serialization;
  8. namespace CommonLang.Geometry
  9. {
  10. public struct BoundingBox : IEquatable<BoundingBox>
  11. {
  12. #region Public Fields
  13. public Vector3 Min;
  14. public Vector3 Max;
  15. public const int CornerCount = 8;
  16. #endregion Public Fields
  17. #region Public Constructors
  18. public BoundingBox(Vector3 min, Vector3 max)
  19. {
  20. this.Min = min;
  21. this.Max = max;
  22. }
  23. #endregion Public Constructors
  24. #region Public Methods
  25. public ContainmentType Contains(BoundingBox box)
  26. {
  27. //test if all corner is in the same side of a face by just checking min and max
  28. if (box.Max.X < Min.X
  29. || box.Min.X > Max.X
  30. || box.Max.Y < Min.Y
  31. || box.Min.Y > Max.Y
  32. || box.Max.Z < Min.Z
  33. || box.Min.Z > Max.Z)
  34. return ContainmentType.Disjoint;
  35. if (box.Min.X >= Min.X
  36. && box.Max.X <= Max.X
  37. && box.Min.Y >= Min.Y
  38. && box.Max.Y <= Max.Y
  39. && box.Min.Z >= Min.Z
  40. && box.Max.Z <= Max.Z)
  41. return ContainmentType.Contains;
  42. return ContainmentType.Intersects;
  43. }
  44. public void Contains(ref BoundingBox box, out ContainmentType result)
  45. {
  46. result = Contains(box);
  47. }
  48. public ContainmentType Contains(BoundingFrustum frustum)
  49. {
  50. //TODO: bad done here need a fix.
  51. //Because question is not frustum contain box but reverse and this is not the same
  52. int i;
  53. ContainmentType contained;
  54. Vector3[] corners = frustum.GetCorners();
  55. // First we check if frustum is in box
  56. for (i = 0; i < corners.Length; i++)
  57. {
  58. this.Contains(ref corners[i], out contained);
  59. if (contained == ContainmentType.Disjoint)
  60. break;
  61. }
  62. if (i == corners.Length) // This means we checked all the corners and they were all contain or instersect
  63. return ContainmentType.Contains;
  64. if (i != 0) // if i is not equal to zero, we can fastpath and say that this box intersects
  65. return ContainmentType.Intersects;
  66. // If we get here, it means the first (and only) point we checked was actually contained in the frustum.
  67. // So we assume that all other points will also be contained. If one of the points is disjoint, we can
  68. // exit immediately saying that the result is Intersects
  69. i++;
  70. for (; i < corners.Length; i++)
  71. {
  72. this.Contains(ref corners[i], out contained);
  73. if (contained != ContainmentType.Contains)
  74. return ContainmentType.Intersects;
  75. }
  76. // If we get here, then we know all the points were actually contained, therefore result is Contains
  77. return ContainmentType.Contains;
  78. }
  79. public ContainmentType Contains(BoundingSphere sphere)
  80. {
  81. if (sphere.Center.X - Min.X >= sphere.Radius
  82. && sphere.Center.Y - Min.Y >= sphere.Radius
  83. && sphere.Center.Z - Min.Z >= sphere.Radius
  84. && Max.X - sphere.Center.X >= sphere.Radius
  85. && Max.Y - sphere.Center.Y >= sphere.Radius
  86. && Max.Z - sphere.Center.Z >= sphere.Radius)
  87. return ContainmentType.Contains;
  88. double dmin = 0;
  89. double e = sphere.Center.X - Min.X;
  90. if (e < 0)
  91. {
  92. if (e < -sphere.Radius)
  93. {
  94. return ContainmentType.Disjoint;
  95. }
  96. dmin += e * e;
  97. }
  98. else
  99. {
  100. e = sphere.Center.X - Max.X;
  101. if (e > 0)
  102. {
  103. if (e > sphere.Radius)
  104. {
  105. return ContainmentType.Disjoint;
  106. }
  107. dmin += e * e;
  108. }
  109. }
  110. e = sphere.Center.Y - Min.Y;
  111. if (e < 0)
  112. {
  113. if (e < -sphere.Radius)
  114. {
  115. return ContainmentType.Disjoint;
  116. }
  117. dmin += e * e;
  118. }
  119. else
  120. {
  121. e = sphere.Center.Y - Max.Y;
  122. if (e > 0)
  123. {
  124. if (e > sphere.Radius)
  125. {
  126. return ContainmentType.Disjoint;
  127. }
  128. dmin += e * e;
  129. }
  130. }
  131. e = sphere.Center.Z - Min.Z;
  132. if (e < 0)
  133. {
  134. if (e < -sphere.Radius)
  135. {
  136. return ContainmentType.Disjoint;
  137. }
  138. dmin += e * e;
  139. }
  140. else
  141. {
  142. e = sphere.Center.Z - Max.Z;
  143. if (e > 0)
  144. {
  145. if (e > sphere.Radius)
  146. {
  147. return ContainmentType.Disjoint;
  148. }
  149. dmin += e * e;
  150. }
  151. }
  152. if (dmin <= sphere.Radius * sphere.Radius)
  153. return ContainmentType.Intersects;
  154. return ContainmentType.Disjoint;
  155. }
  156. public void Contains(ref BoundingSphere sphere, out ContainmentType result)
  157. {
  158. result = this.Contains(sphere);
  159. }
  160. public ContainmentType Contains(Vector3 point)
  161. {
  162. ContainmentType result;
  163. this.Contains(ref point, out result);
  164. return result;
  165. }
  166. public void Contains(ref Vector3 point, out ContainmentType result)
  167. {
  168. //first we get if point is out of box
  169. if (point.X < this.Min.X
  170. || point.X > this.Max.X
  171. || point.Y < this.Min.Y
  172. || point.Y > this.Max.Y
  173. || point.Z < this.Min.Z
  174. || point.Z > this.Max.Z)
  175. {
  176. result = ContainmentType.Disjoint;
  177. }//or if point is on box because coordonate of point is lesser or equal
  178. else if (point.X == this.Min.X
  179. || point.X == this.Max.X
  180. || point.Y == this.Min.Y
  181. || point.Y == this.Max.Y
  182. || point.Z == this.Min.Z
  183. || point.Z == this.Max.Z)
  184. result = ContainmentType.Intersects;
  185. else
  186. result = ContainmentType.Contains;
  187. }
  188. private static readonly Vector3 MaxVector3 = new Vector3(float.MaxValue);
  189. private static readonly Vector3 MinVector3 = new Vector3(float.MinValue);
  190. /// <summary>
  191. /// Create a bounding box from the given list of points.
  192. /// </summary>
  193. /// <param name="points">The list of Vector3 instances defining the point cloud to bound</param>
  194. /// <returns>A bounding box that encapsulates the given point cloud.</returns>
  195. /// <exception cref="System.ArgumentException">Thrown if the given list has no points.</exception>
  196. public static BoundingBox CreateFromPoints(IEnumerable<Vector3> points)
  197. {
  198. if (points == null)
  199. throw new ArgumentNullException();
  200. var empty = true;
  201. var minVec = MaxVector3;
  202. var maxVec = MinVector3;
  203. foreach (var ptVector in points)
  204. {
  205. minVec.X = (minVec.X < ptVector.X) ? minVec.X : ptVector.X;
  206. minVec.Y = (minVec.Y < ptVector.Y) ? minVec.Y : ptVector.Y;
  207. minVec.Z = (minVec.Z < ptVector.Z) ? minVec.Z : ptVector.Z;
  208. maxVec.X = (maxVec.X > ptVector.X) ? maxVec.X : ptVector.X;
  209. maxVec.Y = (maxVec.Y > ptVector.Y) ? maxVec.Y : ptVector.Y;
  210. maxVec.Z = (maxVec.Z > ptVector.Z) ? maxVec.Z : ptVector.Z;
  211. empty = false;
  212. }
  213. if (empty)
  214. throw new ArgumentException();
  215. return new BoundingBox(minVec, maxVec);
  216. }
  217. public static BoundingBox CreateFromSphere(BoundingSphere sphere)
  218. {
  219. BoundingBox result;
  220. CreateFromSphere(ref sphere, out result);
  221. return result;
  222. }
  223. public static void CreateFromSphere(ref BoundingSphere sphere, out BoundingBox result)
  224. {
  225. var corner = new Vector3(sphere.Radius);
  226. result.Min = sphere.Center - corner;
  227. result.Max = sphere.Center + corner;
  228. }
  229. public static BoundingBox CreateMerged(BoundingBox original, BoundingBox additional)
  230. {
  231. BoundingBox result;
  232. CreateMerged(ref original, ref additional, out result);
  233. return result;
  234. }
  235. public static void CreateMerged(ref BoundingBox original, ref BoundingBox additional, out BoundingBox result)
  236. {
  237. result.Min.X = Math.Min(original.Min.X, additional.Min.X);
  238. result.Min.Y = Math.Min(original.Min.Y, additional.Min.Y);
  239. result.Min.Z = Math.Min(original.Min.Z, additional.Min.Z);
  240. result.Max.X = Math.Max(original.Max.X, additional.Max.X);
  241. result.Max.Y = Math.Max(original.Max.Y, additional.Max.Y);
  242. result.Max.Z = Math.Max(original.Max.Z, additional.Max.Z);
  243. }
  244. public bool Equals(BoundingBox other)
  245. {
  246. return (this.Min == other.Min) && (this.Max == other.Max);
  247. }
  248. public override bool Equals(object obj)
  249. {
  250. return (obj is BoundingBox) ? this.Equals((BoundingBox)obj) : false;
  251. }
  252. public Vector3[] GetCorners()
  253. {
  254. return new Vector3[] {
  255. new Vector3(this.Min.X, this.Max.Y, this.Max.Z),
  256. new Vector3(this.Max.X, this.Max.Y, this.Max.Z),
  257. new Vector3(this.Max.X, this.Min.Y, this.Max.Z),
  258. new Vector3(this.Min.X, this.Min.Y, this.Max.Z),
  259. new Vector3(this.Min.X, this.Max.Y, this.Min.Z),
  260. new Vector3(this.Max.X, this.Max.Y, this.Min.Z),
  261. new Vector3(this.Max.X, this.Min.Y, this.Min.Z),
  262. new Vector3(this.Min.X, this.Min.Y, this.Min.Z)
  263. };
  264. }
  265. public void GetCorners(Vector3[] corners)
  266. {
  267. if (corners == null)
  268. {
  269. throw new ArgumentNullException("corners");
  270. }
  271. if (corners.Length < 8)
  272. {
  273. throw new ArgumentOutOfRangeException("corners", "Not Enought Corners");
  274. }
  275. corners[0].X = this.Min.X;
  276. corners[0].Y = this.Max.Y;
  277. corners[0].Z = this.Max.Z;
  278. corners[1].X = this.Max.X;
  279. corners[1].Y = this.Max.Y;
  280. corners[1].Z = this.Max.Z;
  281. corners[2].X = this.Max.X;
  282. corners[2].Y = this.Min.Y;
  283. corners[2].Z = this.Max.Z;
  284. corners[3].X = this.Min.X;
  285. corners[3].Y = this.Min.Y;
  286. corners[3].Z = this.Max.Z;
  287. corners[4].X = this.Min.X;
  288. corners[4].Y = this.Max.Y;
  289. corners[4].Z = this.Min.Z;
  290. corners[5].X = this.Max.X;
  291. corners[5].Y = this.Max.Y;
  292. corners[5].Z = this.Min.Z;
  293. corners[6].X = this.Max.X;
  294. corners[6].Y = this.Min.Y;
  295. corners[6].Z = this.Min.Z;
  296. corners[7].X = this.Min.X;
  297. corners[7].Y = this.Min.Y;
  298. corners[7].Z = this.Min.Z;
  299. }
  300. public override int GetHashCode()
  301. {
  302. return this.Min.GetHashCode() + this.Max.GetHashCode();
  303. }
  304. public bool Intersects(BoundingBox box)
  305. {
  306. bool result;
  307. Intersects(ref box, out result);
  308. return result;
  309. }
  310. public void Intersects(ref BoundingBox box, out bool result)
  311. {
  312. if ((this.Max.X >= box.Min.X) && (this.Min.X <= box.Max.X))
  313. {
  314. if ((this.Max.Y < box.Min.Y) || (this.Min.Y > box.Max.Y))
  315. {
  316. result = false;
  317. return;
  318. }
  319. result = (this.Max.Z >= box.Min.Z) && (this.Min.Z <= box.Max.Z);
  320. return;
  321. }
  322. result = false;
  323. return;
  324. }
  325. public bool Intersects(BoundingFrustum frustum)
  326. {
  327. return frustum.Intersects(this);
  328. }
  329. public bool Intersects(BoundingSphere sphere)
  330. {
  331. if (sphere.Center.X - Min.X > sphere.Radius
  332. && sphere.Center.Y - Min.Y > sphere.Radius
  333. && sphere.Center.Z - Min.Z > sphere.Radius
  334. && Max.X - sphere.Center.X > sphere.Radius
  335. && Max.Y - sphere.Center.Y > sphere.Radius
  336. && Max.Z - sphere.Center.Z > sphere.Radius)
  337. return true;
  338. double dmin = 0;
  339. if (sphere.Center.X - Min.X <= sphere.Radius)
  340. dmin += (sphere.Center.X - Min.X) * (sphere.Center.X - Min.X);
  341. else if (Max.X - sphere.Center.X <= sphere.Radius)
  342. dmin += (sphere.Center.X - Max.X) * (sphere.Center.X - Max.X);
  343. if (sphere.Center.Y - Min.Y <= sphere.Radius)
  344. dmin += (sphere.Center.Y - Min.Y) * (sphere.Center.Y - Min.Y);
  345. else if (Max.Y - sphere.Center.Y <= sphere.Radius)
  346. dmin += (sphere.Center.Y - Max.Y) * (sphere.Center.Y - Max.Y);
  347. if (sphere.Center.Z - Min.Z <= sphere.Radius)
  348. dmin += (sphere.Center.Z - Min.Z) * (sphere.Center.Z - Min.Z);
  349. else if (Max.Z - sphere.Center.Z <= sphere.Radius)
  350. dmin += (sphere.Center.Z - Max.Z) * (sphere.Center.Z - Max.Z);
  351. if (dmin <= sphere.Radius * sphere.Radius)
  352. return true;
  353. return false;
  354. }
  355. public void Intersects(ref BoundingSphere sphere, out bool result)
  356. {
  357. result = Intersects(sphere);
  358. }
  359. public PlaneIntersectionType Intersects(Plane plane)
  360. {
  361. PlaneIntersectionType result;
  362. Intersects(ref plane, out result);
  363. return result;
  364. }
  365. public void Intersects(ref Plane plane, out PlaneIntersectionType result)
  366. {
  367. // See http://zach.in.tu-clausthal.de/teaching/cg_literatur/lighthouse3d_view_frustum_culling/index.html
  368. Vector3 positiveVertex;
  369. Vector3 negativeVertex;
  370. if (plane.Normal.X >= 0)
  371. {
  372. positiveVertex.X = Max.X;
  373. negativeVertex.X = Min.X;
  374. }
  375. else
  376. {
  377. positiveVertex.X = Min.X;
  378. negativeVertex.X = Max.X;
  379. }
  380. if (plane.Normal.Y >= 0)
  381. {
  382. positiveVertex.Y = Max.Y;
  383. negativeVertex.Y = Min.Y;
  384. }
  385. else
  386. {
  387. positiveVertex.Y = Min.Y;
  388. negativeVertex.Y = Max.Y;
  389. }
  390. if (plane.Normal.Z >= 0)
  391. {
  392. positiveVertex.Z = Max.Z;
  393. negativeVertex.Z = Min.Z;
  394. }
  395. else
  396. {
  397. positiveVertex.Z = Min.Z;
  398. negativeVertex.Z = Max.Z;
  399. }
  400. // Inline Vector3.Dot(plane.Normal, negativeVertex) + plane.D;
  401. var distance = plane.Normal.X * negativeVertex.X + plane.Normal.Y * negativeVertex.Y + plane.Normal.Z * negativeVertex.Z + plane.D;
  402. if (distance > 0)
  403. {
  404. result = PlaneIntersectionType.Front;
  405. return;
  406. }
  407. // Inline Vector3.Dot(plane.Normal, positiveVertex) + plane.D;
  408. distance = plane.Normal.X * positiveVertex.X + plane.Normal.Y * positiveVertex.Y + plane.Normal.Z * positiveVertex.Z + plane.D;
  409. if (distance < 0)
  410. {
  411. result = PlaneIntersectionType.Back;
  412. return;
  413. }
  414. result = PlaneIntersectionType.Intersecting;
  415. }
  416. public Nullable<float> Intersects(Ray ray)
  417. {
  418. return ray.Intersects(this);
  419. }
  420. public void Intersects(ref Ray ray, out Nullable<float> result)
  421. {
  422. result = Intersects(ray);
  423. }
  424. public static bool operator ==(BoundingBox a, BoundingBox b)
  425. {
  426. return a.Equals(b);
  427. }
  428. public static bool operator !=(BoundingBox a, BoundingBox b)
  429. {
  430. return !a.Equals(b);
  431. }
  432. internal string DebugDisplayString
  433. {
  434. get
  435. {
  436. return string.Concat(
  437. "Min( ", this.Min.DebugDisplayString, " ) \r\n",
  438. "Max( ",this.Max.DebugDisplayString, " )"
  439. );
  440. }
  441. }
  442. public override string ToString()
  443. {
  444. return "{{Min:" + this.Min.ToString() + " Max:" + this.Max.ToString() + "}}";
  445. }
  446. #endregion Public Methods
  447. }
  448. }