#ifndef TSP_2OPT_H
#define TSP_2OPT_H

#include "TSP_Basic_Functions.h"

// Evaluate the delta after applying a 2-opt move (delta >0 indicates an
// improving solution)
Distance_Type Get_2Opt_Delta(int First_City, int Second_City)
{
    if (Check_If_Two_City_Same_Or_Adjacent(First_City, Second_City) == true)
        return -Inf_Cost;

    int First_Next_City = All_Node[First_City].Next_City;
    int Second_Next_City = All_Node[Second_City].Next_City;

    Distance_Type Delta = Get_Distance(First_City, First_Next_City) + Get_Distance(Second_City, Second_Next_City) -
                          Get_Distance(First_City, Second_City) - Get_Distance(First_Next_City, Second_Next_City);
    // Update the Chosen_Times[][] and Total_Simulation_Times which are used in
    // MCTS
    Chosen_Times[First_City][Second_City]++;
    // Chosen_Times[Second_City][First_City]++;
    Chosen_Times[First_Next_City][Second_Next_City]++;
    // Chosen_Times[Second_Next_City][First_Next_City]++;
    Total_Simulation_Times++;

    return Delta;
}

// Apply a chosen 2-opt move
void Apply_2Opt_Move(int First_City, int Second_City)
{
    Distance_Type Before_Distance = Get_Solution_Total_Distance();
    Distance_Type Delta = Get_2Opt_Delta(First_City, Second_City);

    int First_Next_City = All_Node[First_City].Next_City;
    int Second_Next_City = All_Node[Second_City].Next_City;

    Reverse_Sub_Path(First_Next_City, Second_City);
    All_Node[First_City].Next_City = Second_City;
    All_Node[Second_City].Pre_City = First_City;
    All_Node[First_Next_City].Next_City = Second_Next_City;
    All_Node[Second_Next_City].Pre_City = First_Next_City;

    // Update the values of matrix Weight[][] by back propagation, which would
    // be used in MCTS
    float Increase_Rate = Beta * (pow(2.718, (float)(Delta) / (float)(Before_Distance)) - 1);

    Weight[First_City][Second_City] += Increase_Rate;
    Weight[Second_City][First_City] += Increase_Rate;
    Weight[First_Next_City][Second_Next_City] += Increase_Rate;
    Weight[Second_Next_City][First_Next_City] += Increase_Rate;
}

bool Improve_By_2Opt_Move()
{
    bool If_Improved = false;
    Distance_Type Delta = 0;
    for (int i = 0; i < Virtual_City_Num; i++)
        for (int j = 0; j < Candidate_Num[i]; j++)
        {
            int Candidate_City = Candidate[i][j];
            Delta = Get_2Opt_Delta(i, Candidate_City);
            if (Delta > 0)
            {
                // cout << "Improving by 2-opt move: " << i << " " <<
                // Candidate_City << " Delta: " << Delta << endl;
                Apply_2Opt_Move(i, Candidate_City);
                If_Improved = true;
                break;
            }
        }

    return If_Improved;
}

// Iteratively apply an improving 2-opt move until no improvement is possible
void Local_Search_by_2Opt_Move()
{
    while (Improve_By_2Opt_Move() == true)
        ;
    if (MCTS_Debug)
        cout << "Local search by 2-opt move finished." << endl;

    Distance_Type Cur_Solution_Total_Distance = Get_Solution_Total_Distance();
    if (Cur_Solution_Total_Distance < Current_Instance_Best_Distance)
    {
        // Store the information of the best found solution to Struct_Node
        // *Best_All_Node
        Current_Instance_Best_Distance = Cur_Solution_Total_Distance;
        Store_Best_Solution();
        if (Log_Length_Time)
        {
            Length_Time.push_back(
                std::make_pair(Current_Instance_Best_Distance, Get_Elapsed_Time(Current_Instance_Begin_Time)));
        }
    }
}

#endif // TSP_2OPT_H
