/ManejadorDeMapa/ManejadorDeMapa/PolygonF.cs

# · C# · 464 lines · 297 code · 58 blank · 109 comment · 26 complexity · b2b7848bc8d60d8d827bb0b49a36535f MD5 · raw file

  1. // From http://www.developingfor.net/free-code/polygonf
  2. using System;
  3. using System.Drawing;
  4. using System.Collections.Generic;
  5. using System.Text;
  6. namespace SWallTech.Drawing.Shapes
  7. {
  8. /// <summary>
  9. /// A Class representing a Polygon formed by an array of PointF objects.
  10. /// PolygonF objects are immutable.
  11. /// </summary>
  12. public class PolygonF
  13. {
  14. private PointF[] _pts;
  15. private float _minx = 0;
  16. private float _miny = 0;
  17. private float _maxx = 0;
  18. private float _maxy = 0;
  19. private float _xlength = 0;
  20. private float _ylength = 0;
  21. private PolygonF()
  22. {
  23. }
  24. /// <summary>
  25. /// Creates a new instance of a PolygonF based on the PointF[].
  26. /// </summary>
  27. /// <param name="pts">The array of PointF used to create the PolygonF.</param>
  28. public PolygonF(PointF[] pts)
  29. {
  30. Points = pts;
  31. }
  32. private PointF[] Points
  33. {
  34. get { return _pts; }
  35. set
  36. {
  37. _pts = value;
  38. _minx = _pts[0].X;
  39. _maxx = _pts[0].X;
  40. _miny = _pts[0].Y;
  41. _maxy = _pts[0].Y;
  42. foreach (PointF pt in _pts)
  43. {
  44. if (pt.X < _minx)
  45. {
  46. _minx = pt.X;
  47. }
  48. if (pt.X > _maxx)
  49. {
  50. _maxx = pt.X;
  51. }
  52. if (pt.Y < _miny)
  53. {
  54. _miny = pt.Y;
  55. }
  56. if (pt.Y > _maxy)
  57. {
  58. _maxy = pt.Y;
  59. }
  60. }
  61. _xlength = Math.Abs(_maxx - _minx);
  62. _ylength = Math.Abs(_maxy - _miny);
  63. }
  64. }
  65. /// <summary>
  66. /// The Rectangular Bounds of the Polygon.
  67. /// </summary>
  68. public RectangleF Bounds
  69. {
  70. get
  71. {
  72. return new RectangleF(_minx, _miny, _maxx - _minx, _maxy - _miny);
  73. }
  74. }
  75. /// <summary>
  76. /// The Minimum X coordinate value in the PointF collection.
  77. /// </summary>
  78. public float MinimumX
  79. {
  80. get { return _minx; }
  81. }
  82. /// <summary>
  83. /// The Maximum X coordinate value in the PointF collection.
  84. /// </summary>
  85. public float MaximumX
  86. {
  87. get { return _maxx; }
  88. }
  89. /// <summary>
  90. /// The Minimum Y coordinate value in the PointF collection.
  91. /// </summary>
  92. public float MinimumY
  93. {
  94. get { return _miny; }
  95. }
  96. /// <summary>
  97. /// The Maximum Y coordinate value in the PointF collection.
  98. /// </summary>
  99. public float MaximumY
  100. {
  101. get { return _maxy; }
  102. }
  103. /// <summary>
  104. /// The number of Points in the Polygon.
  105. /// </summary>
  106. public int NumberOfPoints
  107. {
  108. get { return _pts.Length; }
  109. }
  110. /// <summary>
  111. /// Compares the supplied point and determines whether or not it is inside the Rectangular Bounds
  112. /// of the Polygon.
  113. /// </summary>
  114. /// <param name="pt">The PointF to compare.</param>
  115. /// <returns>True if the PointF is within the Rectangular Bounds, False if it is not.</returns>
  116. public bool IsInBounds(PointF pt)
  117. {
  118. return Bounds.Contains(pt);
  119. }
  120. /// <summary>
  121. /// Compares the supplied point and determines whether or not it is inside the Actual Bounds
  122. /// of the Polygon.
  123. /// </summary>
  124. /// <remarks>The calculation formula was converted from the C version available at
  125. /// http://www.ecse.rpi.edu/Homepages/wrf/Research/Short_Notes/pnpoly.html
  126. /// </remarks>
  127. /// <param name="pt">The PointF to compare.</param>
  128. /// <returns>True if the PointF is within the Actual Bounds, False if it is not.</returns>
  129. public bool Contains(PointF pt)
  130. {
  131. bool isIn = false;
  132. if (IsInBounds(pt))
  133. {
  134. int i, j = 0;
  135. // The following code is converted from a C version found at
  136. // http://www.ecse.rpi.edu/Homepages/wrf/Research/Short_Notes/pnpoly.html
  137. for (i = 0, j = NumberOfPoints - 1; i < NumberOfPoints; j = i++)
  138. {
  139. if (
  140. (
  141. ((_pts[i].Y <= pt.Y) && (pt.Y < _pts[j].Y)) || ((_pts[j].Y <= pt.Y) && (pt.Y < _pts[i].Y))
  142. ) &&
  143. (pt.X < (_pts[j].X - _pts[i].X) * (pt.Y - _pts[i].Y) / (_pts[j].Y - _pts[i].Y) + _pts[i].X)
  144. )
  145. {
  146. isIn = !isIn;
  147. }
  148. }
  149. }
  150. return isIn;
  151. }
  152. /// <summary>
  153. /// Returns the PointF that represents the center of the Rectangular Bounds of the Polygon.
  154. /// </summary>
  155. public PointF CenterPointOfBounds
  156. {
  157. get
  158. {
  159. float x = _minx + (_xlength / 2);
  160. float y = _miny + (_ylength / 2);
  161. return new PointF(x, y);
  162. }
  163. }
  164. /// <summary>
  165. /// NOT YET IMPLEMENTED. Currently returns the same as CenterPointOfBounds.
  166. /// This is intended to be the Visual Center of the Polygon, and will be implemented
  167. /// once I can figure out how to calculate that Point.
  168. /// </summary>
  169. public PointF CenterPoint
  170. {
  171. get
  172. {
  173. PointF pt = CenterPointOfBounds;
  174. return pt;
  175. }
  176. }
  177. /// <summary>
  178. /// Calculates the Area of the Polygon.
  179. /// </summary>
  180. public decimal Area
  181. {
  182. get
  183. {
  184. decimal xy = 0M;
  185. for (int i = 0; i < _pts.Length; i++)
  186. {
  187. PointF pt1;
  188. PointF pt2;
  189. if (i == _pts.Length - 1)
  190. {
  191. pt1 = _pts[i];
  192. pt2 = _pts[0];
  193. }
  194. else
  195. {
  196. pt1 = _pts[i];
  197. pt2 = _pts[i + 1];
  198. }
  199. xy += Convert.ToDecimal(pt1.X * pt2.Y);
  200. xy -= Convert.ToDecimal(pt1.Y * pt2.X);
  201. }
  202. decimal area = Convert.ToDecimal(Math.Abs(xy)) * .5M;
  203. return area;
  204. }
  205. }
  206. }
  207. /// <summary>
  208. /// A Class representing a Polygon formed by an array of Point objects.
  209. /// Polygon objects are immutable.
  210. /// </summary>
  211. public class Polygon
  212. {
  213. private Point[] _pts;
  214. private int _minx = 0;
  215. private int _miny = 0;
  216. private int _maxx = 0;
  217. private int _maxy = 0;
  218. private int _xlength = 0;
  219. private int _ylength = 0;
  220. private Polygon()
  221. {
  222. }
  223. /// <summary>
  224. /// Creates a new instance of a Polygon based on the Point[].
  225. /// </summary>
  226. /// <param name="pts">The array of Point used to create the Polygon.</param>
  227. public Polygon(Point[] pts)
  228. {
  229. Points = pts;
  230. }
  231. private Point[] Points
  232. {
  233. get { return _pts; }
  234. set
  235. {
  236. _pts = value;
  237. _minx = _pts[0].X;
  238. _maxx = _pts[0].X;
  239. _miny = _pts[0].Y;
  240. _maxy = _pts[0].Y;
  241. foreach (Point pt in _pts)
  242. {
  243. if (pt.X < _minx)
  244. {
  245. _minx = pt.X;
  246. }
  247. if (pt.X > _maxx)
  248. {
  249. _maxx = pt.X;
  250. }
  251. if (pt.Y < _miny)
  252. {
  253. _miny = pt.Y;
  254. }
  255. if (pt.Y > _maxy)
  256. {
  257. _maxy = pt.Y;
  258. }
  259. }
  260. _xlength = Math.Abs(_maxx - _minx);
  261. _ylength = Math.Abs(_maxy - _miny);
  262. }
  263. }
  264. /// <summary>
  265. /// The Rectangular Bounds of the Polygon.
  266. /// </summary>
  267. public Rectangle Bounds
  268. {
  269. get
  270. {
  271. return new Rectangle(_minx, _miny, _maxx - _minx, _maxy - _miny);
  272. }
  273. }
  274. /// <summary>
  275. /// The Minimum X coordinate value in the PointF collection.
  276. /// </summary>
  277. public int MinimumX
  278. {
  279. get { return _minx; }
  280. }
  281. /// <summary>
  282. /// The Maximum X coordinate value in the PointF collection.
  283. /// </summary>
  284. public int MaximumX
  285. {
  286. get { return _maxx; }
  287. }
  288. /// <summary>
  289. /// The Minimum Y coordinate value in the PointF collection.
  290. /// </summary>
  291. public int MinimumY
  292. {
  293. get { return _miny; }
  294. }
  295. /// <summary>
  296. /// The Maximum Y coordinate value in the PointF collection.
  297. /// </summary>
  298. public int MaximumY
  299. {
  300. get { return _maxy; }
  301. }
  302. /// <summary>
  303. /// The number of Points in the Polygon.
  304. /// </summary>
  305. public int NumberOfPoints
  306. {
  307. get { return _pts.Length; }
  308. }
  309. /// <summary>
  310. /// Compares the supplied point and determines whether or not it is inside the Rectangular Bounds
  311. /// of the Polygon.
  312. /// </summary>
  313. /// <param name="pt">The PointF to compare.</param>
  314. /// <returns>True if the PointF is within the Rectangular Bounds, False if it is not.</returns>
  315. public bool IsInBounds(Point pt)
  316. {
  317. return Bounds.Contains(pt);
  318. }
  319. /// <summary>
  320. /// Compares the supplied point and determines whether or not it is inside the Actual Bounds
  321. /// of the Polygon.
  322. /// </summary>
  323. /// <remarks>The calculation formula was converted from the C version available at
  324. /// http://www.ecse.rpi.edu/Homepages/wrf/Research/Short_Notes/pnpoly.html
  325. /// </remarks>
  326. /// <param name="pt">The PointF to compare.</param>
  327. /// <returns>True if the PointF is within the Actual Bounds, False if it is not.</returns>
  328. public bool Contains(Point pt)
  329. {
  330. bool isIn = false;
  331. if (IsInBounds(pt))
  332. {
  333. int i, j = 0;
  334. // The following code is converted from a C version found at
  335. // http://www.ecse.rpi.edu/Homepages/wrf/Research/Short_Notes/pnpoly.html
  336. for (i = 0, j = NumberOfPoints - 1; i < NumberOfPoints; j = i++)
  337. {
  338. if (
  339. (
  340. ((_pts[i].Y <= pt.Y) && (pt.Y < _pts[j].Y)) || ((_pts[j].Y <= pt.Y) && (pt.Y < _pts[i].Y))
  341. ) &&
  342. (pt.X < (_pts[j].X - _pts[i].X) * (pt.Y - _pts[i].Y) / (_pts[j].Y - _pts[i].Y) + _pts[i].X)
  343. )
  344. {
  345. isIn = !isIn;
  346. }
  347. }
  348. }
  349. return isIn;
  350. }
  351. /// <summary>
  352. /// Returns the PointF that represents the center of the Rectangular Bounds of the Polygon.
  353. /// </summary>
  354. public Point CenterPointOfBounds
  355. {
  356. get
  357. {
  358. int x = _minx + (_xlength / 2);
  359. int y = _miny + (_ylength / 2);
  360. return new Point(x, y);
  361. }
  362. }
  363. /// <summary>
  364. /// NOT YET IMPLEMENTED. Currently returns the same as CenterPointOfBounds.
  365. /// This is intended to be the Visual Center of the Polygon, and will be implemented
  366. /// once I can figure out how to calculate that Point.
  367. /// </summary>
  368. public Point CenterPoint
  369. {
  370. get
  371. {
  372. Point pt = CenterPointOfBounds;
  373. return pt;
  374. }
  375. }
  376. /// <summary>
  377. /// Calculates the Area of the Polygon.
  378. /// </summary>
  379. public decimal Area
  380. {
  381. get
  382. {
  383. decimal xy = 0M;
  384. for (int i = 0; i < _pts.Length; i++)
  385. {
  386. Point pt1;
  387. Point pt2;
  388. if (i == _pts.Length - 1)
  389. {
  390. pt1 = _pts[i];
  391. pt2 = _pts[0];
  392. }
  393. else
  394. {
  395. pt1 = _pts[i];
  396. pt2 = _pts[i + 1];
  397. }
  398. xy += Convert.ToDecimal(pt1.X * pt2.Y);
  399. xy -= Convert.ToDecimal(pt1.Y * pt2.X);
  400. }
  401. decimal area = Convert.ToDecimal(Math.Abs(xy)) * .5M;
  402. return area;
  403. }
  404. }
  405. }
  406. }