圆形与矩形的碰撞检测

作者:追风剑情 发布于:2016-12-4 12:13 分类:Algorithms

将矩形向外扩张圆半径r来进行碰撞检测

111111.png

主要可以分为以下两个阶段进行碰撞:

1. 将需要检测的长方形,在上下左右4个方向均向外扩张,扩张的长度为圆半径r,如果扩张后得到新的长方形内包含了圆心坐标,则认为两物体具备碰撞的可能(反之则无碰撞的可能)。

2. 在满足条件1的情况下,如果圆心坐标在原长方形以外、扩张后的长方形的左上、左下、右上、右下四个角处,且圆内没有包含长方形最近的顶点,则认为两物体没有碰撞。

示例代码

  1. using System;
  2. using System.Collections.Generic;
  3. using System.Linq;
  4. using System.Text;
  5.  
  6. namespace HitTest
  7. {
  8. class Program
  9. {
  10. static void Main(string[] args)
  11. {
  12.  
  13. Console.Read();
  14. }
  15. }
  16.  
  17. /// <summary>
  18. /// 矩形碰撞器
  19. /// </summary>
  20. public class RectCollider
  21. {
  22. //定义左下角为(minX, minY)
  23. public int minX;
  24. public int minY;
  25. //定义右上角为(maxX, maxY)
  26. public int maxX;
  27. public int maxY;
  28.  
  29. //判断某点是否在矩形内
  30. public bool CheckHit(int x, int y)
  31. {
  32. return x >= minX && x <= maxX && y >= minY && y <= maxY;
  33. }
  34.  
  35. public bool CheckHit(RectCollider rect)
  36. {
  37. //水平方向存在重叠
  38. if (rect.maxX >= minX && rect.minX <= maxX)
  39. {
  40. //垂直方向存在重叠
  41. if (rect.maxY >= minY && rect.minY <= maxY)
  42. return true;
  43. }
  44. return false;
  45. }
  46.  
  47. //判断矩形与圆形是否发生碰撞
  48. public bool CheckHit(CircleCollider circle)
  49. {
  50. //判断圆心是否在原矩形内
  51. if (this.CheckHit(circle.x, circle.y))
  52. return true;
  53.  
  54. //扩展矩形区
  55. RectCollider rect_b = new RectCollider();
  56. rect_b.minX = minX - circle.r;
  57. rect_b.minY = minY - circle.r;
  58. rect_b.maxX = maxX + circle.r;
  59. rect_b.maxY = maxY + circle.r;
  60.  
  61. //判断圆心是否在扩展矩形内
  62. if (!rect_b.CheckHit(circle.x, circle.y))
  63. return false;//无碰撞可能
  64.  
  65. //判断矩形左下角是否在圆内
  66. if (AtLeftBottom(circle.x, circle.y))
  67. return circle.CheckHit(minX, minY);
  68. //判断矩形左上角是否在圆内
  69. else if (AtLeftTop(circle.x, circle.y))
  70. return circle.CheckHit(minX, maxY);
  71. //判断矩形的右下角是否在圆内
  72. else if (AtRightBottom(circle.x, circle.y))
  73. return circle.CheckHit(maxX, minY);
  74. //判断矩形右上角是否在圆内
  75. else if (AtRightTop(circle.x, circle.y))
  76. return circle.CheckHit(maxX, maxY);
  77.  
  78. return true;
  79. }
  80.  
  81. //判断某个点是否在矩形的左下方
  82. public bool AtLeftBottom(int x, int y)
  83. {
  84. return x < minX && y < minY;
  85. }
  86.  
  87. //判断某个点是否在矩形的左上方
  88. public bool AtLeftTop(int x, int y)
  89. {
  90. return x < minX && y > maxY;
  91. }
  92.  
  93. //判断某个点是否在矩形的右下方
  94. public bool AtRightBottom(int x, int y)
  95. {
  96. return x > maxX && y < minY;
  97. }
  98.  
  99. //判断某个点是否在矩形的右上方
  100. public bool AtRightTop(int x, int y)
  101. {
  102. return x > maxX && y > maxY;
  103. }
  104. }
  105.  
  106. /// <summary>
  107. /// 圆形碰撞器
  108. /// </summary>
  109. public class CircleCollider
  110. {
  111. public int x;
  112. public int y;
  113. public int r;
  114.  
  115. //判断某点是否在圆内
  116. public bool CheckHit(int x, int y)
  117. {
  118. int d = (this.x - x) * (this.x - x) + (this.y - y) * (this.y - y);
  119. return d <= r * r;
  120. }
  121.  
  122. //判断圆形是否与矩形发生了碰撞
  123. public bool CheckHit(RectCollider rect)
  124. {
  125. return rect.CheckHit(this);
  126. }
  127.  
  128. //判断圆形是否与圆形发生了碰撞
  129. public bool CheckHit(CircleCollider circle)
  130. {
  131. int dx = x - circle.x;
  132. int dy = y - circle.y;
  133. int distSqr = dx * dx + dy * dy;
  134. int drSqr = (r + circle.r) * (r + circle.r);
  135. return distSqr < drSqr;
  136. }
  137. }
  138. }

标签: Algorithms

Powered by emlog  蜀ICP备18021003号-1   sitemap

川公网安备 51019002001593号