#pragma once

#include <random>
#include <vector>

#include "candidates.h"
#include "distance.h"

// -----------------------------
// Local search
// -----------------------------
struct LSWorkspace {
    std::vector<int> tour_ext;
    std::vector<int> pos;
    std::vector<int> dlb;
    std::vector<int> random;
    std::vector<int> h_tour;
    std::vector<int> hh_tour;

    void ensure(int n);
};

void two_opt_first(const DistanceProvider &dist,
                   const CandidateList &cand,
                   int nn_ls,
                   bool use_dlb,
                   int max_passes,
                   std::vector<int> &tour,
                   std::mt19937_64 &rng,
                   LSWorkspace &ws);

void two_h_opt_first(const DistanceProvider &dist,
                     const CandidateList &cand,
                     int nn_ls,
                     bool use_dlb,
                     int max_passes,
                     std::vector<int> &tour,
                     std::mt19937_64 &rng,
                     LSWorkspace &ws);

void three_opt_first(const DistanceProvider &dist,
                     const CandidateList &cand,
                     int nn_ls,
                     bool use_dlb,
                     int max_passes,
                     std::vector<int> &tour,
                     std::mt19937_64 &rng,
                     LSWorkspace &ws);

double lk_search(const DistanceProvider &dist,
                 const CandidateList &cand,
                 std::vector<int> &tour,
                 int passes,
                 int max_depth);
