//
// Created by 76538 on 2/17/2021.
//
#include "nanovoid.h"

using namespace std;

//HashBucket::HashBucket() {
//}

//HashBucket::~HashBucket() {
//}

void PNBucket::merge_with(PNBucket* y, EnumHashTable& n_list_hash, UnionFindDelete &inv) {
  uint root_a = inv.find_(inv.item2pd[this->p_list]);
  uint root_b = inv.find_(inv.item2pd[y->p_list]);
  uint new_root = inv.union_(root_a, root_b);
  //inv.check_dfslist_one_root(new_root);
  //printf("merge with root_a=%u, root_b=%u, new_root=%u\n", root_a, root_b, new_root);
  this->p_list = inv.d_item(new_root);

  uint b_it = n_list_hash.head(this->n_list_id);
  uint this_b_it = 0;
  uint root_it = 0;
  while (b_it != UINT_NULL) {
    this_b_it = b_it;
    b_it = n_list_hash.bit2right(b_it);
    root_it = inv.find_(inv.item2pd[n_list_hash.bit2c(this_b_it)]);
    if (root_it == new_root)
      n_list_hash.delete_(this_b_it);
  }
  b_it = n_list_hash.head(y->n_list_id);
  while (b_it != UINT_NULL) {
    root_it = inv.find_(inv.item2pd[n_list_hash.bit2c(b_it)]);
    if (root_it != new_root)
      n_list_hash.insert_no_duplicate(this->n_list_id, n_list_hash.bit2c(b_it));
    b_it = n_list_hash.bit2right(b_it);
  }
  
  /*
  (this->n_list).insert((y->n_list).begin(), (y->n_list).end());
  uint root_it = 0;
  for (set<uint>::iterator it = (this->n_list).begin();
       it != (this->n_list).end(); ) {
    root_it = inv.find_(inv.item2pd[*it]);
    //printf("after find(%u)\n", inv.item2pd[*it]);
    //inv.check_dfslist_one_root(new_root);
    if (root_it == new_root){
      it = (this->n_list).erase(it);
    }else
      ++ it;
   }
  */

  /*
  uint root_it = 0;
  uint cur_size = (this->n_list).size();
  uint ip = 0;
  /*for (vector<uint>::iterator it = (this->n_list).begin();
       it != (this->n_list).end() && ip < cur_size; ) {
    root_it = inv.find_(inv.item2pd[*it]);
    //printf("after find(%u)\n", inv.item2pd[*it]);
    //inv.check_dfslist_one_root(new_root);
    if (root_it == new_root){
      it = (this->n_list).erase(it);
    }else
      ++ it;
   }
  // xyx: A more efficient implementation of the code above.
  for ( ; ip < cur_size; ) {
    root_it = inv.find_(inv.item2pd[(this->n_list)[ip]]);
    if (root_it == new_root) {
      --cur_size;
      // swap (this->n_list)[ip] and (this->n_list)[cur_size]
      uint tmp = (this->n_list)[ip];
      (this->n_list)[ip] = (this->n_list)[cur_size];
      (this->n_list)[cur_size] = tmp;
    }else{
      ++ ip;
    }
  }
  (this->n_list).resize(cur_size);
  
  for (vector<uint>::iterator it = (y->n_list).begin();
       it != (y->n_list).end(); ++ it) {
    root_it = inv.find_(inv.item2pd[*it]);
    //printf("after find(%u)\n", inv.item2pd[*it]);
    //inv.check_dfslist_one_root(new_root);
    if (root_it != new_root){  // not in plist
      (this->n_list).push_back(*it);
    }
  }

  // unique this->n_list
  sort((this->n_list).begin(), (this->n_list).end());
  vector<uint>::iterator last = unique((this->n_list).begin(), (this->n_list).end());
  (this->n_list).resize(distance((this->n_list).begin(), last));
  */
}
