//
// Created by xuec on 2026/1/20.
//

#ifndef COVERAGE_DENG_ROI_H
#define COVERAGE_DENG_ROI_H
#include <chrono>
#include <iomanip>
#include <queue>
#include "utilityfunction.h"

struct LazyNode {
    int id;
    double score;
    int iteration;

    bool operator<(const LazyNode& other) const {
        return score < other.score;
    }
};


S_class DistortedGreedy_lazy( long long int& oracle_times, int type, vector<S_class>& snapshots){

    S_class S = S_class();
    snapshots.clear();

    snapshots.push_back(S);

    priority_queue<LazyNode> pq;
    for (int u = 0; u < node_num; u++) {
        double mg = S.marginal_gain_g(u);
        oracle_times++;
        double initial_score = 0;
        if (type == 1) initial_score = (mg - Groundset[u].cost) / Groundset[u].cost; // ROI
        else initial_score = mg - 2 * Groundset[u].cost; // Cost-scaled

        pq.push({u, initial_score, 0});
    }
    while (!pq.empty()){
        LazyNode top = pq.top();
        pq.pop();

        if (S.selected[top.id]) continue;
        if(top.iteration < S.solution.size()){

            double mg = S.marginal_gain_g(top.id);
            oracle_times++;
            double new_score = 0;
            if (type == 1) new_score = (mg - Groundset[top.id].cost) / Groundset[top.id].cost;
            else new_score = mg - 2 * Groundset[top.id].cost;

            top.score= new_score;
            top.iteration = S.solution.size();
            pq.push(top);

            continue;
        }


        if(top.score > 1e-9){
            S.add_element(0.0,top.id);

            snapshots.push_back(S);
        }
        else{ break;}

    }


    return S;

}


Result Deng_ROI (double Budget) {

    cout << "Deng-ROI & Budget: " << Budget << "---------start---------" << endl;

    auto start = std::chrono::high_resolution_clock::now();

    long long int oracle_times = 0;

    int type = 1;// 1: ROI; 2: cost scaled
    vector<S_class> snapshots;
    S_class S_star = DistortedGreedy_lazy(oracle_times,type,snapshots);

    vector<double> payments(node_num, 0.0);


    for (int idx = 0; idx<S_star.solution_order.size();++idx) {
        int i = S_star.solution_order[idx];
        S_class current_S = snapshots[idx];
        double p_i = 0.0;

        priority_queue<LazyNode> pq;
        for (int u = 0; u < node_num; u++) {
            if (u == i) continue;

            double mg = current_S.marginal_gain_g(u);
            oracle_times++;
            double score=0;
            if (type==1) score = (mg - Groundset[u].cost)/Groundset[u].cost;
            else score = mg - 2 * Groundset[u].cost;    // Cost-scaled
            pq.push({u, score, (int)current_S.solution.size()});

        }

        for (int k = current_S.solution.size(); k <= node_num; k++) {

            double mg_i = current_S.marginal_gain_g(i);
            oracle_times++;


            double max_other_G = -1e18;
            int best_other_u = -1;

            while (!pq.empty()) {
                LazyNode top = pq.top();
                pq.pop();


                if (current_S.selected[top.id]) continue;


                if (top.iteration < current_S.solution.size()) {
                    double mg_u = current_S.marginal_gain_g(top.id);
                    oracle_times++;

                    if (type==1) top.score =(mg_u - Groundset[top.id].cost)/Groundset[top.id].cost;
                    else top.score = mg_u - 2 * Groundset[top.id].cost; // Cost-scaled
                    top.iteration = k;
                    pq.push(top);
                    continue;
                }


                max_other_G = top.score;
                best_other_u = top.id;
                pq.push(top);
                break;
            }


           //************* compute price ***************
            double threshold = 0;
            double max_other = max(0.0, max_other_G);
            if(type ==1) threshold = mg_i / (1.0 + max_other);
            else  threshold = (mg_i - max_other)/2.0;

            if (threshold > p_i) p_i = threshold;


            if (best_other_u != -1 && max_other_G > 1e-9) {
                current_S.add_element(0.0, best_other_u);
            }
        }
        payments[i] = p_i;
    }


    // ********************** check budget and cut off ******************
    double total_payment = 0.0;
    for (int i : S_star.solution_order) {
        total_payment += payments[i];
    }

    while (total_payment > Budget + 1e-9 && !S_star.solution_order.empty()) {
        int last_id = S_star.solution_order.back();
        total_payment -= payments[last_id];
        S_star.S_cost -= Groundset[last_id].cost;
        payments[last_id] = 0.0;

        S_star.solution.erase(last_id);
        S_star.solution_order.pop_back();
        if(last_id < node_num) S_star.selected[last_id] = 0;

    }

    S_star.S_price = total_payment;

    S_star.S_revenue = S_star.f_S() ;
    oracle_times++;


    auto end = std::chrono::high_resolution_clock::now();

    auto duration = std::chrono::duration_cast<std::chrono::nanoseconds>(end - start);
    long long int total_time_ns = duration.count();

//    cout<<"running time: "<<total_time_ns<<endl;
//    cout<<"S*:"<<endl;
//    cout<<"  revenue: "<<S_star.S_revenue<<" cost: "<<S_star.S_cost<<"Price: "<<S_star.S_price<<" size: "<<S_star.solution_order.size()<<endl;
//    for(const auto &p:S_star.solution_order)
//        cout<<p<<" ";
//    cout<<endl;
//
//    cout<<endl;
//    cout<<"oracle times: "<<oracle_times<<endl;

    cout<<"objective values: "<<S_star.S_revenue<<endl;
    cout<<"oracle queries: "<<oracle_times<<endl;
    cout<<"Deng_ROI ---------end--------- "<<endl<<endl;

    return Result(S_star.S_revenue,S_star.S_cost,S_star.S_price,S_star.solution.size(),oracle_times,total_time_ns);
}

#endif //COVERAGE_DENG_ROI_H
