-
Notifications
You must be signed in to change notification settings - Fork 1
/
Engine.js
89 lines (87 loc) · 3.32 KB
/
Engine.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
function CreateVisited(){
var arr = new Array(SizeY);
for(var i=0;i<SizeY;i++) {
arr[i]=new Array(SizeX);
for(var j=0;j<SizeX;j++)
arr[i][j]=false;
}
return arr;
}
function AStar(test){
var time = new Date().getTime()/1000;
var visited = CreateVisited();
var open_list = new PriorityQueue({ comparator: function(a, b) {
if(a.cost+a.heu>b.cost+b.heu||(a.cost+a.heu==b.cost+b.heu&&a.cost>b.cost)) return 1;
return -1;
}});
visited[Ninja.Top][Ninja.Left]=1;
open_list.queue({ Top:Ninja.Top, Left:Ninja.Left, cost:0, heu:Heuristic(Ninja), parent:null });
while (open_list.length){
var current = open_list.dequeue();
visited[current.Top][current.Left]=1;
if(test==undefined||test==0) UI.GetCell(current.Top,current.Left).classList.add("visited");
if(Heuristic(current)==0){
console.log(new Date().getTime()/1000-time);
return ConstructPath(current);
}
var up = Clone(current);
var down = Clone(current);
var left = Clone(current);
var right = Clone(current);
up.Top--,down.Top++,left.Left--,right.Left++;
up.heu=Heuristic(up),down.heu=Heuristic(down),left.heu=Heuristic(left),right.heu=Heuristic(right);
up.parent=down.parent=left.parent=right.parent=current;
//if(isValid(wait)) open_list.queue(wait);
var uf=0, df=0, lf=0, rf=0;
for(var k=1;k<=1;k++){
if(!isValid(current,current.cost+k-1)) break;
up.cost=down.cost=left.cost=right.cost=current.cost+k;
if(isValid(up,up.cost)&&!visited[up.Top][up.Left]) uf=1,open_list.queue(Clone(up));
if(isValid(down,down.cost)&&!visited[down.Top][down.Left]) df=1,open_list.queue(Clone(down));
if(isValid(left,left.cost)&&!visited[left.Top][left.Left]) lf=1,open_list.queue(Clone(left));
if(isValid(right,right.cost)&&!visited[right.Top][right.Left]) rf=1,open_list.queue(Clone(right));
}
if(uf) visited[up.Top][up.Left]=1;
if(df) visited[down.Top][down.Left]=1;
if(lf) visited[left.Top][left.Left]=1;
if(rf) visited[right.Top][right.Left]=1;
}
console.log(new Date().getTime()/1000-time);
return false; // no path exists
}
function ConstructPath(node){
var path=[];
path.push(Clone(node));
do{
while(node.cost>node.parent.cost) path.push(Clone(node.parent)),node.cost--;
node = node.parent;
}
while(node.parent!=null);
while(node.cost>-1)path.push(Clone(node)),node.cost--;
return path.reverse();
}
function Heuristic(node){
return Math.abs(Coin.Top-node.Top)+Math.abs(Coin.Left-node.Left);
}
function isValid(node,cost){
cost+=UI.lastGuardPos;
if(node.Top<0||node.Top>=SizeX||node.Left<0||node.Left>=SizeY) return 0;
var sol=true;
for (var i = 0; i < Guards.length; i++) {
var length = Guards[i].length;
var index = GetMoveIndex(cost,length);
sol&=!AreEqual(Guards[i][index],node);
if(index>0)sol&=!AreEqual(Guards[i][index-1],node);
if(index<Guards[i].length-1)sol&=!AreEqual(Guards[i][index+1],node);
}
return sol&&Map[node.Top][node.Left];
}
function Clone(obj){
return {
Top: obj.Top,
Left: obj.Left,
heu: obj.heu,
cost: obj.cost,
parent: obj.parent
};
}