astar_vertex.cs 7.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204
  1. using System;
  2. using System.Collections;
  3. using System.Collections.Generic;
  4. using UnityEngine;
  5. public class astar_vertex : MonoBehaviour
  6. {
  7. public Transform VertexGroup;
  8. [SerializeField]
  9. public List<Vertex> nodes;
  10. Vertex startNode;
  11. Vertex endNode;
  12. public Transform startVisualizer;
  13. public Transform endVisualizer;
  14. public PathWalker walker;
  15. public List<Vertex> finalPath;
  16. void Start()
  17. {
  18. finalPath = new List<Vertex>();
  19. nodes = new List<Vertex>();
  20. foreach(Vertex vertex in VertexGroup.GetComponentsInChildren<Vertex>()){
  21. NodeData nodeData = vertex.gameObject.AddComponent(typeof(NodeData)) as NodeData;
  22. nodes.Add(vertex);
  23. }
  24. }
  25. // Update is called once per frame
  26. void Update()
  27. {
  28. if(startNode != null){Debug.DrawLine(startVisualizer.position, startNode.transform.position);}
  29. if(endNode != null){Debug.DrawLine(endVisualizer.position, endNode.transform.position);}
  30. if(Input.GetKeyDown(KeyCode.E)){
  31. Vector2 startPos = Camera.main.ScreenToWorldPoint(Input.mousePosition);
  32. startVisualizer.position = startPos;
  33. //get closest and reachable node to this location
  34. float minDist = Mathf.Infinity;
  35. foreach(Vertex node in nodes){
  36. if(Physics2D.Linecast(startPos, node.transform.position)){}else{
  37. float distToNode = Vector2.Distance(node.transform.position, startPos);
  38. if(distToNode < minDist){
  39. minDist = distToNode;
  40. startNode = node;
  41. }
  42. }
  43. }
  44. }else if(Input.GetKeyDown(KeyCode.F)){
  45. Vector2 endPos = Camera.main.ScreenToWorldPoint(Input.mousePosition);
  46. endVisualizer.position = endPos;
  47. //get closest and reachable node to this location
  48. float minDist = Mathf.Infinity;
  49. foreach(Vertex node in nodes){
  50. if(Physics2D.Linecast(endPos, node.transform.position)){}else{
  51. float distToNode = Vector2.Distance(node.transform.position, endPos);
  52. if(distToNode < minDist){
  53. minDist = distToNode;
  54. endNode = node;
  55. }
  56. }
  57. }
  58. }else if(Input.GetKeyDown(KeyCode.Return)){
  59. //Start Calculation
  60. StartCoroutine(Calculate());
  61. }else if(Input.GetKeyDown(KeyCode.I)){
  62. Recalculate();
  63. }else if(Input.GetKeyDown(KeyCode.S)){
  64. Recalculate();
  65. walker.pathNodes = new List<Transform>();
  66. walker.curIndex = 0;
  67. walker.pathNodes.Add(startNode.transform);
  68. for(int i = finalPath.Count-1; i >=0; i--){
  69. walker.pathNodes.Add(finalPath[i].transform);
  70. }
  71. walker.pathNodes.Add(endVisualizer);
  72. }
  73. if(finalPath.Count > 1){
  74. for(int i = 1; i < finalPath.Count; i++){
  75. Debug.DrawLine(finalPath[i].transform.position, finalPath[i-1].transform.position, Color.green);
  76. }
  77. }
  78. }
  79. public void Recalculate(){
  80. DateTime startTime = DateTime.Now;
  81. foreach(Vertex node in nodes){
  82. node.GetComponent<NodeData>().Reset();
  83. }
  84. List<Vertex> open = new List<Vertex>();
  85. List<Vertex> close = new List<Vertex>();
  86. open.Add(startNode);
  87. bool foundPath = false;
  88. while(!foundPath){
  89. Vertex current = null;
  90. float lowestF = Mathf.Infinity;
  91. foreach(Vertex cell in open){
  92. if(cell.GetComponent<NodeData>().fCost < lowestF){
  93. lowestF = cell.GetComponent<NodeData>().fCost;
  94. current = cell;
  95. }
  96. }
  97. open.Remove(current);
  98. close.Add(current);
  99. if(current == endNode){foundPath=true; continue;}
  100. foreach(Vertex neighbour in current.neighbours){
  101. if(close.Contains(neighbour)){continue;}
  102. bool hasThisNeighbour = open.Contains(neighbour);
  103. if(neighbour.GetComponent<NodeData>().fCost < lowestF || !hasThisNeighbour){
  104. calculateCellCost(neighbour.GetComponent<NodeData>());
  105. if(neighbour.GetComponent<NodeData>().fCost < lowestF || neighbour.GetComponent<NodeData>().parent==null){ neighbour.GetComponent<NodeData>().parent = current;}
  106. if(!hasThisNeighbour){open.Add(neighbour);}
  107. }
  108. }
  109. // yield return new WaitForSeconds(0.05f);
  110. }
  111. //sort
  112. bool pathCleared = false;
  113. Vertex parentCell = endNode;
  114. finalPath= new List<Vertex>();
  115. while(!pathCleared){
  116. if(parentCell == startNode){pathCleared=true; break;}
  117. finalPath.Add(parentCell);
  118. parentCell = parentCell.GetComponent<NodeData>().parent;
  119. // yield return new WaitForSeconds(0.1f);
  120. }
  121. Debug.Log("Calculation finished in " + (DateTime.Now - startTime).TotalMilliseconds + "ms");
  122. }
  123. IEnumerator Calculate(){
  124. DateTime startTime = DateTime.Now;
  125. foreach(Vertex node in nodes){
  126. node.GetComponent<NodeData>().Reset();
  127. }
  128. List<Vertex> open = new List<Vertex>();
  129. List<Vertex> close = new List<Vertex>();
  130. open.Add(startNode);
  131. bool foundPath = false;
  132. while(!foundPath){
  133. Vertex current = null;
  134. float lowestF = Mathf.Infinity;
  135. foreach(Vertex cell in open){
  136. if(cell.GetComponent<NodeData>().fCost < lowestF){
  137. lowestF = cell.GetComponent<NodeData>().fCost;
  138. current = cell;
  139. }
  140. }
  141. open.Remove(current);
  142. close.Add(current);
  143. if(current == endNode){foundPath=true; continue;}
  144. foreach(Vertex neighbour in current.neighbours){
  145. if(close.Contains(neighbour)){continue;}
  146. bool hasThisNeighbour = open.Contains(neighbour);
  147. if(neighbour.GetComponent<NodeData>().fCost < lowestF || !hasThisNeighbour){
  148. calculateCellCost(neighbour.GetComponent<NodeData>());
  149. if(neighbour.GetComponent<NodeData>().fCost < lowestF || neighbour.GetComponent<NodeData>().parent==null){ neighbour.GetComponent<NodeData>().parent = current;}
  150. if(!hasThisNeighbour){open.Add(neighbour);}
  151. }
  152. }
  153. // yield return new WaitForSeconds(0.05f);
  154. }
  155. //sort
  156. bool pathCleared = false;
  157. Vertex parentCell = endNode;
  158. finalPath= new List<Vertex>();
  159. while(!pathCleared){
  160. if(parentCell == startNode){pathCleared=true; break;}
  161. finalPath.Add(parentCell);
  162. parentCell = parentCell.GetComponent<NodeData>().parent;
  163. yield return new WaitForSeconds(0.1f);
  164. }
  165. Debug.Log("Calculation finished in " + (DateTime.Now - startTime).Milliseconds + "ms");
  166. }
  167. bool calculateCellCost(NodeData cell){
  168. if((int)cell.fCost!=0){return false;}
  169. cell.hCost = Vector2.Distance(cell.transform.position, endNode.transform.position);
  170. cell.gCost = Vector2.Distance(cell.transform.position, startNode.transform.position);
  171. cell.fCost = cell.hCost + cell.gCost;
  172. return true;
  173. }
  174. void OnValidate(){
  175. if(VertexGroup !=null && VertexGroup.GetComponentInChildren<Vertex>()==null){VertexGroup = null; Debug.LogError("Please assigna a vertex group with vertecies as children");}
  176. }
  177. }