if (!require(e1071)) {
  install.packages("e1071")
  library(e1071)
}
if (!require(nortest)) {
  install.packages("nortest")
  library(nortest)
}
if (!require(plot.matrix)) {
  install.packages("plot.matrix")
  library(plot.matrix)
}
if (!require(DescTools)) {
  install.packages("DescTools")
  library(DescTools)
}
if (!require(DescTools)) {
  install.packages("DescTools")
  library(DescTools)
}
if (!require(glasso)) {
  install.packages("glasso")
  library(glasso)
}
if (!require(pracma)) {
  install.packages("pracma")
  library(pracma)
}
if (!require(matrixcalc)) {
  install.packages("matrixcalc")
  library(matrixcalc)
}
if (!require(hglasso)) {
  install.packages("hglasso")
  library(hglasso)
}
if (!require(gganimate)) {
  install.packages("gganimate")
  library(gganimate)
}
if (!require(magick)) {
  install.packages("magick")
  library(magick)
}
if (!require(tidyr)) {
  install.packages("tidyr")
  library(tidyr)
}
if (!require(ggplot2)) {
  install.packages("ggplot2")
  library(ggplot2)
}
if (!require(dplyr)) {
  install.packages("dplyr")
  library(dplyr)
}
if (!require(gtable)) {
  install.packages("gtable")
  library(gtable)
}
if (!require(gridExtra)) {
  install.packages("gridExtra")
  library(gridExtra)
}
if (!require(RColorBrewer)) {
  install.packages("RColorBrewer")
  library(RColorBrewer)
}
if (!require(igraph)) {
  install.packages("igraph")
  library(igraph)
}




##############################################
############# Part 0: Functions ##############
##############################################
## BIC function for GLASSO:
.BICemp <- function(sigmaHat, precMatHat, n, p) {
  .bic <- 0
  .bic <- .bic - n * log(det(precMatHat))
  .bic <- .bic + n * Trace(precMatHat %*% sigmaHat)
  
  .edgenum = 0
  for (.i in 2:p) {
    for (.j in 1:(.i-1)) {
      .edgenum = .edgenum + (abs(precMatHat[.i, .j]) > 10^(-6))
    }
  }
  .bic <- .bic + log(n)*.edgenum
  return(.bic)
}


## Function that calculates one GM with a 
## fixed penalty.
wglasso <- function(cov, rho1, rho2, hubshat, p, n) {
  .start_time = Sys.time()
  ## We define all the variables we need 
  ## for the estimation:
  .inv_cov = solve(cov)
  ## The cycle chooses all possible ordered
  ## pairs of weights:
  .hweight  = rho1 
  .nhweight = rho2
  ## First, generate the weight matrix:
  .W  <- matrix(rep(.nhweight, p * p), ncol = p)
  for (.hub in hubshat) {
    for (.var in (1:p)) {
      .W[.hub, .var] = .hweight
      .W[.var, .hub] = .hweight
    }
  }
  .W = .W - diag(diag(.W))
  
  .model.WGL = glasso(s = cov, rho = .W, 
                      nobs = n, zero = NULL, 
                      thr = 1.0e-4, maxit = 200,  approx = FALSE,
                      penalize.diagonal = TRUE, start = "cold",
                      w.init = NULL, wi.init = NULL, trace = TRUE)
  .BIC.WGL = .BICemp(sigmaHat = cov, 
                     precMatHat = .model.WGL$wi, 
                     n = n, p = p) 
  .end_time = Sys.time()
  .total_time = .end_time - .start_time
  
  
  .OUTPUT = list(model = .model.WGL, 
                 BIC = .BIC.WGL, 
                 penalty = c(rho1, rho2),
                 total.time = .total_time)
  return(.OUTPUT)
}

## FUnction that calculates different GM with 
## varying penalty, and chooses the model with
## lowest BIC.
wglassoBIC <- function(cov, rho, hubshat, p, n) {
  ## We define all the variables we need 
  ## for the estimation:
  .inv_cov = solve(cov)
  .rholength = length(rho)
  .BIC.WGL = NULL
  .model.WGL = list()
  .penalty.WGL = NULL
  .count = 0
  ## The cycle chooses all possible ordered
  ## pairs of weights:
  .start_time = Sys.time()
  for (.i in 1:(.rholength - 1)) {
    for (.j in (.i+1):.rholength) {
      
      .hweight  = rho[.i] 
      .nhweight = rho[.j]
      ## First, generate the weight matrix:
      .W  <- matrix(rep(.nhweight, p * p), ncol = p)
      for (.hub in hubshat) {
        for (.var in (1:p)) {
          .W[.hub, .var] = .hweight
          .W[.var, .hub] = .hweight
        }
      }
      .W = .W - diag(diag(.W))
      
      .count = .count + 1
      .model.WGL[[.count]] = glasso(s = cov, rho = .W, 
                                    nobs = n, zero = NULL, 
                                    thr = 1.0e-4, maxit = 200,  approx = FALSE,
                                    penalize.diagonal = TRUE, start = "cold",
                                    w.init = NULL, wi.init = NULL, trace = TRUE)
      .BIC.WGL = c(.BIC.WGL, .BICemp(sigmaHat = cov, 
                                     precMatHat = .model.WGL[[.count]]$wi, 
                                     n = n, p = p) )
      .penalty.WGL = rbind(.penalty.WGL, c(.hweight, .nhweight))
    }
  }
  .end_time = Sys.time()
  .total_time_WGL = .end_time - .start_time
  
  .OUTPUT = list(models = .model.WGL, 
                 BIC = .BIC.WGL, 
                 optimal.penalty = .penalty.WGL[which.min(.BIC.WGL),],
                 optimal.model = .model.WGL[[which.min(.BIC.WGL)]],
                 total.time = .total_time_WGL)
  return(.OUTPUT)
}

## Given a fixed pair of penalty parameters,
## this function finds a cross-validated
## estimation of the graphical model.
cv.wglasso <- function(X, nfolds = 10, 
                       rho1,
                       rho2,
                       hubshat, 
                       threshold = 10^(-6),
                       robustness = 0.5,
                       p, n,
                       print = TRUE) {
  ## Step 1: divide the data into folds:
  if (print == TRUE) print("Step 1: Creating folds.")
  .folds = sample(1:nfolds, n, replace = TRUE)
  
  ## perform the estimation of the graphical model 
  ## by folds:
  if (print == TRUE) print("Step 2: Estimating graphical models for each fold.")
  .pmfold = array(rep(NA, p*p*nfolds), dim = c(p,p,nfolds))
  
  for (.fold in 1:nfolds) {
    .foldindex = (1:n)[.folds == .fold]
    .foldsize = length(.foldindex)
    .Xfold = X[-.foldindex,]
    .covfold = cov(.Xfold)
    .n = n - .foldsize
    .output.wgl = wglasso(cov = .covfold, 
                          rho1 = rho1,
                          rho2 = rho2, 
                          hubshat = hubshat,
                          p = p, n = .n)
    .pmfold[, , .fold] = (.output.wgl$model)$wi
  }
  
  ## Choose the output of the graphical model
  ## by choosing 
  if (print == TRUE) print("Step 3: Finding robust graphical model estimate.")
  .adj = matrix(rep(0, p * p), ncol = p)
  for (.i in 2:p) {
    for (.j in 1:(.i-1)) {
      .entryvalue = .pmfold[.i, .j, ]
      .edgepresence = (abs(.entryvalue) > threshold)
      .edgerobusness = sum(.edgepresence)/nfolds
      if ( .edgerobusness >= robustness ) {
        .adj[.i, .j] = 1
        .adj[.j, .i] = 1
      }
    }
  }
  if (print == TRUE) print("Step 4: Processing results.")
  .OUTPUT = list(pm.folds = .pmfold,
                 robust.gm = .adj)
  return(.OUTPUT)
}


## Function that returns a robust estimate of
## the graphical model by doing crossvalidation
## on the edges of the model.
cv.wglassoBIC <- function(X, nfolds = 10, 
                          rho, hubshat, 
                          threshold = 10^(-6),
                          robustness = 0.5,
                          p, n,
                          print = TRUE) {
  ## Step 1: divide the data into folds:
  if (print == TRUE) print("Step 1: Creating folds.")
  .folds = sample(1:nfolds, n, replace = TRUE)
  
  ## perform the estimation of the graphical model 
  ## by folds:
  if (print == TRUE) print("Step 2: Estimating graphical models for each fold.")
  .pmfold = array(rep(NA, p*p*nfolds), dim = c(p,p,nfolds))
  
  for (.fold in 1:nfolds) {
    .foldindex = (1:n)[.folds == .fold]
    .foldsize = length(.foldindex)
    .Xfold = X[-.foldindex,]
    .covfold = cov(.Xfold)
    .n = n - .foldsize
    output.wgl = wglassoBIC(cov = .covfold, 
                            rho = rho, 
                            hubshat = hubshat,
                            p = p, n = .n)
    .pmfold[, , .fold] = (output.wgl$optimal.model)$wi
  }
  
  ## Choose the output of the graphical model
  ## by choosing 
  if (print == TRUE) print("Step 3: Finding robust graphical model estimate.")
  .adj = matrix(rep(0, p * p), ncol = p)
  for (.i in 2:p) {
    for (.j in 1:(.i-1)) {
      .entryvalue = .pmfold[.i, .j, ]
      .edgepresence = (abs(.entryvalue) > threshold)
      .edgerobusness = sum(.edgepresence)/nfolds
      if ( .edgerobusness >= robustness ) {
        .adj[.i, .j] = 1
        .adj[.j, .i] = 1
      }
    }
  }
  if (print == TRUE) print("Step 4: Processing results.")
  .OUTPUT = list(pm.folds = .pmfold,
                 robust.gm = .adj)
  return(.OUTPUT)
}



## Given a precision matrix, this function finds
## the adjacency matrix associated with this 
## matrix.
adj = function(pm, threshold = 10^(-6)) {
  return( (abs(pm) > threshold ))
}

## Given a sequence of precision matrices obtained
## by cross-validation, it generates an adjacency
## matrix by selecting only the edges that appear 
## in at least a certain % of the graphs. This 
## % is given by "level".
robust.adj = function(pmfold, p, nfolds, threshold = 10^(-5), level = 0.5) {
  .adj = matrix(rep(0, p * p), ncol = p)
  for (.i in 2:p) {
    for (.j in 1:(.i-1)) {
      .entryvalue = pmfold[.i, .j, ]
      .edgepresence = (abs(.entryvalue) > threshold)
      .edgerobusness = sum(.edgepresence)/nfolds
      if ( .edgerobusness >= level ) {
        .adj[.i, .j] = 1
        .adj[.j, .i] = 1
      }
    }
  }
  return(.adj)
}




