Rectangular_Area_Search_Function.cs 6.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193
  1. using OpenCvSharp;
  2. namespace SimulationCommon;
  3. public class Rectangular_Area_Search_Function
  4. {
  5. public static double MokatuoLon(double Lon)
  6. {
  7. double R = 6378.137;
  8. double y = R * Math.Log(Math.Tan((Lon + 90) * Math.PI / 360));
  9. return y;
  10. }
  11. public static double MokatuoLat(double Lat)
  12. {
  13. double R = 6378.137;
  14. double x = (Lat * Math.PI * R) / 180;
  15. return x;
  16. }
  17. //反墨卡托投影
  18. public static double RMokatuoLon(double y)
  19. {
  20. double R = 6378.137;
  21. double lon = Math.Atan(Math.Exp(y / R)) * 360 / Math.PI - 90;
  22. return lon;
  23. }
  24. public static double RMokatuoLat(double x)
  25. {
  26. double R = 6378.137;
  27. double lat = x * 180 / (R * Math.PI);
  28. return lat;
  29. }
  30. private static double[] GetLinearEquation(Point2f point1, Point2f point2)
  31. {
  32. double sign = 1;
  33. double parameter_a = point2.Y - point1.Y;
  34. if (parameter_a < 0)
  35. {
  36. sign = -1;
  37. parameter_a = sign * parameter_a;
  38. }
  39. double parameter_b = sign * (point1.X - point2.X);
  40. double parameter_c = sign * (point1.Y * point2.X - point1.X * point2.Y);
  41. return new double[] { parameter_a, parameter_b, parameter_c };
  42. }
  43. private static double[] GetParallelLine(Point2f point, double[] line)
  44. {
  45. double parameter_a = line[0];
  46. double parameter_b = line[1];
  47. double parameter_c = -point.X * parameter_a - point.Y * parameter_b;
  48. return new double[] { parameter_a, parameter_b, parameter_c };
  49. }
  50. private static double[] GetpPerpendicular(Point2f point, double[] line)
  51. {
  52. double parameter_a = line[1];
  53. double parameter_b = -line[0];
  54. double parameter_c = -point.Y * parameter_b - point.X * parameter_a;
  55. return new double[] { parameter_a, parameter_b, parameter_c };
  56. }
  57. private static double[] GetCrossPoint(double[] line1, double[] line2)
  58. {
  59. double cross_y = (line1[2] * line2[0] - line2[2] * line1[0]) / (line1[0] * line2[1] - line2[0] * line1[1]);
  60. double cross_x = (line2[2] * line1[1] - line1[2] * line2[1]) / (line1[0] * line2[1] - line2[0] * line1[1]);
  61. return new double[] { cross_x, cross_y };
  62. }
  63. private static double GetDistance(Point2f point, double[] line)
  64. {
  65. return Math.Abs(line[0] * point.X + line[1] * point.Y + line[2]) /
  66. Math.Sqrt(Math.Pow(line[0], 2) + Math.Pow(line[1], 2));
  67. }
  68. private static int CheckDirection(Point2f point, double[] line)
  69. {
  70. double result = line[0] * point.X + line[1] * point.Y + line[2];
  71. if (result < 0) return -1;
  72. if (result > 0) return 1;
  73. return 0;
  74. }
  75. private static double GetRectangularArea(double[] Point0, double[] Point1, double[] Point2, double[] Point3)
  76. {
  77. double width = Math.Sqrt(Math.Pow(Point0[0] - Point1[0], 2) + Math.Pow(Point0[1] - Point1[1], 2));
  78. double height = Math.Sqrt(Math.Pow(Point1[0] - Point2[0], 2) + Math.Pow(Point1[1] - Point2[1], 2));
  79. return width * height;
  80. }
  81. public static Point2f[] GetConvexHull(List<Point2f> pointList)
  82. {
  83. Point2f[] hull = Cv2.ConvexHull(pointList);
  84. return hull;
  85. }
  86. public static List<double[]> MinEnclosingRectangle(Point2f[] hull)
  87. {
  88. double min_rectangular_area = double.PositiveInfinity;
  89. List<double[]> Point_muster = new List<double[]>{};
  90. for (int num = 0; num < hull.Length - 1; num++)
  91. {
  92. double[] Base_line = GetLinearEquation(hull[num], hull[num + 1]);
  93. double max_Ditance = 0;
  94. Point2f Point = hull[num + 1];
  95. foreach (Point2f i_point in hull)
  96. {
  97. if ((i_point.X == hull[num].X && i_point.Y == hull[num].Y) ||
  98. (i_point.X == hull[num + 1].X && i_point.Y == hull[num + 1].Y))
  99. {
  100. continue;
  101. }
  102. double distance = GetDistance(i_point, Base_line);
  103. if (distance > max_Ditance)
  104. {
  105. max_Ditance = distance;
  106. Point = i_point;
  107. }
  108. }
  109. double[] line_Parallel_Base = GetParallelLine(Point, Base_line);
  110. double[] line_Perpendicular_Base_right = null;
  111. double[] line_Perpendicular_Base_left = null;
  112. foreach (Point2f j_point in hull)
  113. {
  114. double[] line_Preparatory = GetpPerpendicular(j_point, Base_line);
  115. int point_right = 0;
  116. int point_left = 0;
  117. foreach (Point2f k_point in hull)
  118. {
  119. if (k_point.X == j_point.X && k_point.Y == j_point.Y)
  120. {
  121. continue;
  122. }
  123. if (CheckDirection(k_point, line_Preparatory) < 0)
  124. {
  125. point_left++;
  126. }
  127. else
  128. {
  129. point_right++;
  130. }
  131. }
  132. if (point_right == hull.Length - 1)
  133. {
  134. line_Perpendicular_Base_right = line_Preparatory;
  135. }
  136. if (point_left == hull.Length - 1)
  137. {
  138. line_Perpendicular_Base_left = line_Preparatory;
  139. }
  140. }
  141. double[] Point0 = GetCrossPoint(Base_line, line_Perpendicular_Base_left);
  142. double[] Point1 = GetCrossPoint(Base_line, line_Perpendicular_Base_right);
  143. double[] Point2 = GetCrossPoint(line_Parallel_Base, line_Perpendicular_Base_right);
  144. double[] Point3 = GetCrossPoint(line_Parallel_Base, line_Perpendicular_Base_left);
  145. double rectangular_area = GetRectangularArea(Point0, Point1, Point2, Point3);
  146. if (rectangular_area < min_rectangular_area)
  147. {
  148. min_rectangular_area = rectangular_area;
  149. Point_muster = new List<double[]> {
  150. Point0,
  151. Point1,
  152. Point2,
  153. Point3
  154. };
  155. }
  156. }
  157. return Point_muster;
  158. }
  159. }