/Axe/Axe.Geometry/Plane3.cs

# · C# · 438 lines · 235 code · 58 blank · 145 comment · 51 complexity · 6d5635865dbb93c96064f73e0cb67b4d MD5 · raw file

  1. using System;
  2. using System.ComponentModel; // for TypeConverterAttribute
  3. using System.Diagnostics; // mostly for Debug.Assert(...)
  4. namespace Axe.Geometry
  5. {
  6. /// <summary>
  7. /// Представляет плоскость в пространстве
  8. /// xA + yB + zC - D = 0
  9. /// </summary>
  10. public struct Plane3 : ICloneable
  11. {
  12. /// <summary>
  13. /// Создает новую плоскость
  14. /// </summary>
  15. /// <mParam name="normal"></mParam>
  16. /// <mParam name="constant"></mParam>
  17. public Plane3( Vector3 normal, double constant )
  18. {
  19. this.Normal = normal;
  20. this.Constant = constant;
  21. }
  22. /// <summary>
  23. /// Create the plane that has the given normal and is coplanar with the given point
  24. /// </summary>
  25. /// <mParam name="normal"></mParam>
  26. /// <mParam name="coplanarPoint"></mParam>
  27. /// <returns></returns>
  28. static public Plane3 FromNormalAndPoint( Vector3 normal, Vector3 coplanarPoint )
  29. {
  30. return new Plane3( normal.GetUnit(), Vector3.Dot( normal, coplanarPoint ) );
  31. }
  32. /// <summary>
  33. /// Create the plane defined by the three given points
  34. /// </summary>
  35. /// <mParam name="pt0">Первая вершина</mParam>
  36. /// <mParam name="pt1">Вторая вершина</mParam>
  37. /// <mParam name="pt2">Треться вершина</mParam>
  38. /// <returns>Созданная плоскость</returns>
  39. static public Plane3 FromCoplanarPoints( Vector3 pt0, Vector3 pt1, Vector3 pt2 )
  40. {
  41. Vector3 normal = Vector3.Cross( pt2 - pt1, pt0 - pt1 ).GetUnit();
  42. return new Plane3( normal, Vector3.Dot( normal, pt0 ) );
  43. }
  44. object ICloneable.Clone()
  45. {
  46. return new Plane3( this.Normal, this.Constant );
  47. }
  48. /// <summary>
  49. /// Copy
  50. /// </summary>
  51. /// <returns></returns>
  52. public Plane3 Clone()
  53. {
  54. return new Plane3( this.Normal, this.Constant );
  55. }
  56. /// <summary>
  57. /// The Normal parameter of the plane when defined as "Normal*( pt ) - Constant = 0"
  58. /// </summary>
  59. public Vector3 Normal;
  60. /// <summary>
  61. /// The Constant parameter of the plane when defined as "Normal*( pt ) - Constant = 0"
  62. /// </summary>
  63. public double Constant;
  64. /// <summary>
  65. /// The A parameter of the plane when defined as "Ax + By + Cz - D = 0"
  66. /// </summary>
  67. public double A
  68. {
  69. get { return Normal.X; }
  70. set { Normal.X = value; }
  71. }
  72. /// <summary>
  73. /// The B parameter of the plane when defined as "Ax + By + Cz - D = 0"
  74. /// </summary>
  75. public double B
  76. {
  77. get { return Normal.Y; }
  78. set { Normal.Y = value; }
  79. }
  80. /// <summary>
  81. /// The C parameter of the plane when defined as "Ax + By + Cz - D = 0"
  82. /// </summary>
  83. public double C
  84. {
  85. get { return Normal.Z; }
  86. set { Normal.Z = value; }
  87. }
  88. /// <summary>
  89. /// The D parameter of the plane when defined as "Ax + By + Cz - D = 0"
  90. /// </summary>
  91. public double D
  92. {
  93. get { return Constant; }
  94. set { Constant = value; }
  95. }
  96. /// <summary>
  97. /// Flip the plane.
  98. /// </summary>
  99. public void Flip()
  100. {
  101. Normal = - Normal;
  102. Constant = - Constant;
  103. }
  104. /// <summary>
  105. /// Создает новый plane that is identical the current but flipped.
  106. /// </summary>
  107. /// <returns></returns>
  108. public Plane3 GetFlipped()
  109. {
  110. Plane3 plane = this.Clone();
  111. plane.Flip();
  112. return plane;
  113. }
  114. /// <summary>
  115. /// Get the shortest distance from the point to the plane
  116. /// </summary>
  117. /// <mParam name="pt"></mParam>
  118. /// <returns></returns>
  119. public double GetDistanceToPlane( Vector3 pt )
  120. {
  121. return Vector3.Dot( pt, Normal ) - Constant;
  122. }
  123. /// <summary>
  124. /// Get the sign of the point in relation to the plane
  125. /// </summary>
  126. /// <mParam name="pt"></mParam>
  127. /// <returns></returns>
  128. public Sign GetSign( Vector3 pt )
  129. {
  130. return Math3D.GetSign( GetDistanceToPlane( pt ), Math3D.EpsilonF );
  131. }
  132. /// <summary>
  133. /// Get the aggregate sign of the polygon in relation to the plane
  134. /// </summary>
  135. /// <mParam name="polygon"></mParam>
  136. /// <returns></returns>
  137. public Sign GetSign( Polygon polygon )
  138. {
  139. Sign sign = Sign.Undefined;
  140. foreach( Vector3 p in polygon.Points )
  141. {
  142. sign = Math3D.CombineSigns( sign, this.GetSign( p ) );
  143. }
  144. return sign;
  145. }
  146. /// <summary>
  147. /// Project a point onto the plane
  148. /// </summary>
  149. /// <mParam name="pt"></mParam>
  150. /// <returns></returns>
  151. public Vector3 ProjectOntoPlane( Vector3 pt )
  152. {
  153. return pt + ( Constant - Vector3.Dot( pt, Normal ) ) * Normal;
  154. }
  155. /// <summary>
  156. /// Determine whether the rkLine connecting pt0 to pt1 intersects the plane
  157. /// </summary>
  158. /// <mParam name="pt0"></mParam>
  159. /// <mParam name="pt1"></mParam>
  160. /// <returns></returns>
  161. public bool IsIntersection( Vector3 pt0, Vector3 pt1 )
  162. {
  163. int sign0 = (int) Math3D.GetSign( GetDistanceToPlane( pt0 ) );
  164. int sign1 = (int) Math3D.GetSign( GetDistanceToPlane( pt1 ) );
  165. return ( sign0 < 0 && sign1 > 0 ) || ( sign0 > 0 && sign1 <= 0 );
  166. }
  167. /// <summary>
  168. /// Determine the exact location where the given rkLine segment intersects the plane
  169. /// </summary>
  170. /// <mParam name="pt0"></mParam>
  171. /// <mParam name="pt1"></mParam>
  172. /// <returns></returns>
  173. public Vector3 GetIntersection( Vector3 pt0, Vector3 pt1 )
  174. {
  175. Debug.Assert( IsIntersection( pt0, pt1 ) == true );
  176. Vector3 ptDirection = pt1 - pt0;
  177. double denominator = Vector3.Dot( Normal, ptDirection );
  178. if( denominator == 0 )
  179. {
  180. throw new DivideByZeroException( "Can not get the intersection of a plane with a line when they are parallel to each other." );
  181. }
  182. double t = ( Constant - Vector3.Dot( Normal, pt0 ) ) / denominator;
  183. return pt0 + ptDirection * t;
  184. }
  185. /// <summary>
  186. /// Clip the given polygon by the plane so that only the regions
  187. /// located positive of the plane remain.
  188. /// </summary>
  189. /// <mParam name="polygon"></mParam>
  190. public void ClipPolygon( Polygon polygon )
  191. {
  192. Vector3 ptNormal = polygon.GetNormal();
  193. Vector3Collection vNewPoints = new Vector3Collection();
  194. int iPoints = polygon.Points.Count;
  195. for ( int i = 0; i < iPoints; i ++ )
  196. {
  197. Vector3 pt0 = polygon.Points[ ( i + iPoints - 1 ) % iPoints ];
  198. Vector3 pt1 = polygon.Points[ i ];
  199. int sign0 = (int) Math3D.GetSign( this.GetDistanceToPlane( pt0 ), Math3D.EpsilonF );
  200. int sign1 = (int) Math3D.GetSign( this.GetDistanceToPlane( pt1 ), Math3D.EpsilonF );
  201. if( sign0 > 0 )
  202. {
  203. // rkLine is infront
  204. if ( sign1 >= 0 )
  205. {
  206. vNewPoints.Add( pt1 );
  207. }
  208. // rkLine is entering plane
  209. else if( sign1 < 0 )
  210. {
  211. Debug.Assert( sign0 > 0 && sign1 < 0 );
  212. vNewPoints.Add( this.GetIntersection( pt0, pt1 ) );
  213. }
  214. }
  215. else if( sign0 == 0 )
  216. {
  217. // rkLine is infront
  218. if( sign1 > 0 )
  219. {
  220. vNewPoints.Add( pt1 );
  221. }
  222. // rkLine is coplanar
  223. else if( sign1 == 0 )
  224. {
  225. vNewPoints.Add( pt1 );
  226. }
  227. // rkLine is behind
  228. else if( sign1 < 0 )
  229. {
  230. }
  231. }
  232. else if( sign0 < 0 )
  233. {
  234. // rkLine is leaving plane
  235. if( sign1 > 0 )
  236. {
  237. Debug.Assert( sign0 < 0 && sign1 > 0 );
  238. vNewPoints.Add( this.GetIntersection( pt0, pt1 ) );
  239. vNewPoints.Add( pt1 );
  240. }
  241. // rkLine is leaving plane
  242. else if( sign1 == 0 )
  243. {
  244. vNewPoints.Add( pt1 );
  245. }
  246. // rkLine is behind
  247. else if( sign1 < 0 )
  248. {
  249. }
  250. }
  251. }
  252. // set new points
  253. polygon.Points.Clear();
  254. polygon.Points.AddRange( vNewPoints );
  255. /*if( this.Points.Count >= 3 ) {
  256. if( Vector3.Dot( this.GetNormal(), ptNormal ) < 0 ) {
  257. this.Flip();
  258. }
  259. } */
  260. polygon.Optimize();
  261. }
  262. /// <summary>
  263. /// Are two planes equal?
  264. /// </summary>
  265. /// <mParam name="a"></mParam>
  266. /// <mParam name="b"></mParam>
  267. /// <returns></returns>
  268. static public bool operator==( Plane3 a, Plane3 b )
  269. {
  270. return ( a.Constant == b.Constant ) && ( a.Normal == b.Normal );
  271. }
  272. /// <summary>
  273. /// Are two planes not equal?
  274. /// </summary>
  275. /// <mParam name="a"></mParam>
  276. /// <mParam name="b"></mParam>
  277. /// <returns></returns>
  278. static public bool operator!=( Plane3 a, Plane3 b )
  279. {
  280. return ( a.Constant != b.Constant ) || ( a.Normal != b.Normal );
  281. }
  282. /// <summary>
  283. /// Is given object equal to current planes?
  284. /// </summary>
  285. /// <mParam name="o"></mParam>
  286. /// <returns></returns>
  287. public override bool Equals( object o )
  288. {
  289. if( o is Plane3 )
  290. {
  291. Plane3 plane = (Plane3) o;
  292. return ( this.Constant == plane.Constant ) && ( this.Normal == plane.Normal );
  293. }
  294. return false;
  295. }
  296. /// <summary>
  297. /// Get the hashcode of the planes
  298. /// </summary>
  299. /// <returns></returns>
  300. public override int GetHashCode()
  301. {
  302. return this.Constant.GetHashCode() ^ this.Normal.GetHashCode();
  303. }
  304. /// <summary>
  305. /// Get an orthogonal basis for the plane
  306. /// </summary>
  307. /// <mParam name="u"></mParam>
  308. /// <mParam name="v"></mParam>
  309. public void GetBasis( out Vector3 u, out Vector3 v )
  310. {
  311. Vector3 pt = this.ProjectOntoPlane( Vector3.Origin );
  312. u = this.ProjectOntoPlane( pt + Vector3.XAxis ) - pt;
  313. u = Vector3.Max( u, this.ProjectOntoPlane( pt + Vector3.YAxis ) - pt );
  314. u = Vector3.Max( u, this.ProjectOntoPlane( pt + Vector3.ZAxis ) - pt );
  315. v = Vector3.Cross( u, this.Normal );
  316. u.Normalize();
  317. v.Normalize();
  318. }
  319. /// <summary>
  320. /// Create a polygon coplanar with the current plane that extends to the given extent
  321. /// </summary>
  322. /// <mParam name="fExtent"></mParam>
  323. /// <returns></returns>
  324. public Polygon CreatePolygon( double fExtent )
  325. {
  326. int a, b, c;
  327. double[] n = new double[3] { Math.Abs( Normal.X ), Math.Abs( Normal.Y ), Math.Abs( Normal.Z ) };
  328. if( n[0] >= n[1] && n[0] >= n[2] )
  329. {
  330. a = 1; b = 2; c = 0;
  331. }
  332. else if( n[1] >= n[0] && n[1] >= n[2] )
  333. {
  334. a = 0; b = 2; c = 1;
  335. }
  336. else if( n[2] >= n[0] && n[2] >= n[1] )
  337. {
  338. a = 0; b = 1; c = 2;
  339. }
  340. else
  341. {
  342. a = b = c = -1;
  343. Debug.Assert( false );
  344. }
  345. //Debug.WriteLine( " normal[" + Normal.ToString() + " a[" + a + "] b[" + b + "] c[" + c + "]" );
  346. int[] aSigns = new int[4] { 1, 1, -1, -1 };
  347. int[] bSigns = new int[4] { 1, -1, -1, 1 };
  348. Polygon poly = new Polygon();
  349. for( int i = 0; i < 4; i ++ )
  350. {
  351. Vector3 pt = new Vector3();
  352. pt[a] = fExtent * aSigns[i];
  353. pt[b] = fExtent * bSigns[i];
  354. pt[c] = ( Constant - Normal[a] * pt[a] - Normal[b] * pt[b] ) / Normal[c];
  355. //Debug.WriteLine( " pt[" + i + "] = " + pt.ToString() + " dTp = " + DistanceToPlane( pt ) );
  356. poly.Points.Add( pt );
  357. }
  358. //Debug.WriteLine( " plane.Normal = " + Normal );
  359. //Debug.WriteLine( " plane.Constant = " + Constant );
  360. if( Vector3.Dot( Normal, poly.GetNormal() ) < 0 )
  361. {
  362. poly.Flip();
  363. }
  364. //Debug.WriteLine( " polygon.Normal = " + poly.Normal().ToString() );
  365. return poly;
  366. }
  367. /// <summary>
  368. /// Get a description of the plane.
  369. /// </summary>
  370. /// <returns></returns>
  371. public override string ToString()
  372. {
  373. return "Plane3 [ n=" + Normal.ToString() + " c=" + Constant + " ]";
  374. }
  375. /// <summary>
  376. /// Zero
  377. /// </summary>
  378. static public readonly Plane3 Zero = new Plane3();
  379. }
  380. }