/*    This file is part of the MMA Library -
 * https://gitlab.inria.fr/dloiseau/multipers - which is released under MIT. See
 * file LICENSE for full license details. Author(s):       David Loiseaux
 *
 *    Copyright (C) 2021 Inria
 *
 *    Modification(s):
 *      - 2022/03 Hannah Schreiber: Integration of the new Vineyard_persistence
 * class, renaming and cleanup.
 */
/**
 * @file format_python-cpp.h
 * @author David Loiseaux, Hannah Schreiber
 * @brief Functions that change the format of data to communicate between C++
 * and python.
 */

#ifndef FORMAT_PYTHON_CPP_H_INCLUDED
#define FORMAT_PYTHON_CPP_H_INCLUDED

#include <algorithm>
#include <vector>

#include "combinatory.h"
#include "multiparameter_module_approximation/utilities.h"
#include <gudhi/Simplex_tree.h>
#include <gudhi/Simplex_tree/Simplex_tree_multi.h>

namespace Gudhi::multiparameter::mma {

// Lexical order + dimension
bool is_strictly_smaller_simplex(const boundary_type &s1,
                                 const boundary_type &s2) {
  if (s1.size() < s2.size())
    return true;
  if (s1.size() > s2.size())
    return false;

  for (unsigned int i = 0; i < s1.size(); i++) {
    if (s1[i] < s2[i])
      return true;
    if (s1[i] > s2[i])
      return false;
  }
  return false;
}

std::pair<boundary_matrix, multifiltration_type>
simplextree_to_boundary_filtration(const uintptr_t splxptr) {
  using option =
      Gudhi::multiparameter::Simplex_tree_options_multidimensional_filtration;
  Gudhi::Simplex_tree<option> &simplexTree =
      *(Gudhi::Simplex_tree<option> *)(splxptr);

  unsigned int numberOfSimplices = simplexTree.num_simplices();
  boundary_matrix boundaries(numberOfSimplices);
  boundary_matrix simplices(numberOfSimplices);
  if (simplexTree.num_simplices() <= 0)
    return {{}, {{}}};
  unsigned int filtration_number =
      simplexTree.filtration(*(simplexTree.complex_simplex_range().begin()))
          .size();
  std::vector<filtration_type> filtration(filtration_number,
                                          filtration_type(numberOfSimplices));

  unsigned int count = 0;
  for (auto sh : simplexTree.filtration_simplex_range())
    simplexTree.assign_key(sh, count++);

  unsigned int i = 0;
  for (auto &simplex : simplexTree.filtration_simplex_range()) {
    for (const auto &simplex_id : simplexTree.boundary_simplex_range(simplex)) {
      boundaries[i].push_back(simplexTree.key(simplex_id));
    }
    for (const auto &vertex : simplexTree.simplex_vertex_range(simplex)) {
      simplices[i].push_back(vertex);
    }
    const auto &temp = simplexTree.filtration(simplex);
    for (unsigned int j = 0; j < temp.size(); j++)
      filtration[j][i] = temp[j];
    i++;
  }
  for (boundary_type &simplex : simplices) {
    std::sort(simplex.begin(), simplex.end());
  }
  permutation_type p =
      Combinatorics::sort_and_return_permutation<boundary_type>(
          simplices, &is_strictly_smaller_simplex);

  for (auto &F : filtration) {
    Combinatorics::compose(F, p);
  }

  Combinatorics::compose(boundaries, p);

  auto inv = Combinatorics::inverse(p);

  for (boundary_type &simplex : boundaries) {
    for (auto &b : simplex)
      b = inv[b];
    std::sort(simplex.begin(), simplex.end());
  }

  return std::make_pair(boundaries, filtration);
}

using scc_type =
    std::vector<std::pair<boundary_matrix, std::vector<std::vector<float>>>>;
scc_type simplextree_to_scc(const uintptr_t splxptr) {
  using option =
      Gudhi::multiparameter::Simplex_tree_options_multidimensional_filtration;
  Gudhi::Simplex_tree<option> &st = *(Gudhi::Simplex_tree<option> *)(splxptr);

  scc_type out(st.dimension() + 1);
  if (st.num_simplices() <= 0)
    return out;

  /* Assigns keys to simplices according their dimension */
  std::vector<int> simplices_per_block_dim(st.dimension() + 1);
  for (auto sh : st.filtration_simplex_range())
    st.assign_key(sh, simplices_per_block_dim[st.dimension(sh)]++);

  std::vector<unsigned int> key_boundary_container;
  for (auto &simplex : st.filtration_simplex_range()) {
    key_boundary_container.clear();
    for (const auto &simplex_id : st.boundary_simplex_range(simplex)) {
      key_boundary_container.push_back(st.key(simplex_id));
    }
    auto &[block_matrix, block_filtrations] = out[st.dimension(simplex)];
    const auto &simplex_filtration = st.filtration(simplex);
    block_matrix.push_back(key_boundary_container);
    block_filtrations.push_back(simplex_filtration);
  }
  return out;
}

template <typename MultiFiltration>
using BoundaryFiltration =
    std::pair<boundary_matrix, std::vector<MultiFiltration>>;

template <class SimplexTree>
BoundaryFiltration<typename SimplexTree::Options::Filtration_value>
st2bf(SimplexTree &st) {

  BoundaryFiltration<typename SimplexTree::Options::Filtration_value> out;
  auto &[matrix, filtrations] = out;
  matrix.reserve(st.num_simplices());
  filtrations.reserve(st.num_simplices());
  if (st.num_simplices() <= 0)
    return out;

  /* Assigns keys to simplices according their dimension */
  int count = 0;
  for (auto sh : st.complex_simplex_range())
    st.assign_key(sh, count++);

  std::vector<unsigned int> key_boundary_container;
  for (auto &simplex : st.complex_simplex_range()) {
    key_boundary_container.clear();
    for (const auto &simplex_id : st.boundary_simplex_range(simplex)) {
      key_boundary_container.push_back(st.key(simplex_id));
    }
    matrix.push_back(key_boundary_container);
    filtrations.push_back(st.filtration(simplex));
  }
  return out;
}

} // namespace Gudhi::multiparameter::mma

#endif // FORMAT_PYTHON_CPP_H_INCLUDED
