#include <iostream>
#include <vector>
#include <queue>
#include <cmath>
#include <unordered_set>
#include <unordered_map>
#include <algorithm>
#include <limits>
#include <iomanip>
using namespace std;
struct Point {
double x, y;
} ;
struct AStarNode {
int id;
double f, g;
bool operator> ( const AStarNode& other) const {
return f > other.f ;
}
} ;
unordered_map< string, int > node_ids;
unordered_map< int , string> node_names;
int current_id = 0 ;
int getId( const string& name) {
if ( node_ids.find ( name) == node_ids.end ( ) ) {
node_ids[ name] = current_id;
node_names[ current_id] = name;
current_id++ ;
}
return node_ids[ name] ;
}
double calculate_heuristic( int node_id, const vector< Point> & coordinates, const unordered_set< int > & exits) {
double min_dist = numeric_limits< double > :: infinity ( ) ;
for ( int exit_id : exits) {
double dx = coordinates[ node_id] .x - coordinates[ exit_id] .x ;
double dy = coordinates[ node_id] .y - coordinates[ exit_id] .y ;
double feet_dist = sqrt ( dx * dx + dy * dy) * 0.0729 ;
if ( feet_dist < min_dist) min_dist = feet_dist;
}
return min_dist;
}
vector< int > a_star_evacuation(
int start_node,
const unordered_set< int > & exits,
const vector< vector< pair< int , double >>> & graph,
const vector< Point> & coordinates,
const unordered_set< int > & fire_nodes,
double & out_distance)
{
int num_nodes = graph.size ( ) ;
vector< double > g_score( num_nodes, numeric_limits< double > :: infinity ( ) ) ;
g_score[ start_node] = 0.0 ;
vector< int > parent( num_nodes, - 1 ) ;
priority_queue< AStarNode, vector< AStarNode> , greater< AStarNode>> open_set;
open_set.push ( { start_node, calculate_heuristic( start_node, coordinates, exits) , 0.0 } ) ;
while ( ! open_set.empty ( ) ) {
AStarNode current = open_set.top ( ) ;
open_set.pop ( ) ;
int u = current.id ;
if ( exits.count ( u) ) {
out_distance = current.g ;
vector< int > path;
int curr = u;
while ( curr ! = - 1 ) {
path.push_back ( curr) ;
curr = parent[ curr] ;
}
reverse( path.begin ( ) , path.end ( ) ) ;
return path;
}
if ( current.g > g_score[ u] ) continue ;
for ( const auto & edge : graph[ u] ) {
int v = edge.first ;
double weight = edge.second ;
if ( fire_nodes.count ( v) ) continue ;
double tentative_g = g_score[ u] + weight;
if ( tentative_g < g_score[ v] ) {
parent[ v] = u;
g_score[ v] = tentative_g;
double f = tentative_g + calculate_heuristic( v, coordinates, exits) ;
open_set.push ( { v, f, tentative_g} ) ;
}
}
}
return { } ;
}
int main( ) {
// 1. Initialize Coordinates
vector< Point> coords_temp( 38 ) ;
auto addNode = [ & ] ( string name, double x, double y) {
int id = getId( name) ;
coords_temp[ id] = { x, y} ;
} ;
addNode( "door[0]" , 638.0 , 213.0 ) ; addNode( "door[1]" , 680.0 , 1049.0 ) ;
addNode( "door[2]" , 1103.0 , 1420.0 ) ; addNode( "door[3]" , 1640.0 , 1626.0 ) ;
addNode( "door[4]" , 1635.0 , 1167.0 ) ; addNode( "door[5]" , 2358.0 , 1294.0 ) ;
addNode( "door[6]" , 2718.0 , 1578.0 ) ; addNode( "door[7]" , 2651.0 , 1367.0 ) ;
addNode( "door[8]" , 3132.0 , 1179.0 ) ; addNode( "door[9]" , 3525.0 , 1049.0 ) ;
addNode( "door[10]" , 3522.0 , 1505.0 ) ;
addNode( "exits[0]" , 638.0 , 1240.0 ) ; addNode( "exits[1]" , 1095.0 , 1917.0 ) ;
addNode( "exits[2]" , 2317.0 , 1767.0 ) ; addNode( "exits[3]" , 3038.0 , 959.0 ) ;
addNode( "coridor[0]" , 736.0 , 138.0 ) ; addNode( "coridor[1]" , 1220.0 , 818.0 ) ;
addNode( "coridor[2]" , 718.0 , 1182.0 ) ; addNode( "coridor[3]" , 884.0 , 1417.0 ) ;
addNode( "coridor[4]" , 1032.0 , 1309.0 ) ; addNode( "coridor[5]" , 1385.0 , 1049.0 ) ;
addNode( "coridor[6]" , 1186.0 , 1845.0 ) ; addNode( "coridor[7]" , 1473.0 , 1169.0 ) ;
addNode( "coridor[8]" , 1470.0 , 1045.0 ) ; addNode( "coridor[9]" , 1742.0 , 1545.0 ) ;
addNode( "coridor[10]" , 1934.0 , 1667.0 ) ; addNode( "coridor[11]" , 2320.0 , 1665.0 ) ;
addNode( "coridor[12]" , 2433.0 , 1665.0 ) ; addNode( "coridor[13]" , 2434.0 , 1046.0 ) ;
addNode( "coridor[14]" , 2434.0 , 1293.0 ) ; addNode( "coridor[15]" , 2559.0 , 1049.0 ) ;
addNode( "coridor[16]" , 2558.0 , 1368.0 ) ; addNode( "coridor[17]" , 2559.0 , 1668.0 ) ;
addNode( "coridor[18]" , 2721.0 , 1665.0 ) ; addNode( "coridor[19]" , 3523.0 , 1669.0 ) ;
addNode( "coridor[20]" , 3038.0 , 1049.0 ) ; addNode( "coridor[21]" , 3133.0 , 1048.0 ) ;
addNode( "coridor[22]" , 3398.0 , 1048.0 ) ;
// 2. Build Graph
vector< vector< pair< int , double >>> graph( 38 ) ;
auto addEdge = [ & ] ( string u, string v, double w) {
graph[ getId( u) ] .push_back ( { getId( v) , w} ) ;
graph[ getId( v) ] .push_back ( { getId( u) , w} ) ;
} ;
addEdge( "door[0]" , "coridor[0]" , 9.1 ) ; addEdge( "door[0]" , "door[1]" , 61.02 ) ;
addEdge( "door[1]" , "coridor[2]" , 9.95 ) ; addEdge( "door[2]" , "coridor[4]" , 9.33 ) ;
addEdge( "door[2]" , "door[3]" , 41.53 ) ; addEdge( "door[3]" , "coridor[9]" , 9.52 ) ;
addEdge( "door[4]" , "coridor[7]" , 12.02 ) ; addEdge( "door[4]" , "door[5]" , 52.80 ) ;
addEdge( "door[5]" , "coridor[14]" , 5.74 ) ; addEdge( "door[6]" , "door[7]" , 15.77 ) ;
addEdge( "door[6]" , "coridor[18]" , 6.48 ) ; addEdge( "door[7]" , "door[8]" , 37.92 ) ;
addEdge( "door[7]" , "coridor[16]" , 6.87 ) ; addEdge( "door[8]" , "coridor[21]" , 9.27 ) ;
addEdge( "door[9]" , "door[10]" , 33.09 ) ; addEdge( "door[9]" , "coridor[22]" , 9.13 ) ;
addEdge( "door[10]" , "coridor[19]" , 12.06 ) ;
addEdge( "exits[0]" , "coridor[2]" , 6.84 ) ; addEdge( "exits[1]" , "coridor[6]" , 8.39 ) ;
addEdge( "exits[2]" , "coridor[11]" , 6.53 ) ; addEdge( "exits[3]" , "coridor[20]" , 6.49 ) ;
addEdge( "coridor[0]" , "coridor[1]" , 60.81 ) ; addEdge( "coridor[1]" , "coridor[2]" , 45.29 ) ;
addEdge( "coridor[1]" , "coridor[5]" , 20.31 ) ; addEdge( "coridor[2]" , "coridor[3]" , 20.98 ) ;
addEdge( "coridor[3]" , "coridor[4]" , 13.36 ) ; addEdge( "coridor[3]" , "coridor[6]" , 38.41 ) ;
addEdge( "coridor[4]" , "coridor[5]" , 31.51 ) ; addEdge( "coridor[5]" , "coridor[7]" , 10.75 ) ;
addEdge( "coridor[5]" , "coridor[8]" , 6.08 ) ; addEdge( "coridor[7]" , "coridor[9]" , 34.22 ) ;
addEdge( "coridor[8]" , "coridor[13]" , 70.45 ) ; addEdge( "coridor[9]" , "coridor[10]" , 16.29 ) ;
addEdge( "coridor[10]" , "coridor[11]" , 28.24 ) ; addEdge( "coridor[11]" , "coridor[12]" , 8.52 ) ;
addEdge( "coridor[12]" , "coridor[14]" , 26.64 ) ; addEdge( "coridor[12]" , "coridor[17]" , 8.92 ) ;
addEdge( "coridor[13]" , "coridor[14]" , 17.76 ) ; addEdge( "coridor[13]" , "coridor[15]" , 8.88 ) ;
addEdge( "coridor[15]" , "coridor[16]" , 23.21 ) ; addEdge( "coridor[15]" , "coridor[20]" , 34.74 ) ;
addEdge( "coridor[16]" , "coridor[17]" , 21.24 ) ; addEdge( "coridor[17]" , "coridor[18]" , 11.64 ) ;
addEdge( "coridor[18]" , "coridor[19]" , 59.08 ) ; addEdge( "coridor[20]" , "coridor[21]" , 7.02 ) ;
addEdge( "coridor[21]" , "coridor[22]" , 19.48 ) ;
unordered_set< int > exit_ids = {
getId( "exits[0]" ) , getId( "exits[1]" ) ,
getId( "exits[2]" ) , getId( "exits[3]" )
} ;
vector< string> all_doors = {
"door[0]" , "door[1]" , "door[2]" , "door[3]" , "door[4]" ,
"door[5]" , "door[6]" , "door[7]" , "door[8]" , "door[9]" , "door[10]"
} ;
cout << "--- EVACUATION SYSTEM CONTROL ---\n " ;
cout << "Enter the number of active fire zones: " ;
int num_fires;
cin >> num_fires;
unordered_set< int > active_fires;
for ( int i = 0 ; i < num_fires; ++ i) {
string fire_location;
cout << "Enter location for Fire " << ( i + 1 ) << " (e.g., coridor[15], door[1]): " ;
cin >> fire_location;
if ( node_ids.find ( fire_location) ! = node_ids.end ( ) ) {
active_fires.insert ( getId( fire_location) ) ;
} else {
cout << "Warning: Node '" << fire_location << "' does not exist. Skipping.\n " ;
}
}
cout << "\n Calculating safest routes avoiding all fire zones...\n " ;
cout << string( 60 , '=' ) << "\n \n " ;
for ( const string& door_name : all_doors) {
cout << "From " << door_name << ":\n " ;
if ( active_fires.count ( getId( door_name) ) ) {
cout << " [!] THIS DOOR IS ON FIRE. EVACUATION IMPOSSIBLE.\n \n " ;
continue ;
}
double total_distance = 0.0 ;
vector< int > path = a_star_evacuation( getId( door_name) , exit_ids, graph, coords_temp, active_fires, total_distance) ;
if ( ! path.empty ( ) ) {
cout << " Path: " ;
for ( int i = 0 ; i < path.size ( ) ; ++ i) {
cout << node_names[ path[ i] ] << ( i == path.size ( ) - 1 ? "" : " -> " ) ;
}
double clear_time_sec = total_distance / 4.0 ;
double smoke_time_sec = total_distance / 1.5 ;
cout << fixed << setprecision( 2 ) ;
cout << "\n Distance: " << total_distance << " ft" ;
cout << "\n Est. Time (Clear): " << clear_time_sec << " seconds" ;
cout << "\n Est. Time (Smoky): " << smoke_time_sec << " seconds\n \n " ;
} else {
cout << " [!] NO SAFE ROUTE. FIRES BLOCK ALL PATHS TO EXITS.\n \n " ;
}
}
return 0 ;
}
#include <iostream>
#include <vector>
#include <queue>
#include <cmath>
#include <unordered_set>
#include <unordered_map>
#include <algorithm>
#include <limits>
#include <iomanip> 

using namespace std;

struct Point {
    double x, y;
};

struct AStarNode {
    int id;
    double f, g;
    bool operator>(const AStarNode& other) const {
        return f > other.f;
    }
};

unordered_map<string, int> node_ids;
unordered_map<int, string> node_names;
int current_id = 0;

int getId(const string& name) {
    if (node_ids.find(name) == node_ids.end()) {
        node_ids[name] = current_id;
        node_names[current_id] = name;
        current_id++;
    }
    return node_ids[name];
}

double calculate_heuristic(int node_id, const vector<Point>& coordinates, const unordered_set<int>& exits) {
    double min_dist = numeric_limits<double>::infinity();
    for (int exit_id : exits) {
        double dx = coordinates[node_id].x - coordinates[exit_id].x;
        double dy = coordinates[node_id].y - coordinates[exit_id].y;
        double feet_dist = sqrt(dx * dx + dy * dy) * 0.0729; 
        if (feet_dist < min_dist) min_dist = feet_dist;
    }
    return min_dist;
}


vector<int> a_star_evacuation(
    int start_node, 
    const unordered_set<int>& exits, 
    const vector<vector<pair<int, double>>>& graph, 
    const vector<Point>& coordinates,
    const unordered_set<int>& fire_nodes, 
    double& out_distance) 
{
    int num_nodes = graph.size();
    vector<double> g_score(num_nodes, numeric_limits<double>::infinity());
    g_score[start_node] = 0.0;
    vector<int> parent(num_nodes, -1);

    priority_queue<AStarNode, vector<AStarNode>, greater<AStarNode>> open_set;
    open_set.push({start_node, calculate_heuristic(start_node, coordinates, exits), 0.0});

    while (!open_set.empty()) {
        AStarNode current = open_set.top();
        open_set.pop();
        int u = current.id;

        if (exits.count(u)) {
            out_distance = current.g; 
            vector<int> path;
            int curr = u;
            while (curr != -1) {
                path.push_back(curr);
                curr = parent[curr];
            }
            reverse(path.begin(), path.end());
            return path;
        }

        if (current.g > g_score[u]) continue;

        for (const auto& edge : graph[u]) {
            int v = edge.first;
            double weight = edge.second;

            if (fire_nodes.count(v)) continue; 

            double tentative_g = g_score[u] + weight;

            if (tentative_g < g_score[v]) {
                parent[v] = u;
                g_score[v] = tentative_g;
                double f = tentative_g + calculate_heuristic(v, coordinates, exits);
                open_set.push({v, f, tentative_g});
            }
        }
    }
    return {}; 
}

int main() {
    // 1. Initialize Coordinates
    vector<Point> coords_temp(38);
    auto addNode = [&](string name, double x, double y) {
        int id = getId(name);
        coords_temp[id] = {x, y};
    };

    addNode("door[0]", 638.0, 213.0); addNode("door[1]", 680.0, 1049.0);
    addNode("door[2]", 1103.0, 1420.0); addNode("door[3]", 1640.0, 1626.0);
    addNode("door[4]", 1635.0, 1167.0); addNode("door[5]", 2358.0, 1294.0);
    addNode("door[6]", 2718.0, 1578.0); addNode("door[7]", 2651.0, 1367.0);
    addNode("door[8]", 3132.0, 1179.0); addNode("door[9]", 3525.0, 1049.0);
    addNode("door[10]", 3522.0, 1505.0);

    addNode("exits[0]", 638.0, 1240.0); addNode("exits[1]", 1095.0, 1917.0);
    addNode("exits[2]", 2317.0, 1767.0); addNode("exits[3]", 3038.0, 959.0);

    addNode("coridor[0]", 736.0, 138.0); addNode("coridor[1]", 1220.0, 818.0);
    addNode("coridor[2]", 718.0, 1182.0); addNode("coridor[3]", 884.0, 1417.0);
    addNode("coridor[4]", 1032.0, 1309.0); addNode("coridor[5]", 1385.0, 1049.0);
    addNode("coridor[6]", 1186.0, 1845.0); addNode("coridor[7]", 1473.0, 1169.0);
    addNode("coridor[8]", 1470.0, 1045.0); addNode("coridor[9]", 1742.0, 1545.0);
    addNode("coridor[10]", 1934.0, 1667.0); addNode("coridor[11]", 2320.0, 1665.0);
    addNode("coridor[12]", 2433.0, 1665.0); addNode("coridor[13]", 2434.0, 1046.0);
    addNode("coridor[14]", 2434.0, 1293.0); addNode("coridor[15]", 2559.0, 1049.0);
    addNode("coridor[16]", 2558.0, 1368.0); addNode("coridor[17]", 2559.0, 1668.0);
    addNode("coridor[18]", 2721.0, 1665.0); addNode("coridor[19]", 3523.0, 1669.0);
    addNode("coridor[20]", 3038.0, 1049.0); addNode("coridor[21]", 3133.0, 1048.0);
    addNode("coridor[22]", 3398.0, 1048.0);

    // 2. Build Graph
    vector<vector<pair<int, double>>> graph(38);
    auto addEdge = [&](string u, string v, double w) {
        graph[getId(u)].push_back({getId(v), w});
        graph[getId(v)].push_back({getId(u), w}); 
    };

    addEdge("door[0]", "coridor[0]", 9.1); addEdge("door[0]", "door[1]", 61.02);
    addEdge("door[1]", "coridor[2]", 9.95); addEdge("door[2]", "coridor[4]", 9.33);
    addEdge("door[2]", "door[3]", 41.53); addEdge("door[3]", "coridor[9]", 9.52);
    addEdge("door[4]", "coridor[7]", 12.02); addEdge("door[4]", "door[5]", 52.80);
    addEdge("door[5]", "coridor[14]", 5.74); addEdge("door[6]", "door[7]", 15.77);
    addEdge("door[6]", "coridor[18]", 6.48); addEdge("door[7]", "door[8]", 37.92);
    addEdge("door[7]", "coridor[16]", 6.87); addEdge("door[8]", "coridor[21]", 9.27);
    addEdge("door[9]", "door[10]", 33.09); addEdge("door[9]", "coridor[22]", 9.13);
    addEdge("door[10]", "coridor[19]", 12.06);
    addEdge("exits[0]", "coridor[2]", 6.84); addEdge("exits[1]", "coridor[6]", 8.39);
    addEdge("exits[2]", "coridor[11]", 6.53); addEdge("exits[3]", "coridor[20]", 6.49);
    addEdge("coridor[0]", "coridor[1]", 60.81); addEdge("coridor[1]", "coridor[2]", 45.29);
    addEdge("coridor[1]", "coridor[5]", 20.31); addEdge("coridor[2]", "coridor[3]", 20.98);
    addEdge("coridor[3]", "coridor[4]", 13.36); addEdge("coridor[3]", "coridor[6]", 38.41);
    addEdge("coridor[4]", "coridor[5]", 31.51); addEdge("coridor[5]", "coridor[7]", 10.75);
    addEdge("coridor[5]", "coridor[8]", 6.08); addEdge("coridor[7]", "coridor[9]", 34.22);
    addEdge("coridor[8]", "coridor[13]", 70.45); addEdge("coridor[9]", "coridor[10]", 16.29);
    addEdge("coridor[10]", "coridor[11]", 28.24); addEdge("coridor[11]", "coridor[12]", 8.52);
    addEdge("coridor[12]", "coridor[14]", 26.64); addEdge("coridor[12]", "coridor[17]", 8.92);
    addEdge("coridor[13]", "coridor[14]", 17.76); addEdge("coridor[13]", "coridor[15]", 8.88);
    addEdge("coridor[15]", "coridor[16]", 23.21); addEdge("coridor[15]", "coridor[20]", 34.74);
    addEdge("coridor[16]", "coridor[17]", 21.24); addEdge("coridor[17]", "coridor[18]", 11.64);
    addEdge("coridor[18]", "coridor[19]", 59.08); addEdge("coridor[20]", "coridor[21]", 7.02);
    addEdge("coridor[21]", "coridor[22]", 19.48);

    unordered_set<int> exit_ids = {
        getId("exits[0]"), getId("exits[1]"), 
        getId("exits[2]"), getId("exits[3]")
    };

    vector<string> all_doors = {
        "door[0]", "door[1]", "door[2]", "door[3]", "door[4]",
        "door[5]", "door[6]", "door[7]", "door[8]", "door[9]", "door[10]"
    };

    cout << "--- EVACUATION SYSTEM CONTROL ---\n";
    cout << "Enter the number of active fire zones: ";
    int num_fires;
    cin >> num_fires;

    unordered_set<int> active_fires;
    for (int i = 0; i < num_fires; ++i) {
        string fire_location;
        cout << "Enter location for Fire " << (i + 1) << " (e.g., coridor[15], door[1]): ";
        cin >> fire_location;

        if (node_ids.find(fire_location) != node_ids.end()) {
            active_fires.insert(getId(fire_location));
        } else {
            cout << "Warning: Node '" << fire_location << "' does not exist. Skipping.\n";
        }
    }

    cout << "\nCalculating safest routes avoiding all fire zones...\n";
    cout << string(60, '=') << "\n\n";

    for (const string& door_name : all_doors) {
        cout << "From " << door_name << ":\n";

        if (active_fires.count(getId(door_name))) {
            cout << "  [!] THIS DOOR IS ON FIRE. EVACUATION IMPOSSIBLE.\n\n";
            continue;
        }

        double total_distance = 0.0;
        vector<int> path = a_star_evacuation(getId(door_name), exit_ids, graph, coords_temp, active_fires, total_distance);

        if (!path.empty()) {
            cout << "  Path: ";
            for (int i = 0; i < path.size(); ++i) {
                cout << node_names[path[i]] << (i == path.size() - 1 ? "" : " -> ");
            }
            
            double clear_time_sec = total_distance / 4.0;
            double smoke_time_sec = total_distance / 1.5;

            cout << fixed << setprecision(2);
            cout << "\n  Distance: " << total_distance << " ft";
            cout << "\n  Est. Time (Clear): " << clear_time_sec << " seconds";
            cout << "\n  Est. Time (Smoky): " << smoke_time_sec << " seconds\n\n";
        } else {
            cout << "  [!] NO SAFE ROUTE. FIRES BLOCK ALL PATHS TO EXITS.\n\n";
        }
    }

    return 0;
}