mirror of
https://github.com/nzp-team/quakec.git
synced 2025-04-05 09:21:28 +00:00
Adds traversal-based navmesh pathfinding to SSQC
Adds proof-of-concept navmesh + traversal use to AI_Chase
This commit is contained in:
parent
cf0ec43631
commit
992c1137da
4 changed files with 656 additions and 536 deletions
|
@ -2245,7 +2245,7 @@ float cl_pathfind_get_lowest_f_score_poly() {
|
|||
}
|
||||
|
||||
|
||||
//Returns the traversal with the lowest f score from polygons the open set
|
||||
//Returns the traversal with the lowest f score from traversals the open set
|
||||
float cl_pathfind_get_lowest_f_score_traversal() {
|
||||
//TODO: implement a better algorithm for finding the lowest score
|
||||
float best_score = 100000;
|
||||
|
@ -2407,9 +2407,6 @@ void cl_pathfind_smooth_path(vector start_point, vector goal_point) {
|
|||
float cur_funnel_left_portal_idx = 0;
|
||||
float cur_funnel_right_portal_idx = 0;
|
||||
|
||||
float next_portal_left_in_funnel;
|
||||
float next_portal_right_in_funnel;
|
||||
|
||||
// Loop through all portals, adding necessary funnel corners to final path
|
||||
for(float i = 0; i < cl_test_pathfind_result->portals_length; i++) {
|
||||
cur_portal_left_pos = cl_test_pathfind_result->portals_left_pos[i];
|
||||
|
@ -2519,456 +2516,11 @@ void cl_pathfind_smooth_path(vector start_point, vector goal_point) {
|
|||
cl_test_pathfind_result->point_path_points[cl_test_pathfind_result->point_path_length] = goal_point;
|
||||
cl_test_pathfind_result->point_path_traversals[cl_test_pathfind_result->point_path_length] = -1;
|
||||
cl_test_pathfind_result->point_path_length++;
|
||||
|
||||
|
||||
// TODO - Rewrite this algorithm:
|
||||
// we have a cur funnel composed of the corner, and the portal
|
||||
// we have a cur portal which is a left edge and a right edge
|
||||
// if cur portal is narrower, update the current funnel
|
||||
// if cur portal is wider, keep the current funnel
|
||||
// if cur portal is outside of funnel, add funnel right or left to list of path points
|
||||
// if cur portal is a traversal start, add traversal start to list of points, denote that its a traversal
|
||||
// continue from traversal end point, setting exit as corner, and getting the next portal from there...
|
||||
|
||||
}
|
||||
|
||||
// void cl_pathfind_smooth_path(vector start_point, vector goal_point) {
|
||||
// //Evaluate portals and identify left / right portal vertices
|
||||
// for(float i = 1; i < cl_test_pathfind_result->result_node_length; i++) {
|
||||
// float prev_poly = cl_test_pathfind_result->result_node_path[i-1];
|
||||
// float poly = cl_test_pathfind_result->result_node_path[i];
|
||||
|
||||
// // Find what index prev_poly is linked to poly
|
||||
// float link_index = -1;
|
||||
// for(float j = 0; j < cl_navmesh_polies[prev_poly].connected_polies_count; j++) {
|
||||
// if(cl_navmesh_polies[prev_poly].connected_polies[j] == poly) {
|
||||
// link_index = j;
|
||||
// break;
|
||||
// }
|
||||
// }
|
||||
|
||||
// //Use that index to get left and right vertices
|
||||
// cl_test_pathfind_result->portals_left_vert[cl_test_pathfind_result->portals_length] = cl_navmesh_polies[prev_poly].connected_polies_left_vert[link_index];
|
||||
// cl_test_pathfind_result->portals_right_vert[cl_test_pathfind_result->portals_length] = cl_navmesh_polies[prev_poly].connected_polies_right_vert[link_index];
|
||||
// cl_test_pathfind_result->portals_length++;
|
||||
// }
|
||||
|
||||
// // The last portal's left and right edges are both the goal position
|
||||
// print("Portals: ");
|
||||
// for(float i = 0; i < cl_test_pathfind_result->portals_length; i++) {
|
||||
// print("[");
|
||||
// print(ftos(i));
|
||||
// print("] = (");
|
||||
// print(ftos(cl_test_pathfind_result->portals_left_vert[i]));
|
||||
// print(" , ");
|
||||
// print(ftos(cl_test_pathfind_result->portals_right_vert[i]));
|
||||
// print(") , ");
|
||||
// }
|
||||
|
||||
// //starting at start_pos (not at start center point)
|
||||
// //get vectors that point to first left edge and first right edge
|
||||
// vector funnel_apex = start_point;
|
||||
// float funnel_left_index = 0;
|
||||
// //Index of the funnel's left vertex in the portal list
|
||||
// vector funnel_left = cl_pathfind_get_left_portal_corner(cl_test_pathfind_result, 0);
|
||||
// //Index of the funnel's right vertex in the portal list
|
||||
// float funnel_right_index = 0;
|
||||
// vector funnel_right = cl_pathfind_get_right_portal_corner(cl_test_pathfind_result, 0);
|
||||
|
||||
// vector next_funnel_left = '0 0 0';
|
||||
// vector next_funnel_right = '0 0 0';
|
||||
// vector corner;
|
||||
|
||||
// float inside_right_edge;
|
||||
// float inside_left_edge;
|
||||
|
||||
// while(1) {
|
||||
// print("=============Funnel iteration.==========\n");
|
||||
// print("\tCurrent left edge: ");
|
||||
// print(ftos(cl_test_pathfind_result->portals_left_vert[funnel_left_index]));
|
||||
// print("\n");
|
||||
// print("\tCurrent right edge: ");
|
||||
// print(ftos(cl_test_pathfind_result->portals_right_vert[funnel_right_index]));
|
||||
// print("\n");
|
||||
|
||||
// //Check if we have reached the end of the portals, consider the goal position as the last portal
|
||||
// //If we are not at the last index of the left portal
|
||||
|
||||
// //Keeping track of whether or not advancing the left edge or right edge narrows the funnel
|
||||
// float advanced_left = FALSE;
|
||||
// float advanced_right = FALSE;
|
||||
|
||||
// //Consider the end goal point as the last portal
|
||||
// //================ Checking left funnel edge =================
|
||||
// if(funnel_left_index < cl_test_pathfind_result->portals_length) {
|
||||
// print("\t-- Trying to advance left edge. --\n");
|
||||
|
||||
// if(funnel_left_index < cl_test_pathfind_result->portals_length - 1) {
|
||||
// // next_funnel_left = cl_navmesh_verts[cl_test_pathfind_result->portals_left_vert[funnel_left_index+1]].pos;
|
||||
// next_funnel_left = cl_pathfind_get_left_portal_corner(cl_test_pathfind_result, funnel_left_index + 1);
|
||||
// print("\t\tNext left edge is vert: ");
|
||||
// print(ftos(cl_test_pathfind_result->portals_left_vert[funnel_left_index+1]));
|
||||
// print(".\n");
|
||||
// }
|
||||
// else {
|
||||
// print("\t\tTrying next left edge as goal point.\n");
|
||||
// //If funnel_left is pointing to the last portal in the array, consider the goal_point the last portal
|
||||
// next_funnel_left = goal_point;
|
||||
// }
|
||||
|
||||
// inside_right_edge = pathfind_point_is_to_left(funnel_apex,funnel_right,next_funnel_left);
|
||||
// //If the next left edge crosses the current right edge
|
||||
// if(inside_right_edge < 0) {
|
||||
// print("\t\tnext left edge crosses current right edge.\n");
|
||||
// //Add funnel right point to path
|
||||
// // cl_test_pathfind_result->result_path[cl_test_pathfind_result->result_length++] = funnel_right;
|
||||
// corner = funnel_right;
|
||||
// // // ------------------------------------------------------------
|
||||
// // // Instead of adding the funnel's corner to the path,
|
||||
// // // move away from the corner by some small amount
|
||||
// // // ------------------------------------------------------------
|
||||
// // // Find the first portal along the path with this corner
|
||||
// // int portal_idx = funnel_right_index;
|
||||
// // for(int i = funnel_right_index; i >= 0; i--) {
|
||||
// // if(cl_test_pathfind_result->portals_right_vert[i] == cl_test_pathfind_result->portals_right_vert[funnel_right_index]) {
|
||||
// // portal_idx = i;
|
||||
// // continue;
|
||||
// // }
|
||||
// // break;
|
||||
// // }
|
||||
// // // For every portal with this corner, add a point to the path
|
||||
// // for(int i = portal_idx; i <= funnel_right_index; i++) {
|
||||
// // // Find the vertex opposite this portal edge
|
||||
// // vector opposite_corner = cl_navmesh_verts[cl_test_pathfind_result->portals_left_vert[i]].pos;
|
||||
// // // Move 20 qu or 20% of the edge length, whichever is smaller:
|
||||
// // float edge_len = vlen(opposite_corner - corner);
|
||||
// // vector edge_dir = normalize(opposite_corner - corner);
|
||||
// // vector delta_corner = edge_dir * min(edge_len * 0.2, 20);
|
||||
// // cl_test_pathfind_result->result_path[cl_test_pathfind_result->result_length++] = corner + delta_corner;
|
||||
// // }
|
||||
// // // ------------------------------------------------------------
|
||||
// cl_test_pathfind_result->result_path[cl_test_pathfind_result->result_length++] = corner;
|
||||
|
||||
// //Check if we are at the end of our portal list
|
||||
// if(funnel_right_index >= cl_test_pathfind_result->portals_length - 1) {
|
||||
// cl_test_pathfind_result->result_path[cl_test_pathfind_result->result_length++] = goal_point;
|
||||
// return;
|
||||
// }
|
||||
|
||||
// print("\t\tnew funnel apex is now vert: ");
|
||||
// print(ftos(cl_test_pathfind_result->portals_right_vert[funnel_right_index]));
|
||||
// print("\n");
|
||||
|
||||
// //Restart algorithm with the portal after the right point
|
||||
// funnel_apex = funnel_right;
|
||||
|
||||
// funnel_left_index = funnel_right_index + 1;
|
||||
// // funnel_left = cl_navmesh_verts[cl_test_pathfind_result->portals_left_vert[funnel_left_index]].pos;
|
||||
// funnel_left = cl_pathfind_get_left_portal_corner(cl_test_pathfind_result, funnel_left_index);
|
||||
// funnel_right_index = funnel_right_index + 1;
|
||||
// // funnel_right = cl_navmesh_verts[cl_test_pathfind_result->portals_right_vert[funnel_right_index]].pos;
|
||||
// funnel_right = cl_pathfind_get_right_portal_corner(cl_test_pathfind_result, funnel_right_index);
|
||||
// continue;
|
||||
// }
|
||||
// //If the next left edge is in the funnel
|
||||
// inside_left_edge = pathfind_point_is_to_left(funnel_left,funnel_apex,next_funnel_left);
|
||||
// if(inside_left_edge >= 0 && inside_right_edge >= 0) {
|
||||
// print("\t\tnext left edge is in the funnel, narrowing funnel.\n");
|
||||
// //Advance the left edge
|
||||
// funnel_left = next_funnel_left;
|
||||
// funnel_left_index++;
|
||||
// advanced_left = TRUE;
|
||||
// //Check if the portal we just added was the goal point (will be after the list)
|
||||
// if(funnel_left_index >= cl_test_pathfind_result->portals_length) {
|
||||
// cl_test_pathfind_result->result_path[cl_test_pathfind_result->result_length++] = goal_point;
|
||||
// return;
|
||||
// }
|
||||
// }
|
||||
// }
|
||||
// //================ Checking right funnel edge =================
|
||||
// if(funnel_right_index < cl_test_pathfind_result->portals_length) {
|
||||
// print("\t -- Trying to advance right edge --\n");
|
||||
|
||||
// if(funnel_right_index < cl_test_pathfind_result->portals_length - 1) {
|
||||
// next_funnel_right = cl_pathfind_get_right_portal_corner(cl_test_pathfind_result, funnel_right_index + 1);
|
||||
// // next_funnel_right = cl_navmesh_verts[cl_test_pathfind_result->portals_right_vert[funnel_right_index+1]].pos;
|
||||
// print("\t\tNext right edge is vert: ");
|
||||
// print(ftos(cl_test_pathfind_result->portals_right_vert[funnel_right_index+1]));
|
||||
// print(".\n");
|
||||
// }
|
||||
// else {
|
||||
// print("\t\tTrying next right edge as goal point.\n");
|
||||
// //If funnel_right is pointing to the last portal in the array, consider the goal_point the last portal
|
||||
// next_funnel_right = goal_point;
|
||||
// }
|
||||
|
||||
// //If the next right edge crosses the current left edge
|
||||
// inside_left_edge = pathfind_point_is_to_left(funnel_left,funnel_apex,next_funnel_right);
|
||||
// if(inside_left_edge < 0) {
|
||||
// print("\t\tnext right edge crosses current left edge.\n");
|
||||
// //Add funnel left point to path
|
||||
// // cl_test_pathfind_result->result_path[cl_test_pathfind_result->result_length++] = funnel_left;
|
||||
// corner = funnel_left;
|
||||
// // // ------------------------------------------------------------
|
||||
// // // Instead of adding the funnel's corner to the path,
|
||||
// // // move away from the corner by some small amount
|
||||
// // // ------------------------------------------------------------
|
||||
// // // Find the first portal along the path with this corner
|
||||
// // int portal_idx = funnel_left_index;
|
||||
// // for(int i = funnel_left_index; i >= 0; i--) {
|
||||
// // if(cl_test_pathfind_result->portals_left_vert[i] == cl_test_pathfind_result->portals_left_vert[funnel_left_index]) {
|
||||
// // portal_idx = i;
|
||||
// // continue;
|
||||
// // }
|
||||
// // break;
|
||||
// // }
|
||||
// // // For every portal with this corner, add a point to the path
|
||||
// // for(int i = portal_idx; i <= funnel_left_index; i++) {
|
||||
// // // Find the vertex opposite this portal edge
|
||||
// // vector opposite_corner = cl_navmesh_verts[cl_test_pathfind_result->portals_right_vert[i]].pos;
|
||||
// // // Move 20 qu or 20% of the edge length, whichever is smaller:
|
||||
// // float edge_len = vlen(opposite_corner - corner);
|
||||
// // vector edge_dir = normalize(opposite_corner - corner);
|
||||
// // vector delta_corner = edge_dir * min(edge_len * 0.2, 20);
|
||||
// // cl_test_pathfind_result->result_path[cl_test_pathfind_result->result_length++] = corner + delta_corner;
|
||||
// // }
|
||||
// // // ------------------------------------------------------------
|
||||
// cl_test_pathfind_result->result_path[cl_test_pathfind_result->result_length++] = corner;
|
||||
|
||||
// //Check if we are at the end of our portal list
|
||||
// if(funnel_left_index >= cl_test_pathfind_result->portals_length - 1) {
|
||||
// cl_test_pathfind_result->result_path[cl_test_pathfind_result->result_length++] = goal_point;
|
||||
// return;
|
||||
// }
|
||||
|
||||
// //Restart algorithm with the portal after the right point
|
||||
// funnel_apex = funnel_left;
|
||||
// print("\t\tnew funnel apex is now vert: ");
|
||||
// print(ftos(cl_test_pathfind_result->portals_left_vert[funnel_left_index]));
|
||||
// print("\n");
|
||||
|
||||
// funnel_right_index = funnel_left_index + 1;
|
||||
// funnel_right = cl_pathfind_get_right_portal_corner(cl_test_pathfind_result, funnel_right_index);
|
||||
// // funnel_right = cl_navmesh_verts[cl_test_pathfind_result->portals_right_vert[funnel_right_index]].pos;
|
||||
// funnel_left_index = funnel_left_index + 1;
|
||||
// funnel_left = cl_pathfind_get_left_portal_corner(cl_test_pathfind_result, funnel_left_index);
|
||||
// // funnel_left = cl_navmesh_verts[cl_test_pathfind_result->portals_left_vert[funnel_left_index]].pos;
|
||||
// continue;
|
||||
// }
|
||||
// //If the next right edge is in the funnel
|
||||
// inside_right_edge = pathfind_point_is_to_left(funnel_apex,funnel_right,next_funnel_right);
|
||||
// if(inside_left_edge >= 0 && inside_right_edge >= 0) {
|
||||
// print("\t\tnext right edge is in the funnel, narrowing funnel.\n");
|
||||
// //Advance the left edge
|
||||
// funnel_right = next_funnel_right;
|
||||
// funnel_right_index++;
|
||||
// advanced_right = TRUE;
|
||||
|
||||
// //Check if the portal we just added was the goal point (will be after the list)
|
||||
// if(funnel_right_index >= cl_test_pathfind_result->portals_length) {
|
||||
// cl_test_pathfind_result->result_path[cl_test_pathfind_result->result_length++] = goal_point;
|
||||
// return;
|
||||
// }
|
||||
// }
|
||||
// }
|
||||
// if(advanced_left == FALSE && advanced_right == FALSE) {
|
||||
// print("Hit funnel freeze condition.\n");
|
||||
|
||||
// float left_vert = -1;
|
||||
// float left_type = -1;
|
||||
// float left_portal_index = -1;//index of the vertex in the list of portals
|
||||
|
||||
// float right_vert = -1;
|
||||
// float right_type = -1;
|
||||
// float right_portal_index = -1;//index of the vertex in the list of portals
|
||||
|
||||
// float crossed = 1;//Indicates the vert crossed the funnel
|
||||
// float contained = 2;//Indicates a vert is in the funnel
|
||||
|
||||
// float last_vert = -1;//used to skip calculating the same vertex more than once
|
||||
|
||||
|
||||
// //Find the closest vertex in the portal left with the lowest index that crosses the funnel or is in the funnel
|
||||
// for(float i = funnel_left_index + 1; i < cl_test_pathfind_result->portals_length + 1; i++) {
|
||||
// if(i < cl_test_pathfind_result->portals_length - 1) {
|
||||
// //Don't calculate the same vertex again
|
||||
// if(last_vert == cl_test_pathfind_result->portals_left_vert[i])
|
||||
// continue;
|
||||
// last_vert = cl_test_pathfind_result->portals_left_vert[i];
|
||||
|
||||
// next_funnel_left = cl_pathfind_get_left_portal_corner(cl_test_pathfind_result, i);
|
||||
// // next_funnel_left = cl_navmesh_verts[last_vert].pos;
|
||||
// }
|
||||
// else {
|
||||
// //consider goal pos as last portal edge
|
||||
// next_funnel_left = goal_point;
|
||||
// }
|
||||
|
||||
// inside_right_edge = pathfind_point_is_to_left(funnel_apex,funnel_right,next_funnel_left);
|
||||
|
||||
// //If the left vertex crosses the funnel (left vertex is outside the funnel's right edge)
|
||||
// if(inside_right_edge < 0) {
|
||||
// left_vert = cl_test_pathfind_result->portals_left_vert[i];
|
||||
// left_type = crossed;
|
||||
// break;
|
||||
// }
|
||||
|
||||
// inside_left_edge = pathfind_point_is_to_left(funnel_left,funnel_apex,next_funnel_left);
|
||||
|
||||
// //If the left vertex is within the funnel
|
||||
// if(inside_left_edge >= 0 && inside_right_edge >= 0) {
|
||||
// left_vert = cl_test_pathfind_result->portals_left_vert[i];
|
||||
// left_type = contained;
|
||||
// left_portal_index = i;
|
||||
// break;
|
||||
// }
|
||||
// }
|
||||
|
||||
// last_vert = -1;
|
||||
|
||||
// //Find the closest vertex in the portal right with the lowest index that crosses the funnel or is in the funnel
|
||||
// for(float i = funnel_right_index + 1; i < cl_test_pathfind_result->portals_length; i++) {
|
||||
// if(i < cl_test_pathfind_result->portals_length - 1) {
|
||||
// //Don't calculate the same vertex again
|
||||
// if(last_vert == cl_test_pathfind_result->portals_right_vert[i])
|
||||
// continue;
|
||||
// last_vert = cl_test_pathfind_result->portals_right_vert[i];
|
||||
|
||||
// next_funnel_right = cl_pathfind_get_right_portal_corner(cl_test_pathfind_result, i);
|
||||
// // next_funnel_right = cl_navmesh_verts[last_vert].pos;
|
||||
// }
|
||||
// //consider goal pos as last portal edge
|
||||
// else {
|
||||
// next_funnel_right = goal_point;
|
||||
// }
|
||||
|
||||
// inside_left_edge = pathfind_point_is_to_left(funnel_left,funnel_apex,next_funnel_right);
|
||||
|
||||
// //If the right vertex crosses the funnel (right vertex is outside the funnel's left edge)
|
||||
// if(inside_left_edge < 0) {
|
||||
// right_vert = cl_test_pathfind_result->portals_right_vert[i];
|
||||
// right_type = crossed;
|
||||
// break;
|
||||
// }
|
||||
|
||||
// inside_right_edge = pathfind_point_is_to_left(funnel_apex,funnel_right,next_funnel_right);
|
||||
|
||||
// //If the right vertex is within the funnel
|
||||
// if(inside_left_edge >= 0 && inside_right_edge >= 0) {
|
||||
// right_vert = cl_test_pathfind_result->portals_right_vert[i];
|
||||
// right_type = contained;
|
||||
// right_portal_index = i;
|
||||
// break;
|
||||
// }
|
||||
// }
|
||||
|
||||
// //If no vertices were found, goal is reachable from here
|
||||
// if(left_vert == -1 && right_vert == -1) {
|
||||
// cl_test_pathfind_result->result_path[cl_test_pathfind_result->result_length++] = goal_point;
|
||||
// return;
|
||||
// }
|
||||
|
||||
// float use_left;
|
||||
// //If both verts were found
|
||||
// //Find which vertex has a lower index
|
||||
// if(left_vert != -1 && right_vert != -1) {
|
||||
// if(left_portal_index < right_portal_index) {
|
||||
// use_left = TRUE;
|
||||
// }
|
||||
// else {
|
||||
// use_left = FALSE;
|
||||
// }
|
||||
// }
|
||||
|
||||
// //if left vert was found
|
||||
// else if(left_vert != -1) {
|
||||
// use_left = TRUE;
|
||||
// }
|
||||
// //else right vert was found
|
||||
// else {
|
||||
// use_left = FALSE;
|
||||
// }
|
||||
|
||||
// if(use_left) {
|
||||
// if(left_type == crossed) {
|
||||
// //Place corner at current right funnel edge
|
||||
// cl_test_pathfind_result->result_path[cl_test_pathfind_result->result_length++] = funnel_right;
|
||||
|
||||
// //Restart algorithm with the portal after the right point
|
||||
// funnel_apex = funnel_right;
|
||||
|
||||
// //Check if portal after this portal is goal:
|
||||
// funnel_right_index++;
|
||||
// if(funnel_right_index >= cl_test_pathfind_result->portals_length) {
|
||||
// cl_test_pathfind_result->result_path[cl_test_pathfind_result->result_length++] = goal_point;
|
||||
// return;
|
||||
// }
|
||||
|
||||
// funnel_left_index = funnel_right_index;
|
||||
// // funnel_left = cl_navmesh_verts[cl_test_pathfind_result->portals_left_vert[funnel_left_index]].pos;
|
||||
// // funnel_right = cl_navmesh_verts[cl_test_pathfind_result->portals_right_vert[funnel_right_index]].pos;
|
||||
// funnel_left = cl_pathfind_get_left_portal_corner(cl_test_pathfind_result, funnel_left_index);
|
||||
// funnel_right = cl_pathfind_get_right_portal_corner(cl_test_pathfind_result, funnel_right_index);
|
||||
// continue;
|
||||
// }
|
||||
// //in funnel
|
||||
// else {
|
||||
// //If the next funnel right is the goal, we are done
|
||||
// //Check if portal after this portal is goal:
|
||||
// if(right_portal_index >= cl_test_pathfind_result->portals_length) {
|
||||
// cl_test_pathfind_result->result_path[cl_test_pathfind_result->result_length++] = goal_point;
|
||||
// return;
|
||||
// }
|
||||
// //Update current left funnel edge to use this vertex
|
||||
// funnel_left = next_funnel_left;
|
||||
// funnel_left_index = left_portal_index;
|
||||
// continue;
|
||||
// }
|
||||
// }
|
||||
// //use right
|
||||
// else {
|
||||
// if(right_type == crossed) {
|
||||
// //Place corner at current left funnel edge
|
||||
// cl_test_pathfind_result->result_path[cl_test_pathfind_result->result_length++] = funnel_left;
|
||||
|
||||
// //Restart algorithm with the portal after the left point
|
||||
// funnel_apex = funnel_left;
|
||||
|
||||
// //Check if portal after this portal is goal:
|
||||
// funnel_left_index++;
|
||||
// if(funnel_left_index >= cl_test_pathfind_result->portals_length) {
|
||||
// cl_test_pathfind_result->result_path[cl_test_pathfind_result->result_length++] = goal_point;
|
||||
// return;
|
||||
// }
|
||||
|
||||
// funnel_right_index = funnel_left_index;
|
||||
// funnel_left = cl_pathfind_get_left_portal_corner(cl_test_pathfind_result, funnel_left_index);
|
||||
// funnel_right = cl_pathfind_get_right_portal_corner(cl_test_pathfind_result, funnel_right_index);
|
||||
// continue;
|
||||
// }
|
||||
// //in funnel
|
||||
// else {
|
||||
// //If the next funnel right is the goal, we are done
|
||||
// //Check if portal after this portal is goal:
|
||||
// if(right_portal_index >= cl_test_pathfind_result->portals_length) {
|
||||
// cl_test_pathfind_result->result_path[cl_test_pathfind_result->result_length++] = goal_point;
|
||||
// return;
|
||||
// }
|
||||
// //Update current right funnel edge to use this vertex
|
||||
// funnel_right = next_funnel_right;
|
||||
// funnel_right_index = right_portal_index;
|
||||
// continue;
|
||||
// }
|
||||
// }
|
||||
// }
|
||||
// print("==========================\n");
|
||||
// }
|
||||
// }
|
||||
|
||||
|
||||
//Evaluates the path that is found in the pathfinding algorithm and populates an array with the nodes in the path from start to goal
|
||||
void cl_pathfind_trace_path(float start_poly, float goal_poly)
|
||||
{
|
||||
void cl_pathfind_trace_path(float start_poly, float goal_poly) {
|
||||
// print("Polygon from poly: ");
|
||||
// for(float i = 0; i < cl_navmesh_poly_count; i++) {
|
||||
// print(ftos(i));
|
||||
|
@ -3100,13 +2652,7 @@ float cl_navmesh_pathfind_start(float start_poly, float goal_poly, vector start_
|
|||
cl_test_pathfind_result->poly_set[start_poly] = PATHFIND_POLY_SET_OPEN;
|
||||
cl_test_pathfind_result->poly_g_score[start_poly] = 0;
|
||||
cl_test_pathfind_result->poly_f_score[start_poly] = 0 + cl_pathfind_calc_poly_h_score(start_poly , goal_poly);
|
||||
|
||||
//Fields that need to be set:
|
||||
//cl_test_pathfind_result->poly_set[NAV_MAX_POLIES];
|
||||
//cl_test_pathfind_result->prev_poly[NAV_MAX_POLIES];
|
||||
//cl_test_pathfind_result->poly_g_score[NAV_MAX_POLIES];
|
||||
//cl_test_pathfind_result->poly_f_score[NAV_MAX_POLIES];
|
||||
|
||||
|
||||
// print("Pathfind init. Start: ");
|
||||
// print(ftos(start_poly));
|
||||
// print(" , Goal: ");
|
||||
|
@ -3331,7 +2877,6 @@ float cl_navmesh_pathfind_start(float start_poly, float goal_poly, vector start_
|
|||
}
|
||||
}
|
||||
|
||||
// current = cl_pathfind_get_lowest_f_score();
|
||||
float best_openset_poly = cl_pathfind_get_lowest_f_score_poly();
|
||||
float best_openset_traversal = cl_pathfind_get_lowest_f_score_traversal();
|
||||
|
||||
|
@ -3359,9 +2904,6 @@ float cl_navmesh_pathfind_start(float start_poly, float goal_poly, vector start_
|
|||
// --------------------------------------------------------------------
|
||||
}
|
||||
|
||||
|
||||
|
||||
|
||||
//Tracing the pathfind results
|
||||
if(pathfind_success == TRUE) {
|
||||
cl_pathfind_trace_path(start_poly, goal_poly);
|
||||
|
|
|
@ -148,6 +148,12 @@ void (framegroup fgroup) AI_Chase::set_framegroup = {
|
|||
};
|
||||
|
||||
|
||||
// Computes the length of a vector ignoring the Z-component
|
||||
float (vector ofs) vlen_xy = {
|
||||
return sqrt(ofs.x * ofs.x + ofs.y * ofs.y);
|
||||
}
|
||||
|
||||
|
||||
void (float dist) AI_Chase::do_walk_to_goal = {
|
||||
if(dist == 0) {
|
||||
return;
|
||||
|
@ -158,6 +164,44 @@ void (float dist) AI_Chase::do_walk_to_goal = {
|
|||
goal_pos = this.path_target.origin;
|
||||
}
|
||||
|
||||
|
||||
|
||||
// TODO - For PSP, call engine-side function that gets us next walk point
|
||||
#ifdef PC
|
||||
navmesh_pathfind_result* res = &(sv_zombie_pathfind_result[this.pathfind_result_idx]);
|
||||
float start_poly = sv_navmesh_get_containing_poly(this.origin);
|
||||
float goal_poly = sv_navmesh_get_containing_poly(goal_pos);
|
||||
float pathfind_success = sv_navmesh_pathfind_start(start_poly, goal_poly, this.origin, goal_pos, res);
|
||||
this.pathfind_cur_point_idx = 0;
|
||||
|
||||
if(pathfind_success) {
|
||||
goal_pos = res->point_path_points[this.pathfind_cur_point_idx];
|
||||
|
||||
// TOOD - If close, continue to next one...
|
||||
// TODO - Also make sure we're "close enough" in Z...
|
||||
if(vlen_xy(this.origin - goal_pos) < 20) {
|
||||
print("Distance from target: ", ftos(vlen_xy(this.origin - goal_pos)), ", Cur point: ", ftos(this.pathfind_cur_point_idx), "\n");
|
||||
print("Current traversal: ", ftos(res->point_path_traversals[this.pathfind_cur_point_idx]),"\n");
|
||||
// If this point path is a traversal, teleport to traversal end spot
|
||||
if(res->point_path_traversals[this.pathfind_cur_point_idx] != -1) {
|
||||
vector traversal_end_pos = sv_navmesh_get_traversal_end_pos(res->point_path_traversals[this.pathfind_cur_point_idx]);
|
||||
print("Setting origin to end position: ", vtos(traversal_end_pos), "\n");
|
||||
this.origin = traversal_end_pos;
|
||||
}
|
||||
this.pathfind_cur_point_idx += 1;
|
||||
goal_pos = res->point_path_points[this.pathfind_cur_point_idx];
|
||||
}
|
||||
}
|
||||
else {
|
||||
dist = 0;
|
||||
// TODO - Idle animation?
|
||||
}
|
||||
#endif // PC
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
this.ideal_yaw = vectoyaw(goal_pos - this.origin);
|
||||
ChangeYaw();
|
||||
vector new_velocity;
|
||||
|
@ -304,6 +348,7 @@ void zombie_walk_think_callback(entity ent) {
|
|||
ent.enemy = player_ent;
|
||||
}
|
||||
|
||||
|
||||
float n_anim_frames = fabs(zombie_ent.cur_fg_end_frame - zombie_ent.cur_fg_start_frame);
|
||||
// float dist_per_frame = 5.23125; // qu
|
||||
float dist_per_frame = get_zombie_walk_dist_for_frame(zombie_ent.frame);
|
||||
|
|
|
@ -29,26 +29,58 @@
|
|||
//==================================== Navmesh-based Pathfinding Functions ==================================
|
||||
//===========================================================================================================
|
||||
|
||||
//Calculates the heuristic h score value (Our best guess for how far this node is from the goal node)
|
||||
float sv_pathfind_calc_h_score(float current, float goal)
|
||||
{
|
||||
//FIXME: we could just as easily return vlen()^2 for comparisons... (saves a sqrt operation)
|
||||
return vlen(sv_navmesh_polies[goal].center - sv_navmesh_polies[current].center);
|
||||
// Calculates world-space coordinate of traversal endpoint
|
||||
vector sv_navmesh_get_traversal_midpoint(float traversal_index) {
|
||||
vector start_pos = sv_navmesh_traversals[traversal_index].start_pos;
|
||||
vector midpoint_pos = sv_navmesh_traversals[traversal_index].midpoint_pos;
|
||||
// vector end_pos = sv_navmesh_traversals[traversal_index].end_pos;
|
||||
float angle = sv_navmesh_traversals[traversal_index].angle;
|
||||
// Midpoint / endpoint pos are relative to start_pos:
|
||||
makevectors([0, angle, 0]);
|
||||
midpoint_pos = start_pos + (v_right * midpoint_pos.x) + (v_forward * midpoint_pos.y) + (v_up * midpoint_pos.z);
|
||||
// end_pos = start_pos + (v_right * end_pos.x) + (v_forward * end_pos.y) + (v_up * end_pos.z);
|
||||
return midpoint_pos;
|
||||
}
|
||||
|
||||
// Calculates world-space coordinate of traversal midpoint
|
||||
vector sv_navmesh_get_traversal_end_pos(float traversal_index) {
|
||||
vector start_pos = sv_navmesh_traversals[traversal_index].start_pos;
|
||||
// vector midpoint_pos = sv_navmesh_traversals[traversal_index].midpoint_pos;
|
||||
vector end_pos = sv_navmesh_traversals[traversal_index].end_pos;
|
||||
float angle = sv_navmesh_traversals[traversal_index].angle;
|
||||
// Midpoint / endpoint pos are relative to start_pos:
|
||||
makevectors([0, angle, 0]);
|
||||
// midpoint_pos = start_pos + (v_right * midpoint_pos.x) + (v_forward * midpoint_pos.y) + (v_up * midpoint_pos.z);
|
||||
end_pos = start_pos + (v_right * end_pos.x) + (v_forward * end_pos.y) + (v_up * end_pos.z);
|
||||
return end_pos;
|
||||
}
|
||||
|
||||
// Calculates the heuristic h score value for a navmesh polygon (Our best guess for how far this node is from the goal node)
|
||||
float sv_pathfind_calc_poly_h_score(float current_poly_idx, float goal_poly_idx) {
|
||||
//FIXME: we could just as easily return vlen()^2 for comparisons... (saves a sqrt operation)
|
||||
return vlen(sv_navmesh_polies[goal_poly_idx].center - sv_navmesh_polies[current_poly_idx].center);
|
||||
}
|
||||
|
||||
// Calculates the heuristic h score value for a navmesh traversal (Our best guess for how far this node is from the goal node)
|
||||
float sv_pathfind_calc_traversal_h_score(float current_traversal_idx, float goal_poly_idx) {
|
||||
//FIXME: we could just as easily return vlen()^2 for comparisons... (saves a sqrt operation)
|
||||
vector traversal_end_pos = sv_navmesh_get_traversal_end_pos(current_traversal_idx);
|
||||
return vlen(sv_navmesh_polies[goal_poly_idx].center - traversal_end_pos);
|
||||
}
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
//Returns the polygon with the lowest f score from polygons the open set
|
||||
float sv_pathfind_get_lowest_f_score(navmesh_pathfind_result* res)
|
||||
{
|
||||
float sv_pathfind_get_lowest_f_score_poly(navmesh_pathfind_result* res) {
|
||||
//TODO: implement a better algorithm for finding the lowest score
|
||||
float best_score = 100000;
|
||||
float best_score_index = -1;
|
||||
|
||||
for(float i = 0; i < sv_navmesh_poly_count; i++)
|
||||
{
|
||||
if(res->poly_set[i] == PATHFIND_POLY_SET_OPEN)
|
||||
{
|
||||
if(res->poly_f_score[i] < best_score)
|
||||
{
|
||||
for(float i = 0; i < sv_navmesh_poly_count; i++) {
|
||||
if(res->poly_set[i] == PATHFIND_POLY_SET_OPEN) {
|
||||
if(res->poly_f_score[i] < best_score) {
|
||||
best_score = res->poly_f_score[i];
|
||||
best_score_index = i;
|
||||
}
|
||||
|
@ -57,6 +89,25 @@ float sv_pathfind_get_lowest_f_score(navmesh_pathfind_result* res)
|
|||
return best_score_index;
|
||||
}
|
||||
|
||||
|
||||
//Returns the traversal with the lowest f score from traversals the open set
|
||||
float sv_pathfind_get_lowest_f_score_traversal(navmesh_pathfind_result* res) {
|
||||
//TODO: implement a better algorithm for finding the lowest score
|
||||
float best_score = 100000;
|
||||
float best_score_index = -1;
|
||||
|
||||
for(float i = 0; i < sv_navmesh_traversal_count; i++) {
|
||||
if(res->traversal_set[i] == PATHFIND_POLY_SET_OPEN) {
|
||||
if(res->traversal_f_score[i] < best_score) {
|
||||
best_score = res->traversal_f_score[i];
|
||||
best_score_index = i;
|
||||
}
|
||||
}
|
||||
}
|
||||
return best_score_index;
|
||||
}
|
||||
|
||||
|
||||
void sv_pathfind_clear_result_data(navmesh_pathfind_result* res) {
|
||||
//Technically we only need to iterate over navmesh_poly_count...
|
||||
for(float i = 0; i < NAV_MAX_POLIES; i++) {
|
||||
|
@ -95,7 +146,198 @@ void sv_pathfind_clear_result_data(navmesh_pathfind_result* res) {
|
|||
|
||||
//Applies a funnel algorithm to the path defined by the array res->result_node_path
|
||||
// and populates pathfind result path
|
||||
void sv_pathfind_smooth_path(vector start_point, vector goal_point, navmesh_pathfind_result* res) {};
|
||||
|
||||
void sv_pathfind_smooth_path(vector start_point, vector goal_point, navmesh_pathfind_result* res) {
|
||||
|
||||
float cur_poly;
|
||||
float cur_traversal;
|
||||
float prev_poly;
|
||||
float prev_traversal;
|
||||
vector cur_portal_left_pos = '0 0 0';
|
||||
vector cur_portal_right_pos = '0 0 0';
|
||||
float cur_portal_traversal = -1;
|
||||
vector traversal_start_pos;
|
||||
vector traversal_end_pos;
|
||||
|
||||
// Build the list of portals
|
||||
for(float i = 1; i < res->path_length; i++) {
|
||||
cur_poly = res->path_polygons[i];
|
||||
cur_traversal = res->path_traversals[i];
|
||||
prev_poly = res->path_polygons[i-1];
|
||||
prev_traversal = res->path_traversals[i-1];
|
||||
|
||||
// If polygon and came from polygon, portal = shared edge
|
||||
if(cur_poly != -1 && prev_poly != -1) {
|
||||
// Find which of prev_poly's edges connects to cur_poly
|
||||
float prev_poly_edge_index = -1;
|
||||
for(float j = 0; j < sv_navmesh_polies[prev_poly].connected_polies_count; j++) {
|
||||
if(sv_navmesh_polies[prev_poly].connected_polies[j] == cur_poly) {
|
||||
prev_poly_edge_index = j;
|
||||
break;
|
||||
}
|
||||
}
|
||||
cur_portal_left_pos = sv_navmesh_verts[sv_navmesh_polies[prev_poly].connected_polies_left_vert[prev_poly_edge_index]].pos;
|
||||
cur_portal_right_pos = sv_navmesh_verts[sv_navmesh_polies[prev_poly].connected_polies_right_vert[prev_poly_edge_index]].pos;
|
||||
cur_portal_traversal = -1;
|
||||
|
||||
// ----------------------------------------------------------------
|
||||
// Narrow the portal from each side by 20% or 20qu, whichever is smaller
|
||||
// ----------------------------------------------------------------
|
||||
float edge_len = vlen(cur_portal_right_pos - cur_portal_left_pos);
|
||||
vector edge_dir = normalize(cur_portal_right_pos - cur_portal_left_pos);
|
||||
float ofs = min(edge_len * 0.2, 20);
|
||||
cur_portal_left_pos = cur_portal_left_pos + edge_dir * ofs;
|
||||
cur_portal_right_pos = cur_portal_right_pos - edge_dir * ofs;
|
||||
// ----------------------------------------------------------------
|
||||
}
|
||||
// If polygon and came from traversal, portal = traversal exit
|
||||
else if(cur_poly != -1 && prev_traversal != -1) {
|
||||
traversal_end_pos = sv_navmesh_get_traversal_end_pos(prev_traversal);
|
||||
cur_portal_left_pos = traversal_end_pos;
|
||||
cur_portal_right_pos = traversal_end_pos;
|
||||
cur_portal_traversal = -1;
|
||||
// TODO - Denote that it came from traversal? maybe we don't care?
|
||||
}
|
||||
// If traversal and came from polygon, portal = traversal start
|
||||
else if(cur_traversal != -1 && prev_poly != -1) {
|
||||
traversal_start_pos = sv_navmesh_traversals[cur_traversal].start_pos;
|
||||
cur_portal_left_pos = traversal_start_pos;
|
||||
cur_portal_right_pos = traversal_start_pos;
|
||||
cur_portal_traversal = cur_traversal;
|
||||
}
|
||||
|
||||
res->portals_left_pos[res->portals_length] = cur_portal_left_pos;
|
||||
res->portals_right_pos[res->portals_length] = cur_portal_right_pos;
|
||||
res->portals_traversal[res->portals_length] = cur_portal_traversal;
|
||||
res->portals_length++;
|
||||
}
|
||||
|
||||
|
||||
// TODO - Should I add goal_pos as the final portal?
|
||||
res->portals_left_pos[res->portals_length] = goal_point;
|
||||
res->portals_right_pos[res->portals_length] = goal_point;
|
||||
res->portals_traversal[res->portals_length] = -1;
|
||||
res->portals_length++;
|
||||
|
||||
|
||||
|
||||
vector cur_funnel_corner = start_point;
|
||||
vector cur_funnel_left = res->portals_left_pos[0];
|
||||
vector cur_funnel_right = res->portals_right_pos[0];
|
||||
float cur_funnel_left_portal_idx = 0;
|
||||
float cur_funnel_right_portal_idx = 0;
|
||||
|
||||
// Loop through all portals, adding necessary funnel corners to final path
|
||||
for(float i = 0; i < res->portals_length; i++) {
|
||||
cur_portal_left_pos = res->portals_left_pos[i];
|
||||
cur_portal_right_pos = res->portals_right_pos[i];
|
||||
cur_portal_traversal = res->portals_traversal[i];
|
||||
|
||||
|
||||
// TODO - Project portal points to axis made by cur funnel points
|
||||
// Then we can check
|
||||
// TODO - Compute value for a point:
|
||||
// TODO 0: On funnel center
|
||||
// TODO < -1 outside of funnel to the left
|
||||
// TODO > 1 outside of funnel to the right
|
||||
|
||||
float cur_portal_left_in_funnel = pathfind_point_in_funnel( cur_portal_left_pos, cur_funnel_corner, cur_funnel_left, cur_funnel_right);
|
||||
float cur_portal_right_in_funnel = pathfind_point_in_funnel(cur_portal_right_pos, cur_funnel_corner, cur_funnel_left, cur_funnel_right);
|
||||
|
||||
// print("{'portal_idx': ", ftos(i), ", 'portal': ");
|
||||
// print(" [(", ftos(cur_portal_left_pos.x), ",", ftos(cur_portal_left_pos.y), "), (", ftos(cur_portal_right_pos.x), ",", ftos(cur_portal_right_pos.y));
|
||||
// print(")], 'traversal': ", ftos(cur_portal_traversal), ", ");
|
||||
// print("'funnel_result': (", ftos(cur_portal_left_in_funnel), ",", ftos(cur_portal_right_in_funnel), "), ");
|
||||
// print("'funnel': [");
|
||||
// print("(", ftos(cur_funnel_left.x), ",", ftos(cur_funnel_left.y), "), ");
|
||||
// print("(", ftos(cur_funnel_corner.x), ",", ftos(cur_funnel_corner.y), "), ");
|
||||
// print("(", ftos(cur_funnel_right.x), ",", ftos(cur_funnel_right.y), ")");
|
||||
// print("]},\n");
|
||||
|
||||
|
||||
// If cur portal left point is in cur funnel, narrow the cur funnel
|
||||
if(cur_portal_left_in_funnel >= -1 && cur_portal_left_in_funnel <= 1) {
|
||||
cur_funnel_left = cur_portal_left_pos;
|
||||
cur_funnel_left_portal_idx = i;
|
||||
}
|
||||
// If cur portal right point is in cur funnel, narrow the cur funnel
|
||||
if(cur_portal_right_in_funnel >= -1 && cur_portal_right_in_funnel <= 1) {
|
||||
cur_funnel_right = cur_portal_right_pos;
|
||||
cur_funnel_right_portal_idx = i;
|
||||
}
|
||||
|
||||
// If cur portal is completely outside (to left of) current funnel
|
||||
if(cur_portal_left_in_funnel < -1 && cur_portal_right_in_funnel < -1) {
|
||||
// Add cur funnel left point to final path:
|
||||
res->point_path_points[res->point_path_length] = cur_funnel_left;
|
||||
res->point_path_traversals[res->point_path_length] = -1;
|
||||
res->point_path_length++;
|
||||
|
||||
// Update funnel to start at the current funnel left endpoint, pointing to the next portal in the list
|
||||
// Set next funnel to start at cur funnel left point
|
||||
cur_funnel_corner = cur_funnel_left;
|
||||
|
||||
// Advance the funnel's left point to the next portal
|
||||
cur_funnel_left_portal_idx += 1;
|
||||
cur_funnel_right = res->portals_right_pos[cur_funnel_left_portal_idx];
|
||||
cur_funnel_left = res->portals_left_pos[cur_funnel_left_portal_idx];
|
||||
cur_funnel_right_portal_idx = cur_funnel_left_portal_idx;
|
||||
// Restart funnel algorithm from the funnel's endpoint
|
||||
i = cur_funnel_left_portal_idx - 1;
|
||||
continue;
|
||||
}
|
||||
// If cur portal is completely outside (to right of) current funnel
|
||||
else if(cur_portal_left_in_funnel > 1 && cur_portal_right_in_funnel > 1) {
|
||||
// Add cur funnel right point to final path:
|
||||
res->point_path_points[res->point_path_length] = cur_funnel_right;
|
||||
res->point_path_traversals[res->point_path_length] = -1;
|
||||
res->point_path_length++;
|
||||
|
||||
// Update funnel to start at the current funnel right endpoint, pointing to the next portal in the list
|
||||
// Set next funnel to start at cur funnel right point
|
||||
cur_funnel_corner = cur_funnel_right;
|
||||
|
||||
// Advance the funnel's right point to the next portal
|
||||
cur_funnel_right_portal_idx += 1;
|
||||
cur_funnel_right = res->portals_right_pos[cur_funnel_right_portal_idx];
|
||||
cur_funnel_left = res->portals_left_pos[cur_funnel_right_portal_idx];
|
||||
cur_funnel_left_portal_idx = cur_funnel_right_portal_idx;
|
||||
// Restart funnel algorithm from the funnel's endpoint
|
||||
i = cur_funnel_right_portal_idx - 1;
|
||||
continue;
|
||||
}
|
||||
|
||||
// If cur portal is a traversal
|
||||
if(cur_portal_traversal != -1) {
|
||||
vector start_pos = sv_navmesh_traversals[cur_portal_traversal].start_pos;
|
||||
vector end_pos = sv_navmesh_get_traversal_end_pos(cur_portal_traversal);
|
||||
// Add traversal start point to final path:
|
||||
res->point_path_points[res->point_path_length] = start_pos;
|
||||
res->point_path_traversals[res->point_path_length] = cur_portal_traversal;
|
||||
res->point_path_length++;
|
||||
// Add traversal end point to final path:
|
||||
// FIXME - Should I remove this?
|
||||
res->point_path_points[res->point_path_length] = end_pos;
|
||||
res->point_path_traversals[res->point_path_length] = -1;
|
||||
res->point_path_length++;
|
||||
|
||||
|
||||
// Set funnel to start at traversal endpoint and use the next portal
|
||||
cur_funnel_corner = end_pos;
|
||||
cur_funnel_left = res->portals_left_pos[i+1];
|
||||
cur_funnel_right = res->portals_right_pos[i+1];
|
||||
cur_funnel_left_portal_idx = i+1;
|
||||
cur_funnel_right_portal_idx = i+1;
|
||||
}
|
||||
}
|
||||
|
||||
// Always add goal point to the path
|
||||
res->point_path_points[res->point_path_length] = goal_point;
|
||||
res->point_path_traversals[res->point_path_length] = -1;
|
||||
res->point_path_length++;
|
||||
}
|
||||
|
||||
|
||||
// void sv_pathfind_smooth_path(vector start_point, vector goal_point, navmesh_pathfind_result* res) {
|
||||
// //Evaluate portals and identify left / right portal vertices
|
||||
// for(float i = 1; i < res->result_node_length; i++)
|
||||
|
@ -561,50 +803,346 @@ void sv_pathfind_smooth_path(vector start_point, vector goal_point, navmesh_path
|
|||
// }
|
||||
|
||||
//Evaluates the path that is found in the pathfinding algorithm and populates an array with the nodes in the path from start to goal
|
||||
void sv_pathfind_trace_path(float start, float goal, navmesh_pathfind_result* res) {}
|
||||
// void sv_pathfind_trace_path(float start, float goal, navmesh_pathfind_result* res)
|
||||
// {
|
||||
// float current = start;
|
||||
// //Count the length of the path (how many polygons the path traverses)
|
||||
// current = goal;
|
||||
// res->result_node_length = 1;
|
||||
// do
|
||||
// {
|
||||
// //print("Poly ");
|
||||
// //print(ftos(current));
|
||||
// //print(" came from ");
|
||||
// //print(ftos(res->poly_prev[current]));
|
||||
// //print(".\n");
|
||||
// current = res->poly_prev[current];
|
||||
// res->result_node_length++;
|
||||
// } while(current != start);
|
||||
void sv_pathfind_trace_path(float start_poly, float goal_poly, navmesh_pathfind_result* res) {
|
||||
// print("Polygon from poly: ");
|
||||
// for(float i = 0; i < sv_navmesh_poly_count; i++) {
|
||||
// print(ftos(i));
|
||||
// print(": ");
|
||||
// print(ftos(res->poly_prev_poly[i]));
|
||||
// print(", ");
|
||||
// }
|
||||
// print("\n");
|
||||
// print("Polygon from traversal: ");
|
||||
// for(float i = 0; i < sv_navmesh_poly_count; i++) {
|
||||
// print(ftos(i));
|
||||
// print(": ");
|
||||
// print(ftos(res->poly_prev_traversal[i]));
|
||||
// print(", ");
|
||||
// }
|
||||
// print("\n");
|
||||
// print("Traversal from polygon:");
|
||||
// for(float i = 0; i < sv_navmesh_traversal_count; i++) {
|
||||
// print(ftos(i));
|
||||
// print(": ");
|
||||
// print(ftos(res->traversal_prev_poly[i]));
|
||||
// print(", ");
|
||||
// }
|
||||
// print("\n");
|
||||
|
||||
|
||||
//Count the length of the path (how many polygons the path traverses)
|
||||
float current_poly = goal_poly;
|
||||
float current_traversal = -1;
|
||||
res->path_length = 1;
|
||||
do {
|
||||
// Current node is a polygon
|
||||
if(current_poly != -1) {
|
||||
current_traversal = res->poly_prev_traversal[current_poly];
|
||||
current_poly = res->poly_prev_poly[current_poly];
|
||||
}
|
||||
// Current node is a traversal
|
||||
else if(current_traversal != -1) {
|
||||
current_poly = res->traversal_prev_poly[current_traversal];
|
||||
current_traversal = -1;
|
||||
}
|
||||
res->path_length++;
|
||||
} while(current_poly != start_poly);
|
||||
|
||||
|
||||
// //Starting at goal waypoint, add the traversed waypoints to the result path in reverse order
|
||||
// current = goal;
|
||||
// for(float i = 0; i < res->result_node_length; i++)
|
||||
// {
|
||||
// res->result_node_path[res->result_node_length - 1 - i] = current;
|
||||
// current = res->poly_prev[current];
|
||||
// }
|
||||
// Starting at goal_poly, add the traversed polygons / traversals to the result path in reverse order
|
||||
current_poly = goal_poly;
|
||||
current_traversal = -1;
|
||||
for(float i = 0; i < res->path_length; i++) {
|
||||
// print("Current node (P=");
|
||||
// print(ftos(current_poly));
|
||||
// print(",T=");
|
||||
// print(ftos(current_traversal));
|
||||
// print(") came from ");
|
||||
|
||||
res->path_polygons[res->path_length - 1 - i] = current_poly;
|
||||
res->path_traversals[res->path_length - 1 - i] = current_traversal;
|
||||
// Current node is a polygon
|
||||
if(current_poly != -1) {
|
||||
current_traversal = res->poly_prev_traversal[current_poly];
|
||||
current_poly = res->poly_prev_poly[current_poly];
|
||||
}
|
||||
// Current node is a traversal
|
||||
else if(current_traversal != -1) {
|
||||
current_poly = res->traversal_prev_poly[current_traversal];
|
||||
current_traversal = -1;
|
||||
}
|
||||
|
||||
// print(" (P=");
|
||||
// print(ftos(current_poly));
|
||||
// print(",T=");
|
||||
// print(ftos(current_traversal));
|
||||
// print(")\n");
|
||||
}
|
||||
|
||||
|
||||
// // print("Pathfind success, path length: ");
|
||||
// // print(ftos(res->result_node_length));
|
||||
// // print(", Path: [");
|
||||
// // for(float i = 0; i < res->result_node_length; i++)
|
||||
// // {
|
||||
// // if(i > 0)
|
||||
// // print(" , ");
|
||||
// // print(ftos(res->result_node_path[i]));
|
||||
// // }
|
||||
// // print(" ].\n");
|
||||
// }
|
||||
print("Pathfind success, path length: ");
|
||||
print(ftos(res->path_length));
|
||||
print(".\n");
|
||||
print("Path: ");
|
||||
for(float i = 0; i < res->path_length; i++) {
|
||||
if(i > 0) {
|
||||
print(" , ");
|
||||
}
|
||||
if(res->path_polygons[i] != -1) {
|
||||
print("P");
|
||||
print(ftos(res->path_polygons[i]));
|
||||
}
|
||||
else if(res->path_traversals[i] != -1) {
|
||||
print("T");
|
||||
print(ftos(res->path_traversals[i]));
|
||||
}
|
||||
}
|
||||
print(" .\n");
|
||||
}
|
||||
|
||||
|
||||
|
||||
//Accepts start polygon and goal polygon
|
||||
//Returns 1 on success.
|
||||
//Returns 0 on fail.
|
||||
float sv_navmesh_pathfind_start(float start, float goal, vector start_pos, vector end_pos, navmesh_pathfind_result* res) {}
|
||||
float sv_navmesh_pathfind_start(float start_poly, float goal_poly, vector start_pos, vector end_pos, navmesh_pathfind_result* res) {
|
||||
if(start_poly == -1) {
|
||||
print("Error: pathfind start node invalid.\n");
|
||||
return 0;
|
||||
}
|
||||
if(goal_poly == -1) {
|
||||
print("Error: pathfind goal node invalid.\n");
|
||||
return 0;
|
||||
}
|
||||
if(start_poly == goal_poly) {
|
||||
//Calculating node path
|
||||
res->path_polygons[0] = start_poly;
|
||||
res->path_traversals[0] = -1;
|
||||
res->path_length = 1;
|
||||
print("Pathind success: trivial case (start = goal).\n");
|
||||
|
||||
//Calculating vector based path (go directly to goal)
|
||||
res->point_path_points[0].x = end_pos.x;
|
||||
res->point_path_points[0].y = end_pos.y;
|
||||
res->point_path_points[0].z = end_pos.z;
|
||||
// Indicate non-traversal
|
||||
res->point_path_traversals[0] = -1;
|
||||
res->point_path_length = 1;
|
||||
return 1;
|
||||
}
|
||||
|
||||
//Clearing previous data
|
||||
sv_pathfind_clear_result_data(res);
|
||||
|
||||
//Adding start polygon to the open set
|
||||
res->poly_set[start_poly] = PATHFIND_POLY_SET_OPEN;
|
||||
res->poly_g_score[start_poly] = 0;
|
||||
res->poly_f_score[start_poly] = 0 + sv_pathfind_calc_poly_h_score(start_poly , goal_poly);
|
||||
|
||||
// print("Pathfind init. Start: ");
|
||||
// print(ftos(start_poly));
|
||||
// print(" , Goal: ");
|
||||
// print(ftos(goal_poly));
|
||||
// print(".\n");
|
||||
|
||||
float current_poly = start_poly;
|
||||
float current_traversal = -1;
|
||||
float pathfind_success = FALSE;
|
||||
|
||||
// While we have a traversal OR a polygon
|
||||
while(current_poly != -1 || current_traversal != -1) {
|
||||
|
||||
// If we are currently evaluating a polygon:
|
||||
if(current_poly != -1) {
|
||||
if(current_poly == goal_poly) {
|
||||
//print("Current is now goal. Breaking.\n");
|
||||
pathfind_success = TRUE;
|
||||
break;
|
||||
}
|
||||
|
||||
//Add current node to the closed set
|
||||
res->poly_set[current_poly] = PATHFIND_POLY_SET_CLOSED;
|
||||
//Add connected neighbor polygons to the open set
|
||||
for(float i = 0; i < sv_navmesh_polies[current_poly].connected_polies_count; i++) {
|
||||
float neighbor_poly = sv_navmesh_polies[current_poly].connected_polies[i];
|
||||
|
||||
// print("Checking poly ");
|
||||
// print(ftos(current_poly));
|
||||
// print("'s neighbor poly ");
|
||||
// print(ftos(neighbor_poly));
|
||||
// print(".\n");
|
||||
|
||||
// ----------------------------------------------------------------
|
||||
// Door check
|
||||
// ----------------------------------------------------------------
|
||||
// NOTE - If polygon's door hasn't been opened, don't consider it.
|
||||
// NOTE - This check isn't done in the navmesh editor.
|
||||
// ----------------------------------------------------------------
|
||||
|
||||
|
||||
if(res->poly_set[neighbor_poly] != PATHFIND_POLY_SET_CLOSED) {
|
||||
//print("Neighbor is not in closed list.\n");
|
||||
//Calculate tentative f score
|
||||
//Calculate tentative g score (distance from start to current + distance from current to neighbor)
|
||||
float tentative_g_score = res->poly_g_score[current_poly] + vlen(sv_navmesh_polies[neighbor_poly].center - sv_navmesh_polies[current_poly].center);
|
||||
|
||||
// If not in open-set, or this is a better score, set score for this polygon
|
||||
if(res->poly_set[neighbor_poly] != PATHFIND_POLY_SET_OPEN || tentative_g_score < res->poly_g_score[neighbor_poly]) {
|
||||
//print("Neighbor is not in open list, or a better g score has been found.\n");
|
||||
//Adding neighbor to open set
|
||||
res->poly_set[neighbor_poly] = PATHFIND_POLY_SET_OPEN;
|
||||
|
||||
//Updating scores for neighbor node
|
||||
float tentative_f_score = tentative_g_score + sv_pathfind_calc_poly_h_score(neighbor_poly , goal_poly);
|
||||
res->poly_g_score[neighbor_poly] = tentative_g_score;
|
||||
res->poly_f_score[neighbor_poly] = tentative_f_score;
|
||||
|
||||
//Linking neighbor node to current node (for tracing path)
|
||||
res->poly_prev_poly[neighbor_poly] = current_poly;
|
||||
res->poly_prev_traversal[neighbor_poly] = -1;
|
||||
|
||||
// print("Assigning Poly ");
|
||||
// print(ftos(neighbor_poly));
|
||||
// print(" came from poly ");
|
||||
// print(ftos(current_poly));
|
||||
// print(".\n");
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
//Add connected neighbor traversals to the open set
|
||||
for(float i = 0; i < sv_navmesh_polies[current_poly].connected_traversals_count; i++) {
|
||||
float neighbor_traversal = sv_navmesh_polies[current_poly].connected_traversals[i];
|
||||
|
||||
// print("Checking poly ");
|
||||
// print(ftos(current_poly));
|
||||
// print("'s neighbor traversal ");
|
||||
// print(ftos(neighbor_traversal));
|
||||
// print(".\n");
|
||||
|
||||
if(res->traversal_set[neighbor_traversal] != PATHFIND_POLY_SET_CLOSED) {
|
||||
//Calculate tentative f score
|
||||
//Calculate tentative g score (distance from start to current + distance from current to neighbor)
|
||||
// vector traversal_start_pos = sv_navmesh_traversals[neighbor].start_pos;
|
||||
vector traversal_end_pos = sv_navmesh_get_traversal_end_pos(neighbor_traversal);
|
||||
// TODO - Should I cache traversal length?
|
||||
float tentative_g_score = res->poly_g_score[current_poly] + vlen(traversal_end_pos - sv_navmesh_polies[current_poly].center);
|
||||
|
||||
// If not in open-set, or this is a better score, set score for this traversal
|
||||
if(res->traversal_set[neighbor_traversal] != PATHFIND_POLY_SET_OPEN || tentative_g_score < res->traversal_g_score[neighbor_traversal]) {
|
||||
//print("Neighbor is not in open list, or a better g score has been found.\n");
|
||||
//Adding neighbor to open set
|
||||
res->traversal_set[neighbor_traversal] = PATHFIND_POLY_SET_OPEN;
|
||||
|
||||
//Updating scores for neighbor node
|
||||
float tentative_f_score = tentative_g_score + sv_pathfind_calc_traversal_h_score(neighbor_traversal, goal_poly);
|
||||
res->traversal_g_score[neighbor_traversal] = tentative_g_score;
|
||||
res->traversal_f_score[neighbor_traversal] = tentative_f_score;
|
||||
|
||||
//Linking neighbor node to current node (for tracing path)
|
||||
res->traversal_prev_poly[neighbor_traversal] = current_poly;
|
||||
// print("Assigning traversal ");
|
||||
// print(ftos(neighbor_traversal));
|
||||
// print(" came from poly ");
|
||||
// print(ftos(current_poly));
|
||||
// print(".\n");
|
||||
}
|
||||
|
||||
}
|
||||
}
|
||||
}
|
||||
else if(current_traversal != -1) {
|
||||
// print("Checking traversal: ");
|
||||
// print(ftos(current_traversal));
|
||||
// print(" with end polygon: ");
|
||||
// print(ftos(sv_navmesh_traversals[current_traversal].end_poly));
|
||||
// print("\n");
|
||||
//Add current traversal to the closed set
|
||||
res->traversal_set[current_traversal] = PATHFIND_POLY_SET_CLOSED;
|
||||
|
||||
// Traversals have single neighbor polygons (at the traversal endpoint)
|
||||
float neighbor_poly = sv_navmesh_traversals[current_traversal].end_poly;
|
||||
if(neighbor_poly != -1) {
|
||||
|
||||
// print("Checking traversal ");
|
||||
// print(ftos(current_traversal));
|
||||
// print("'s neighbor poly ");
|
||||
// print(ftos(neighbor_poly));
|
||||
// print(".\n");
|
||||
|
||||
if(res->poly_set[neighbor_poly] != PATHFIND_POLY_SET_CLOSED) {
|
||||
//print("Neighbor is not in closed list.\n");
|
||||
//Calculate tentative f score
|
||||
//Calculate tentative g score (distance from start to current + distance from current to neighbor)
|
||||
vector traversal_end_pos = sv_navmesh_get_traversal_end_pos(current_traversal);
|
||||
float tentative_g_score = res->traversal_g_score[current_traversal] + vlen(sv_navmesh_polies[neighbor_poly].center - traversal_end_pos);
|
||||
|
||||
// If not in open-set, or this is a better score, set score for this polygon
|
||||
if(res->poly_set[neighbor_poly] != PATHFIND_POLY_SET_OPEN || tentative_g_score < res->poly_g_score[neighbor_poly]) {
|
||||
//print("Neighbor is not in open list, or a better g score has been found.\n");
|
||||
//Adding neighbor to open set
|
||||
res->poly_set[neighbor_poly] = PATHFIND_POLY_SET_OPEN;
|
||||
|
||||
//Updating scores for neighbor node
|
||||
float tentative_f_score = tentative_g_score + sv_pathfind_calc_poly_h_score(neighbor_poly , goal_poly);
|
||||
res->poly_g_score[neighbor_poly] = tentative_g_score;
|
||||
res->poly_f_score[neighbor_poly] = tentative_f_score;
|
||||
|
||||
//Linking neighbor node to current node (for tracing path)
|
||||
res->poly_prev_poly[neighbor_poly] = -1;
|
||||
res->poly_prev_traversal[neighbor_poly] = current_traversal;
|
||||
|
||||
// print("Assigning Poly ");
|
||||
// print(ftos(neighbor_poly));
|
||||
// print(" came from traversal ");
|
||||
// print(ftos(current_traversal));
|
||||
// print(".\n");
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
float best_openset_poly = sv_pathfind_get_lowest_f_score_poly(res);
|
||||
float best_openset_traversal = sv_pathfind_get_lowest_f_score_traversal(res);
|
||||
|
||||
// print("Best openset poly: ");
|
||||
// print(ftos(best_openset_poly));
|
||||
// print("\n");
|
||||
// print("Best openset traversal: ");
|
||||
// print(ftos(best_openset_traversal));
|
||||
// print("\n");
|
||||
|
||||
// If we have both, compare their f-scores
|
||||
if(best_openset_poly != -1 && best_openset_traversal != -1) {
|
||||
// If traversal score is better, don't consider polygon
|
||||
if(res->traversal_f_score[best_openset_poly] < res->poly_f_score[best_openset_poly]) {
|
||||
best_openset_poly = -1;
|
||||
}
|
||||
// If polygon score is better, don't consider traversal
|
||||
else {
|
||||
best_openset_traversal = -1;
|
||||
}
|
||||
}
|
||||
// Assign what we found, loop will terminate if we have neither
|
||||
current_traversal = best_openset_traversal;
|
||||
current_poly = best_openset_poly;
|
||||
// --------------------------------------------------------------------
|
||||
}
|
||||
|
||||
|
||||
|
||||
|
||||
//Tracing the pathfind results
|
||||
if(pathfind_success == TRUE) {
|
||||
sv_pathfind_trace_path(start_poly, goal_poly, res);
|
||||
sv_pathfind_smooth_path(start_pos, end_pos, res);
|
||||
return 1;
|
||||
}
|
||||
print("Pathfind fail");
|
||||
return 0;
|
||||
}
|
||||
|
||||
|
||||
|
||||
// float sv_navmesh_pathfind_start(float start, float goal, vector start_pos, vector end_pos, navmesh_pathfind_result* res) {
|
||||
// if(start == -1) {
|
||||
// //print("Error: pathfind start node invalid.\n");
|
||||
|
@ -748,32 +1286,26 @@ float sv_navmesh_pathfind_start(float start, float goal, vector start_pos, vecto
|
|||
// return 0;
|
||||
// }
|
||||
|
||||
float sv_navmesh_pathfind(entity from, entity to) {}
|
||||
// float sv_navmesh_pathfind(entity from, entity to) {
|
||||
// // TODO ... How do I get...?
|
||||
// // Every AI_Chase entity gets an integer
|
||||
// // TODO - In pathfinding, each entity needs a place to store its results...
|
||||
// // TODO - I should really make a MAX_AI number
|
||||
// // TODO - I can have an array of pre-allocated pathfind results, and dynamically use them?
|
||||
// // TODO - Or maybe I have one reserved per AI, and one supplemental one reesrved per AI...
|
||||
float sv_navmesh_pathfind(entity from, entity to) {
|
||||
// TODO ... How do I get...?
|
||||
// Every AI_Chase entity gets an integer
|
||||
// TODO - In pathfinding, each entity needs a place to store its results...
|
||||
// TODO - I should really make a MAX_AI number
|
||||
// TODO - I can have an array of pre-allocated pathfind results, and dynamically use them?
|
||||
// TODO - Or maybe I have one reserved per AI, and one supplemental one reesrved per AI...
|
||||
// FIXME - Will need to pre-allocate a certain number of AI_Chase ents and reuse them... or maybe
|
||||
// FIXME I need to pre-allocate a certain number of zombies, and dogs?
|
||||
// FIXME I wanted to subclass zombies to crawlers to override their behaviors...
|
||||
// We can perform pathfinding logic without losing our current path.
|
||||
|
||||
|
||||
|
||||
// // FIXME - Will need to pre-allocate a certain number of AI_Chase ents and reuse them... or maybe
|
||||
// // FIXME I need to pre-allocate a certain number of zombies, and dogs?
|
||||
// // FIXME I wanted to subclass zombies to crawlers to override their behaviors...
|
||||
|
||||
|
||||
// // We can perform pathfinding logic without losing our current path.
|
||||
|
||||
// // TODO - In pathfinding, I need a reference to the calling entity to know which traversal types this entity can use
|
||||
// AI_Chase chase_ent = (AI_Chase) from;
|
||||
// navmesh_pathfind_result* res = &(sv_zombie_pathfind_result[chase_ent.pathfind_result_idx]);
|
||||
// TODO - In pathfinding, I need a reference to the calling entity to know which traversal types this entity can use
|
||||
AI_Chase chase_ent = (AI_Chase) from;
|
||||
navmesh_pathfind_result* res = &(sv_zombie_pathfind_result[chase_ent.pathfind_result_idx]);
|
||||
|
||||
// float start = sv_navmesh_get_containing_poly(from.origin);
|
||||
// float goal = sv_navmesh_get_containing_poly(to.origin);
|
||||
float start = sv_navmesh_get_containing_poly(from.origin);
|
||||
float goal = sv_navmesh_get_containing_poly(to.origin);
|
||||
|
||||
// return sv_navmesh_pathfind_start(start,goal,from.origin,to.origin,res);
|
||||
// }
|
||||
return sv_navmesh_pathfind_start(start,goal,from.origin,to.origin,res);
|
||||
}
|
||||
|
||||
//===========================================================================================================
|
||||
|
|
|
@ -532,6 +532,7 @@ class AI_Chase : entity {
|
|||
// ------------------------------
|
||||
|
||||
float pathfind_result_idx; // TODO - Need to increment this on instantiation
|
||||
float pathfind_cur_point_idx;
|
||||
|
||||
|
||||
// Constructor. Called when calling `spawn(AI_Chase);`
|
||||
|
|
Loading…
Reference in a new issue