#include "distance.h"

namespace {

inline double euc_2d(double dx, double dy) {
    double d = std::sqrt(dx * dx + dy * dy);
    return std::floor(d + 0.5);
}

} // namespace

DistanceProvider::DistanceProvider(const Instance &inst, const Config &cfg) : inst_(inst) {
    bool want_precompute = false;
    if (cfg.distance_mode == "precompute") {
        want_precompute = true;
    } else if (cfg.distance_mode == "auto") {
        want_precompute = inst.n <= cfg.distance_precompute_limit;
    }

    if (want_precompute) {
        precompute_ = true;
        dist_ = std::make_unique<TriMatrix>(inst.n);
        for (int i = 0; i < inst.n; ++i) {
            for (int j = i + 1; j < inst.n; ++j) {
                double dx = inst.x[i] - inst.x[j];
                double dy = inst.y[i] - inst.y[j];
                float d = static_cast<float>(euc_2d(dx, dy));
                dist_->set(i, j, d);
            }
        }
    }
}

double DistanceProvider::distance(int a, int b) const {
    if (a == b) {
        return 0.0;
    }
    if (precompute_) {
        return dist_->get(a, b);
    }
    double dx = inst_.x[a] - inst_.x[b];
    double dy = inst_.y[a] - inst_.y[b];
    return euc_2d(dx, dy);
}

double DistanceProvider::distance_sq(int a, int b) const {
    if (a == b) {
        return 0.0;
    }
    if (precompute_) {
        double d = dist_->get(a, b);
        return d * d;
    }
    double dx = inst_.x[a] - inst_.x[b];
    double dy = inst_.y[a] - inst_.y[b];
    double d = euc_2d(dx, dy);
    return d * d;
}

double DistanceProvider::inv_dist_beta(int a, int b, double beta) const {
    if (beta == 2.0) {
        double d2 = distance_sq(a, b);
        return d2 <= 0.0 ? 0.0 : (1.0 / d2);
    }
    double d = distance(a, b);
    d = std::max(d, 1e-12);
    if (beta == 1.0) {
        return 1.0 / d;
    }
    return std::pow(1.0 / d, beta);
}
