//
//

#ifndef FALCONNLITE_H
#define FALCONNLITE_H

#include "Header.h"

class FalconnLite {

private:
    int n_rotate = 3;
    int fhtDim;

protected:

    int n_points;
    int n_features;

    int n_proj = 1024;
    int iProbe = 5;
    int top_m = 50; // for dense bucket

    int qProbe = 5;

    int n_threads = 8;

    float ker_sigma = 1.0;
    int ker_n_features = 1024;
    float ker_intervalSampling = 0.4;

    bool verbose = false;

    string output;

    string distance = "Cosine";
    int seed = -1;

    // 1 Layer
    boost::dynamic_bitset<> bitHD;

    // New implementation of 2 layers of random sign
    boost::dynamic_bitset<> bitHD1;
    boost::dynamic_bitset<> bitHD2;

    // for Fourier embedding on L1 and L2
    // Need to store it for sDbscan-1NN with L1 and L2
    // Chi^2 and JS do not need as its embeddings are deterministic
    MatrixXf matrix_R;

    MatrixXi matrix_qProbes; // For each point (each col), keep topK closest/furthest random vectors
    vector< IVector> vec2D_Buckets;
    vector< vector<IFPair> > vec2D_Pair_Buckets; // useful for dot product or estimate distance

    // Note: If you want to read matrix_kNN from Python via def_readonly, it must be public.
    // However, vector can be accessed via def_readonly
    // RowMajorMatrixXi matrix_kNN;

    RowMajorMatrixXf matrix_X; // public as we will have to load data into it (when data is big)


public:



    FalconnLite(int n, int d){
        n_points = n;
        n_features = d;
    }

//    void set_params(int numProj = 1024, int i = 5, int l = 1000, int q = 5, string dist = "Cosine",
//                   int kDim = 1024, float kSigma = 1.0, float kSam = 0.4,
//                   bool ver = false, int numThreads = 8, int randomSeed = -1, string filename = ""){

    void set_params(int numProj = 1024, int i = 5, int m = 100, int q = 5, string dist = "Cosine",
                    int kDim=1024, float kSigma=1.0,
                    bool ver = false, int numThreads = 8, int randomSeed = -1){

        n_proj = numProj;

        iProbe = i;
        top_m = m;

        qProbe = q;

        distance = dist;
        ker_n_features = kDim;
        ker_sigma = kSigma;
//        ker_intervalSampling = kSam;
        verbose = ver;

        set_threads(numThreads);

        seed = randomSeed;
//        output = filename;

        // Must set
        if (distance == "Cosine") {
            ker_n_features = n_features;
        }

        // have to set fhtDim
        if (distance == "Cosine")
        {
            // Must set
            ker_n_features = n_features;

            if (n_proj <= n_features){
                fhtDim = 1 << int(ceil(log2(n_features)));
            }
            else{
                fhtDim = 1 << int(ceil(log2(n_proj)));
            }
        }
        else // the rest uses kernel embedding
        {
            if (n_proj <= ker_n_features)
                fhtDim = 1 << int(ceil(log2(ker_n_features)));
            else
                fhtDim = 1 << int(ceil(log2(n_proj)));
        }
        std::cout << "[set_params] distance = " << distance
          << ", ker_n_features = " << ker_n_features
          << ", ker_sigma = " << ker_sigma
          << ", fhtDim = " << fhtDim << std::endl;

    }



    void clear(){

        bitHD.clear();

        bitHD1.clear();
        bitHD2.clear();

        matrix_R.resize(0, 0); // Random matrix for Fourier features (L1, L2)
        matrix_qProbes.resize(0, 0); // For each point (each col), keep topK closest/furthest random vectors

        vec2D_Buckets.clear(); // vector of approx neighborhoods and its distances
        vec2D_Pair_Buckets.clear();

        // matrix_kNN.resize(0, 0);


    }

    ~FalconnLite(){
        matrix_X.resize(0, 0);
        clear();
    }


    void set_qProbe(int q){ qProbe = q; }
    void set_iProbe(int i){ iProbe = i; }
    void set_top_m(int m){ top_m = m; }

    bool centering = false;

    void set_threads(int t)
    {
        if (t <= 0)
            n_threads = omp_get_max_threads();
        else
            n_threads = t;
    }

    RowMajorMatrixXi bucket_sampling(const Ref<const RowMajorMatrixXf> &, int ); // return kNN matrix in row-major order (due to numpy)
    RowMajorMatrixXi freq_counting_old(const Ref<const RowMajorMatrixXf> & , int , int);

    RowMajorMatrixXi coll_counting(const Ref<const RowMajorMatrixXf> & , int , int);
    RowMajorMatrixXi dist_estimating(const Ref<const RowMajorMatrixXf> & , int , int);
    tuple<RowMajorMatrixXi, RowMajorMatrixXf> approx_kNN(const Ref<const RowMajorMatrixXf> &, int, int); // return kNN matrix in row-major order (due to numpy)
    tuple<RowMajorMatrixXi, RowMajorMatrixXf> approx_join(const Ref<const RowMajorMatrixXf> &, int, int);
private:

    //    void buildKernelFeatures(); // TODO: add build kernel features as a function
    // void rp_parIndex1(); // 1 layer
    void fht_Index2(); // 2 layers like Falconn++

    void fht_Index2_repeat(int);
    void fht_Index2_asym_centering_repeat(int);

    void fht_pairIndex2_repeat(int);
    void fht_pairIndex2_asym_centering_repeat(int);

    // void bf_findNeighborhood(int);
    // void rp_findNeighborhood0();
    RowMajorMatrixXi rp_sampleNeighborhood2(int);
};


#endif // FALCONNLITE_H

