Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/andrewcomputsci2019/pathfinderfx
JavaFx Program that visualizes serval pathfinding and maze generation algorithms
https://github.com/andrewcomputsci2019/pathfinderfx
Last synced: 18 days ago
JSON representation
JavaFx Program that visualizes serval pathfinding and maze generation algorithms
- Host: GitHub
- URL: https://github.com/andrewcomputsci2019/pathfinderfx
- Owner: andrewcomputsci2019
- License: mit
- Created: 2023-07-31T03:40:29.000Z (over 1 year ago)
- Default Branch: master
- Last Pushed: 2023-09-23T00:32:09.000Z (over 1 year ago)
- Last Synced: 2024-11-05T23:06:15.720Z (2 months ago)
- Language: Java
- Homepage:
- Size: 276 KB
- Stars: 0
- Watchers: 1
- Forks: 0
- Open Issues: 0
-
Metadata Files:
- Readme: README.md
Awesome Lists containing this project
README
# Pathfinding Algorithm Visualizer
___
A program inspired by the design of [Francisco Zacarias own pathfinder](https://github.com/FranciscoZacarias/Algorithm-Visualizer-JavaFX), written from the ground up using Java and JavaFX 17.
The program supports a verity of four directional [pathfinding](#supported-pathfinding-algorithms) and [maze](#supported-maze-generation-algorithms) generation algorithms
___
## How to Run
The preferred way to run the program is to clone the source code and import it into a Gradle build-supported
I.D.E, like IntelliJ, and use it to run and build the program. Otherwise, you can use the provided gradle wrapper
to execute and build the program as shown below.
```shell
##cd into project root dir with the gradlew and gradlew.bat file
##if on Windows run the gradlew.bat file
##change /path_to_jdk_directory to your jdk distribution
./gradlew -Dorg.gradle.java.home=/path_to_jdk17_directory
./gradlew run
```
___
## How to Use
The left side of the screen is the program control section and allows the user
to change the grid size, modify what cell type is selected, change what pathfinding algorithm
is being utilized and heuristic if applicable, create mazes using different algorithms, and finally the ability to start
and skip visualization of the current pathfinding algorithm.\
To the right of the program control sidebar is the grid. The grid allows the user to drag their mouse to add or remove cells
to the grid. Holding/pressing left click will add cell/cells of the selected cell type and holding/pressing right click will remove cell/cells. Control right
mouse click on the grid will bring up a context menu that provides some fine-tuning, such as accelerating/decelerating the animation speed,
and the ability to run the garbage collector to free up excess memory the JVM is holding onto. The program only allows
a single source and a single target to be placed on the grid, if the user wants to change their locations they must first remove them from the grid and
replace them in the desired position.
___
## Demo Video
[![video](https://i.ytimg.com/vi/Tt1ZOXGZ7Vw/maxresdefault.jpg?sqp=-oaymwEmCIAKENAF8quKqQMa8AEB-AH-CYAC0AWKAgwIABABGEsgWihlMA8=&rs=AOn4CLCyn4IBs5N2bEZW-IyPYhgaom-rlg)](https://www.youtube.com/watch?v=Tt1ZOXGZ7Vw)
___
## Supported Pathfinding Algorithms
* [A*](#a-a-star)(Supports Weights)
* Manhattan Heuristic
* Euclidean Heuristic
* Chebyshev Heuristic
* [Breadth First Search](#breadth-first-search-bfs) (BFS)
* [Breadth First Search Greedy](#breadth-first-search-greedy)
* Manhattan Heuristic
* Euclidean Heuristic
* Chebyshev Heuristic
* [Depth First Search (DFS)](#depth-first-search-dfs)
* [Dijkstra](#dijkstra) (only works for weighted graphs)
* [Wave Front Propagation](#wave-front-propagation)
___
### A* (A-star)
[A*](https://en.wikipedia.org/wiki/A*_search_algorithm) is a single pair pathfinding algorithm that leverages heuristics to help it efficiently search a weighted graph to find a near-optimal or optimal path solution\
This program implementation allows the support of three different Heuristics: Manhattan, Euclidean, and Chebyshev\
Pseudocode
```
h(x):double = ranking heuristic
path(node,precessor):List
list = []
while(node != predecessor[node]):
list.add(node)
node = predecessor[node]
return list
A*(start, goal):
visitedNodes = []
predecessor = []
searchQueue = {[start,start,0.0],0.0} //min heap
occurredCost = []
while(searchQueue is not empty):
node = searchQueue.get()[0]
predecessor[node] = searchQueue.get()[1]
double weight = searchQueue.remove()[2]
occurredCost[node] = weight
visitedNodes[node] = true
for(nodes in AdjcentCells(node)):
if(nodes is goal):
return path(node)
searchQueue.add([nodes,node,weight+nodes.weight],h(nodes))
```
***demo***https://github.com/andrewcomputsci2019/PathFinderFX/assets/54915639/a0f9ef10-801a-4851-a9ef-044abecc01c3
___
### Breadth First Search (BFS)
[BFS](https://en.wikipedia.org/wiki/Breadth-first_search#Bias_towards_nodes_of_high_degree) is a graph traversal and single source pathfinding algorithm, that expands in respect of the nodes added to the search, or in other words it explores
all adjacent cells in order before moving on to other cells. Given that BFS expands respectfully of order, it guarantees that the shortest single source path is found in an unweighted graph\
Pseudo Code
```
path(node,predecessor):List
list = []
while(node != predecessor[node]):
list.add(node)
node = predecessor[node]
return list
BFS(start, goal):
searchQueue = []
visited = []
searchQueue.add(start)
predecessor = []
while(searchQueue is not empty):
visited[searchQueue.get()] = true;
node = searchQueue.remove()
for(nodes in AdjcentCells(node)):
if(nodes is goal):
return path(node)
predecessor[nodes] = node
visited[nodes] = true;
searchQueue.add(nodes)
```***demo***
[![video](https://i.ytimg.com/vi/8cu3DMnLPtk/maxresdefault.jpg?sqp=-oaymwEmCIAKENAF8quKqQMa8AEB-AH-CYAC0AWKAgwIABABGEsgWihlMA8=&rs=AOn4CLCyn4IBs5N2bEZW-IyPYhgaom-rlg)](https://www.youtube.com/watch?v=8cu3DMnLPtk)
___
### BREADTH FIRST SEARCH GREEDY
[BFS Greedy](https://en.wikipedia.org/wiki/Best-first_search#Greedy_BFS) is a variation of BFS that uses a heuristic function to evaluate adjacent cells to figure out what order
to visit them in. BFS Greedy in most variations uses some sort of min heap or similar structure to handle the order of cell
traversal.\
Pseudocode
```
h(x):double = ranking heuristic
path(node,predecessor):List
list = []
while(node != predecessor[node]):
list.add(node)
node = predecessor[node]
return list
BFS(start, goal):
searchQueue = [] //min-heap
visited = []
searchQueue.add(start)
predecessor = []
predecessor[start] = start
while(searchQueue is not empty):
visited[searchQueue.get()] = true;
node = searchQueue.remove()
for(nodes in AdjcentCells(node)):
if(nodes is goal):
return path(node)
predecessor[nodes] = node
visited[nodes] = true;
searchQueue.add(nodes,h(nodes))
```
https://github.com/andrewcomputsci2019/PathFinderFX/assets/54915639/ffdbbd7e-f540-4077-a1ff-edb261e78119
___### DEPTH FIRST SEARCH (DFS)
[DFS](https://en.wikipedia.org/wiki/Depth-first_search) similar to BFS is a graph traversal that expands only a single search path until a dead end is met, where it then back-tracks
to an unvisited node and continues this process until all nodes have been searched or the goal has been found. This algorithm does
not guarantee to find the shortest path.\
Pseudocode
```
path(node,predecessor):List
list = []
while(node != predecessor[node]):
list.add(node)
node = predecessor[node]
return list
DFS(start, goal):
predecessor = []
predecessor[start] = start
stack = []
visited = []
stack.add(start)
vistied[start] = true
while(stack is not empty):
node = stack.pop()
for(nodes in AdjcentCells(node)):
if(nodes == goal):
return path(node)
predecessor[nodes] = node
stack.push(nodes)
visited[nodes] = true
```
***demo***
[![video](https://i.ytimg.com/vi/6G0ewv1wMW4/maxresdefault.jpg?sqp=-oaymwEmCIAKENAF8quKqQMa8AEB-AH-CYAC0AWKAgwIABABGEsgWihlMA8=&rs=AOn4CLCyn4IBs5N2bEZW-IyPYhgaom-rlg)](https://www.youtube.com/watch?v=6G0ewv1wMW4)
___
### Dijkstra
[Dijkstra](https://en.wikipedia.org/wiki/Dijkstra%27s_algorithm) is a weighted graph single source pathfinding algorithm that **guarantees** to find the shortest path. This algorithm
leverages a greedy process in which it picks the shortest edge plus occurred weight from the list of current queued cells, it repeats this process
until the goal has been found or all cells have been searched\
Pseudocode
```
path(node,predecessor): List
list = []
while(node != predecessor[node]):
list.add(node)
node = predecessor[node]
return listDijkstra(start, goal):
visited = []
queue = [start] //min-heap
occurredCost = []
occurredCost[start] = 0
predecessor = []
predecessor[start] = start
while(queue is not empty):
node = queue.remove()
for(nodes in AdjcentCells(node):
if(nodes equals goal):
return path(node)
occurredCost[nodes] = occurredCost[node]+nodes.weigth
queue.add(nodes, occurredCost[nodes])
```
***demo***
[![video](https://i.ytimg.com/vi/LeJfMaz5edY/maxresdefault.jpg?sqp=-oaymwEmCIAKENAF8quKqQMa8AEB-AH-CYAC0AWKAgwIABABGEsgWihlMA8=&rs=AOn4CLCyn4IBs5N2bEZW-IyPYhgaom-rlg)](https://www.youtube.com/watch?v=LeJfMaz5edY)
___
### Wave Front Propagation
[Wave Front Propagation](https://en.wikipedia.org/wiki/Wavefront_expansion_algorithm) works like BFS in the way it respects the order of cells as they are added but differs as it starts from the target and expands outward.
It searches the entire grid assigning a number to each cell representing the number of steps required to reach that cell. Once all cells have been searched the algorithm
then chooses the least weighted cell around the source and repeats until the target is reached guaranteeing the shortest path.\
Pseudocode
```
path(start,goal,cost[]):List
path = []
curr = start
while(curr not null):
path.add(curr)
if(curr equals goal) break
min = null
minVal = Infinity
for(nodes in AdjcentCells(curr)):
minVal = Math.min(cost[nodes],minVal)
if(minVal equals cost[nodes]):
min = nodes
return path
WaveFront(start, goal):
visted = []
queue = [goal]
visted[start] = true
cost = []
cost[goal] = 0
while(queue is not empty):
node = queue.remove()
currentCost = cost[node]
for(nodes in AdjcentCells(node)):
cost[nodes] = currentCost+1
queue.add(nodes)
visited[nodes] = true
return path(start,goal,cost)
```
***demo***
[![video](https://i.ytimg.com/vi/JSsE4OVgRfI/maxresdefault.jpg?sqp=-oaymwEmCIAKENAF8quKqQMa8AEB-AH-CYAC0AWKAgwIABABGEsgWihlMA8=&rs=AOn4CLCyn4IBs5N2bEZW-IyPYhgaom-rlg)](https://www.youtube.com/watch?v=JSsE4OVgRfI)
___
## Supported Maze Generation Algorithms
* [Binary Tree](#binary-tree)
* [Depth First Search Randomized Backtracking](#depth-first-search-randomized-backtracking)
* [Kruskal](#kruskal)
* [Hunt and Kill](#hunt-and-kill)
* [Prims randomized](#prims-randomized)
___
### Binary Tree
[Binary Tree Maze](https://weblog.jamisbuck.org/2011/2/1/maze-generation-binary-tree-algorithm) generation algorithm is a simple and stateless way of creating a perfect maze, such that only a single path
connects any two cells. The algorithm functions by iterating over all cells and making a binary decision of a direction to connect
two particular cells together. The implementation in this program has the binary choices of north or east.\
Pseudo Code
```
BinaryTree(list{nodes}):
for(cells in list):
node = getNorthOrWestAdjcentCell()
join(cells,node)
```
***demo***
[![video](https://i.ytimg.com/vi/3ETcPLxgpZ0/maxresdefault.jpg?sqp=-oaymwEmCIAKENAF8quKqQMa8AEB-AH-CYAC0AWKAgwIABABGEsgWihlMA8=&rs=AOn4CLCyn4IBs5N2bEZW-IyPYhgaom-rlg)](https://www.youtube.com/watch?v=3ETcPLxgpZ0)
___
### Depth First Search Randomized Backtracking
[Depth First Search Randomized Backtracking](https://en.wikipedia.org/wiki/Maze_generation_algorithm#Randomized_depth-first_search) is an extension of the DFS algorithm in which cells are traversed randomly
using the DFS algorithm. The process starts by choosing any cell randomly in the grid as the starting point and then choosing
a random unvisited neighboring cell removing the wall between them and adding it to the stack. The last step in this process is to remove
any cell from the stack once it has no unvisited neighboring cell remaining. The maze is generated once all cells have been removed from the stack\
Pseudocode
```
DFSRandomized(list{nodes}):
stack = []
visited = []
stack = random cell from list
while(stack is not empty):
node = randomAdjcentCell(stack.peek()) not in visited
if node is null:
stack.pop()
visited[node] = true
join(stack.peek(),node)
stack.push(node)
```
***demo***
[![video](https://i.ytimg.com/vi/EseRl5cTdNk/maxresdefault.jpg?sqp=-oaymwEmCIAKENAF8quKqQMa8AEB-AH-CYAC0AWKAgwIABABGEsgWihlMA8=&rs=AOn4CLCyn4IBs5N2bEZW-IyPYhgaom-rlg)](https://www.youtube.com/watch?v=EseRl5cTdNk)
___
### Kruskal
[Kruskal](https://en.wikipedia.org/wiki/Kruskal%27s_algorithm) is a [minimum spanning tree](https://en.wikipedia.org/wiki/Minimum_spanning_tree)(MST)
algorithm that can be utilized to create a perfect maze. The algorithm starts by first creating a set for each cell that contains what cell their connected to.
It then randomly selects an edge between two cells and checks whether their sets contain each other if so the edge is skipped otherwise the two cells are then
connected and their sets updated accordingly. This process continues till all edges have been exhausted, at which point a perfect maze has been created\
Pseudo Code
```
Kruskal(nodes[]):
edges = [all edges in nodes] //add all edges to a set
pathSet = {node in nodes -> node : [node]} //create path set for all nodes with only that node in it to start
while(edges is not empty):
node1, node2 = get random edge and remove from edges
if(node2 not in pathSet[node1]):
join(node2, node1)
pathSet[node2] uion pathSet[node1] //update both sets to contain each other connected nodes
//once all edges have been iterated over return
```
***demo***
[![video](https://i.ytimg.com/vi/ZzaqWHcavBU/maxresdefault.jpg?sqp=-oaymwEmCIAKENAF8quKqQMa8AEB-AH-CYAC0AWKAgwIABABGEsgWihlMA8=&rs=AOn4CLCyn4IBs5N2bEZW-IyPYhgaom-rlg)](https://www.youtube.com/watch?v=ZzaqWHcavBU)
___
### Hunt and Kill
Hunt and Kill algorithm is a [random walk](https://en.wikipedia.org/wiki/Random_walk) algorithm that guarantees a perfect maze when it finishes. The algorithm starts
by first picking a random cell to be the starting point of the first random walk. The program then random walks across the grid
and stops once no more steps can be taken, i.e. it's been boxed in. The algorithm then scans across the grid to find a cell to be the next
starting point of the random walk, this cell must be adjacent to an already visited cell. The algorithm then random walks and repeats the process
above until no valid starting points remain.\
Pseudo Code
````
hunt(nodes[]):
start = random node from nodes
visted = []
visited[start] = true
kill(start,visited)
while(true):
found = false
for(node in nodes and not in visited):
if(node has visited neighbor):
join(node, random visited neighbor)
walk(node,visited)
found = true
break
if(!found):
break
kill(node start, visited[]):
visited[start] = true;
cells
while((cells = list of unvisted neighbors from start) is not empty):
cell = get random node from cells
visited[cell] = true
join(start,cell)
start = cell
````
***demo***
[![video](https://i.ytimg.com/vi/55qawcown6k/maxresdefault.jpg?sqp=-oaymwEmCIAKENAF8quKqQMa8AEB-AH-CYAC0AWKAgwIABABGEsgWihlMA8=&rs=AOn4CLCyn4IBs5N2bEZW-IyPYhgaom-rlg)](https://www.youtube.com/watch?v=55qawcown6k)
___
### Prims Randomized
Prims Randomized is another MST based algorithm that can be used to create a perfect maze. The algorithm starts by
first picking a random cell in the grid. It then finds all the so-called frontier cells, which are cells that are adjacent but separated by a cell
from the current cell and adds them to the set of potential future cells to visit. Prims then randomly picks one of the cells from the future cells set, and then
finds a neighboring cell that has already been visited and joins them together. It then adds its frontier cells to the future cell set and repeats till
no cells remain.\
Pseudo Code
```
PrimsRandomized(nodes[]):
frontierCells = []
visited = []
start = random node from nodes[]
visited = [start]
frontierCells.add( get frontier cells from start)
while(frontierCells is not empty):
randomFrontier = random node from frontierCells
validVisitedNode = get random valid visited neighbor
join(randomFrontier,validVisitedNode)
frontierCells.add( get frontier cells from randomFrontier)
```
***demo***
[![video](https://i.ytimg.com/vi/J17v-zOE9Xo/maxresdefault.jpg?sqp=-oaymwEmCIAKENAF8quKqQMa8AEB-AH-CYAC0AWKAgwIABABGEsgWihlMA8=&rs=AOn4CLCyn4IBs5N2bEZW-IyPYhgaom-rlg)](https://www.youtube.com/watch?v=J17v-zOE9Xo)
___
## Utilized Libraries
* [ControlsFX](https://github.com/controlsfx/controlsfx)
* Validation support
* [ikonli Icons](https://github.com/kordamp/ikonli)
* BootStrap Icons
* [atlantafx](https://github.com/mkpaz/atlantafx)
* Primer Light