astar.cs 8.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217
  1. using System.Collections;
  2. using System.Collections.Generic;
  3. using UnityEngine;
  4. public class astar : MonoBehaviour
  5. {
  6. [Header("Grid Information")]
  7. public int cellDensity = 1;
  8. public float width;
  9. public float height;
  10. public Color gridBoundsVisualizeColor = Color.red;
  11. public GameObject cellGameObject;
  12. public Transform cellsParent;
  13. public cellData[,] cells;
  14. public cellData targetCell;
  15. public cellData startCell;
  16. void Start()
  17. {
  18. StartCoroutine(generateCells());
  19. }
  20. IEnumerator generateCells(){
  21. //Generate cells in grid
  22. float cellSize = cellGameObject.transform.localScale.x / (float)cellDensity;
  23. Vector2 cellStartPosition = new Vector2( transform.position.x - width /2f, transform.position.y - height /2f)
  24. + new Vector2(cellSize/2f, cellSize/2f);
  25. int horizontalCellCount = (int)(width / cellSize);
  26. int verticalCellCount = (int)(height / cellSize);
  27. cells = new cellData[horizontalCellCount,verticalCellCount];
  28. // cellData[,] cellMatrix = new cellData[horizontalCellCount, verticalCellCount];
  29. for(int x= 0; x < horizontalCellCount; x++){
  30. for(int y = 0; y < verticalCellCount; y++){
  31. Vector2 cellSpawnPos = cellStartPosition + new Vector2(cellSize * x, cellSize *y );
  32. GameObject cell = Instantiate(cellGameObject, cellSpawnPos , Quaternion.identity);
  33. cell.transform.localScale = cell.transform.localScale / (float)cellDensity;
  34. cell.GetComponent<cellData>().x = x;
  35. cell.GetComponent<cellData>().y = y;
  36. cell.transform.parent = cellsParent;
  37. cell.transform.name = "cell["+x+","+y+"]";
  38. //isObstacle?
  39. cell.GetComponent<cellData>().setObstacle(Physics2D.BoxCast(cell.transform.position, cell.transform.localScale/2f, 0 ,Vector2.zero));
  40. //cell.GetComponent<cellData>().setObstacle(Physics2D.CircleCast(cellSpawnPos,cellSize /2f,Vector2.zero));
  41. //cells.Add(cell.GetComponent<cellData>());
  42. cells[x,y] = cell.GetComponent<cellData>();
  43. yield return new WaitForEndOfFrame();
  44. }
  45. }
  46. //Add neighbours
  47. for(int x= 0; x < horizontalCellCount; x++){
  48. for(int y = 0; y < verticalCellCount; y++){
  49. for(int h = -1; h <= 1; h++){
  50. for(int v = -1; v <= 1; v++){
  51. if(h == 0 && v ==0){continue;}//I'm not a neighbour of me right?
  52. int targetX = x + h;
  53. int targetY = y + v;
  54. if(targetY <0 || targetY > verticalCellCount-1 || targetX < 0 || targetX > horizontalCellCount-1){continue;}
  55. if(cells[targetX,targetY].isObstacle){continue;}
  56. cells[x,y].neighbours.Add(cells[targetX,targetY]);
  57. }
  58. }
  59. }
  60. }
  61. }
  62. // Update is called once per frame
  63. Coroutine calculator = null;
  64. void Update()
  65. {
  66. if(Input.GetKeyDown(KeyCode.E)){
  67. Vector2 worldPos = Camera.main.ScreenToWorldPoint(Input.mousePosition);
  68. cellData closestCell = getClosestCell(worldPos);
  69. if(startCell!=null){startCell.setObstacle(startCell.isObstacle);}
  70. startCell = closestCell;
  71. startCell.GetComponent<SpriteRenderer>().color = Color.green;
  72. }else if(Input.GetKeyDown(KeyCode.F)){
  73. Vector2 worldPos = Camera.main.ScreenToWorldPoint(Input.mousePosition);
  74. if(targetCell!=null){targetCell.setObstacle(targetCell.isObstacle);}
  75. cellData closestCell = getClosestCell(worldPos);
  76. targetCell = closestCell;
  77. targetCell.GetComponent<SpriteRenderer>().color = Color.blue;
  78. }else if(Input.GetKeyDown(KeyCode.Return)){
  79. if(calculator != null){StopCoroutine(calculator); resetCellsCost();}
  80. Debug.Log("Starting calculation");
  81. calculator = StartCoroutine(Calculate());
  82. }
  83. }
  84. public cellData getClosestCell(Vector2 worldPos){
  85. float minDist = Mathf.Infinity;
  86. cellData closestCell = null;
  87. foreach(cellData cell in cells){
  88. if(cell.isObstacle){continue;}
  89. float DistToCell = Vector3.Distance(worldPos, cell.transform.position);
  90. if(DistToCell < minDist){
  91. minDist = DistToCell;
  92. closestCell = cell;
  93. }
  94. }
  95. return closestCell;
  96. }
  97. void OnDrawGizmos(){
  98. //Gizmos.Color = gridBoundsVisualizeColor;
  99. Gizmos.color = gridBoundsVisualizeColor;
  100. Gizmos.DrawWireCube(transform.position, new Vector3(width, height, 0));
  101. }
  102. List<cellData> calculatedCells = new List<cellData>();
  103. List<cellData> openCells = new List<cellData>();
  104. IEnumerator Calculate(){
  105. // calculatedCells = new List<cellData>();
  106. // bool foundPath = false;
  107. // openCells=new List<cellData>();
  108. // openCells.Add(startCell);
  109. // while(!foundPath){
  110. // cellData closeCell = null;
  111. // float lowestFcost = Mathf.Infinity;
  112. // foreach(cellData cell in openCells){
  113. // if(cell.fCost < lowestFcost){
  114. // closeCell=cell;
  115. // lowestFcost=cell.fCost;
  116. // }
  117. // }
  118. // if(closeCell == targetCell){Debug.Log("GOT THE TARGET!"); foundPath=true;}
  119. // openCells.Remove(closeCell);
  120. // calculateCellCost(closeCell);
  121. // // if(! calculateCellCost(closeCell)){Debug.Log("YOU DUMB!");yield return new WaitForSeconds(0.01f);continue;}
  122. // calculatedCells.Add(closeCell);
  123. // closeCell.setClose(true);
  124. // float minDist = Mathf.Infinity;
  125. // cellData nextCloseCell = null;
  126. // foreach(cellData cell in closeCell.neighbours){
  127. // if(cell == null){continue;}
  128. // if(cell.isClose){continue;}
  129. // calculateCellCost(cell);
  130. // if(cell.fCost < lowestFcost){
  131. // openCells.Add(cell);
  132. // }
  133. // }
  134. // if(nextCloseCell!=null){
  135. // }else{
  136. // Debug.Log("Hit a deadend, backtracking?");
  137. // }
  138. // // cellsToCalculate.Remove(cell);
  139. // yield return new WaitForSeconds(0.1f);
  140. // }
  141. List<cellData> open = new List<cellData>();
  142. List<cellData> close = new List<cellData>();
  143. open.Add(startCell);
  144. bool foundPath = false;
  145. while(!foundPath){
  146. cellData current = null;
  147. float lowestF = Mathf.Infinity;
  148. foreach(cellData cell in open){
  149. if(cell.fCost < lowestF){
  150. lowestF = cell.fCost;
  151. current = cell;
  152. }
  153. }
  154. open.Remove(current);
  155. close.Add(current);
  156. current.setClose(true);
  157. if(current == targetCell){foundPath=true; continue;}
  158. foreach(cellData neighbour in current.neighbours){
  159. if(close.Contains(neighbour)){continue;}
  160. bool hasThisNeighbour = open.Contains(neighbour);
  161. if(neighbour.fCost < lowestF || !hasThisNeighbour){
  162. calculateCellCost(neighbour);
  163. if(neighbour.fCost < lowestF || neighbour.parent==null){ neighbour.parent = current;}
  164. if(!hasThisNeighbour){open.Add(neighbour);}
  165. }
  166. }
  167. yield return new WaitForSeconds(0.05f);
  168. }
  169. //sort
  170. bool pathCleared = false;
  171. cellData parentCell = targetCell;
  172. while(!pathCleared){
  173. if(parentCell == startCell){pathCleared=true; break;}
  174. parentCell.GetComponent<SpriteRenderer>().color =Color.blue;
  175. parentCell = parentCell.parent;
  176. yield return new WaitForSeconds(0.1f);
  177. }
  178. }
  179. bool calculateCellCost(cellData cell){
  180. if((int)cell.fCost!=0){Debug.Log("already calculated, skipping");return false;}
  181. float hCost = Vector2.Distance(cell.transform.position, targetCell.transform.position);
  182. float gCost = Vector2.Distance(cell.transform.position, startCell.transform.position);
  183. float fCost = hCost + gCost;
  184. cell.setCellCost(gCost, hCost, fCost);
  185. return true;
  186. }
  187. public void resetCellsCost(){
  188. foreach(cellData cell in cells){
  189. cell.resetCost();
  190. }
  191. }
  192. }