library(spNNGP)       # Build neighbor index


i_dist <- function(i, neighbor_index, s){
  dist(s[c(i, neighbor_index[[i - 1]]), ])
}	
ho_dist <- function(i, nn.ind, coords) c(dist(coords[nn.ind[i, ], ]))
get_NN_distM <- function (ind, ind_distM_d, M) {
  if (ind < M ){l = ind } else {l = M}; 
  M_i <- rep(0, M * (M - 1) / 2);
  if (l == 1) {}
  else{
    M_i[1: (l * (l - 1) / 2)] <- 
      c(ind_distM_d[[ind]])[(l + 1): (l * (l + 1) / 2)]
  }
  return(M_i)
}

get_NN_dist <- function (ind, ind_distM_d, M) {
  if (ind < M ){l = ind } else {l = M}; 
  D_i <- rep(0, M);
  D_i[1:l] <- c(ind_distM_d[[ind]])[1:l]
  return(D_i)
}

get_NN_ind <- function (ind, ind_distM_i, M) {
  if (ind < M ){l = ind } else {l = M}; 
  D_i <- rep(0, M);
  D_i[1:l] <- c(ind_distM_i[[ind]])[1:l]
  return(D_i)
}


#### NNMatrix: A wrapper of spConjNNGP to build Nearest Neighbor matrics ####
##' coords:        An n x 2 matrix of the observation coordinates in R^2
##' n.neighbors:   Number of neighbors used in the NNGP.  
##' n.omp.threads: A positive integer indicating the number of threads to use 
##'                for SMP parallel processing. 
##' search.type:   a quoted keyword that specifies type of nearest neighbor 
##'                search algorithm. Supported method key words are: "cb" and 
##'                "brute". The "cb" should generally be much faster. If 
##'                locations do not have identical coordinate values on the 
##'                axis used for the nearest neighbor ordering (see ord 
##'                argument) then "cb" and "brute" should produce identical 
##'                neighbor sets. However, if there are identical coordinate 
##'                values on the axis used for nearest neighbor ordering, 
##'                then "cb" and "brute" might produce different, but equally 
##'                valid, neighbor sets, e.g., if data are on a grid.
##' ord:           an index vector of length n used for the nearest neighbor 
##'                search. Internally, this vector is used to order coords, 
##'                i.e., coords[ord,], and associated data. Nearest neighbor 
##'                candidates for the i-th row in the ordered coords are rows 
##'                1:(i-1), with the n.neighbors nearest neighbors being those 
##'                with the minimum euclidean distance to the location defined 
##'                by ordered coords[i,]. The default used when ord is not 
##'                specified is x-axis ordering, i.e., order(coords[,1]). This 
##'                argument should typically be left blank. 
####

NNMatrix <- function(coords, n.neighbors, n.omp.threads = 2,
                     search.type = "cb", ord = order(coords[, 1])){
  
  N <- nrow(coords)
  m.c <- spConjNNGP(rep(0, N) ~ 1, coords = coords,
                    n.neighbors = n.neighbors,
                    theta.alpha = c("phi" = 5, "alpha" = 0.5),
                    sigma.sq.IG = c(2, 1),
                    cov.model = "exponential",
                    n.omp.threads = n.omp.threads,
                    search.type = search.type,
                    ord = ord,
                    return.neighbor.info = T, fit.rep = F,
                    verbose = F)
  M = n.neighbors
  NN_ind <- t(sapply(1: (N - 1), get_NN_ind, m.c$neighbor.info$n.indx[-1], M))
  neighbor_dist <- sapply(2:N, i_dist, m.c$neighbor.info$n.indx[-1], 
                          m.c$coords[m.c$neighbor.info$ord, ])
  NN_distM <- t(sapply(1: (N - 1), get_NN_distM, neighbor_dist, M))
  NN_dist <- t(sapply(1: (N - 1), get_NN_dist, neighbor_dist, M))
  
  return(list(ord = m.c$neighbor.info$ord, 
              coords.ord = m.c$coords[m.c$neighbor.info$ord, ],
              NN_ind = NN_ind, NN_distM = NN_distM, NN_dist = NN_dist))
}

#### Function for checking neighbors ####

Check_Neighbors <- function(coords.ord, n.neighbors, NN.matrix, ind){
  
  plot(coords.ord)
  points(coords.ord[1:ind, , drop = FALSE], col = "grey", pch = 19)
  
  # neighbors
  if (ind < n.neighbors) {dim = ind} else {dim = n.neighbors}
  for (j in 1:dim){
    points(coords.ord[NN.matrix$NN_ind[ind - 1, j], , drop = FALSE],
           col = "orange",  pch = 19)
  }
  points(coords.ord[ind, , drop = FALSE], col = "blue", pch = 19)
  legend("topright", inset=.05, 
         c("obs", paste0( ind, "th obs"), 
           paste0("neighbors of ", ind,"th obs"), 
           paste0("obs indexed <", ind)), pch = c(1, 19, 19, 19),
         col=c("black", "blue", "orange", "grey"), horiz = F)
}

