R/depth.betaSkeleton.r

Defines functions depth.betaSkeleton .betaSkeleton_depths .betaSkeleton_learn .betaSkeleton_validate

Documented in depth.betaSkeleton

################################################################################
# File:             depth.betaSkeleton.r
# Created by:       Pavlo Mozharovskyi
# First published:  08.03.2018
# Last revised:     08.03.2018
# 
# Computation of the beta-skeleton depth.
################################################################################

.betaSkeleton_validate <- function(ddalpha, beta = 2, distance = "Lp", Lp.p = 2, 
                          mah.estimate = "moment", mah.parMcd = 0.75, ...) {
  # only validate and stop if anything is wrong
  if (!is.numeric(beta) || length(beta) > 1 || beta < 1){
    stop("Argument \"beta\" should be a real value >= 1")
  }
  if (!is.numeric(Lp.p) || length(Lp.p) > 1 || Lp.p < 0){
    stop("Argument \"Lp.p\" should be a real value >= 0")
  }
  if (toupper(distance) == "MAHALANOBIS"){
    if( !(toupper(mah.estimate) %in% c("NONE", "MOMENT", "MCD"))){
        stop("Wrong argument \"mah.estimate\", should be one of \"moment\", \"MCD\", \"none\"")
    }
  } else if (toupper(distance) != "LP") {
      stop("Argument \"distance\" should be either \"Lp\" or \"Mahalanobis\"")
  }
  
  
  return (list(beta = beta, distance = distance, Lp.p = Lp.p, 
              mah.estimate = mah.estimate, mah.parMcd = mah.parMcd))
}

.betaSkeleton_learn <- function(ddalpha) {
  #1. Calculate statistics based on data
  
  if (toupper(ddalpha$distance) == "MAHALANOBIS"){
    ddalpha$distance.code <- 5L
    if(toupper(ddalpha$mah.estimate) == "NONE"){
      for (i in 1:ddalpha$numPatterns){
        ddalpha$patterns[[i]]$sigma <- diag(ddalpha$dimension)
      }
    } else {
      for (i in 1:ddalpha$numPatterns){
        if(toupper(ddalpha$mah.estimate) == "MOMENT"){
          cov <- cov(as.matrix(ddalpha$patterns[[i]]$points))
        } else if(toupper(ddalpha$mah.estimate) == "MCD"){
          cov <- covMcd(as.matrix(ddalpha$patterns[[i]]$points), ddalpha$mah.parMcd)$cov
        } else {stop("Wrong argument \"mah.estimate\", should be one of \"moment\", \"MCD\", \"none\"")}
        if(sum(is.na(cov)) == 0){
          ddalpha$patterns[[i]]$sigma <- solve(cov)
        } else{
          ddalpha$patterns[[i]]$sigma <- diag(ddalpha$dimension)
          warning("Covariance estimate not found for pattern ", ddalpha$patterns[[i]]$name, ", no affine-invariance-adjustment")
        }
      }
    }
  }else{
    for (i in 1:ddalpha$numPatterns){
      ddalpha$patterns[[i]]$sigma <- 0;
    }
    if (toupper(ddalpha$distance) == "LP"){
      ddalpha$distance.code <- 4L
      if (ddalpha$Lp.p == 1){
        ddalpha$distance.code <- 1L
      }
      if (ddalpha$Lp.p == 2){
        ddalpha$distance.code <- 2L
      }
      if (is.infinite(ddalpha$Lp.p) && ddalpha$Lp.p > 0){
        ddalpha$distance.code <- 3L
      }
    } else {stop("Argument \"distance\" should be either \"Lp\" or \"Mahalanobis\"")}
  }
  
  #2. Calculate depths for each pattern
  for (i in 1:ddalpha$numPatterns){
    ddalpha$patterns[[i]]$depths = .betaSkeleton_depths(ddalpha, ddalpha$patterns[[i]]$points)
  }
  
  return (ddalpha)
}

.betaSkeleton_depths <- function(ddalpha, objects){
  depths <- NULL
  for (i in 1:ddalpha$numPatterns){
    pattern <- ddalpha$patterns[[i]]$points
    x <- as.vector(t(pattern))
    y <- as.vector(t(objects))

    ds <- .C("BetaSkeletonDepth", 
             as.double(x), 
             as.double(y), 
             as.integer(nrow(pattern)), 
             as.integer(nrow(objects)), 
             as.integer(ncol(pattern)), 
             as.double(ddalpha$beta), 
             as.integer(ddalpha$distance.code), 
             as.double(ddalpha$Lp.p), 
             as.double(ddalpha$patterns[[i]]$sigma), 
             depths=double(nrow(objects)))$depths
    
    depths <- cbind(depths, ds)
  }
  return (depths)
}

depth.betaSkeleton <- function(x, data, beta = 2, distance = "Lp", Lp.p = 2, 
                               mah.estimate = "moment", mah.parMcd = 0.75){
  if (!is.matrix(x) 
      && is.vector(x)){
    x <- matrix(x, nrow=1)
  }
  if (!(is.matrix(data) && is.numeric(data)
        || is.data.frame(data) && prod(sapply(data, is.numeric))) 
      || ncol(data) < 2){
    stop("Argument \"data\" should be a numeric matrix of at least 2-dimensional data")
  }
  if (!(is.matrix(x) && is.numeric(x)
        || is.data.frame(x) && prod(sapply(x, is.numeric))) ){
    stop("Argument \"x\" should be numeric")
  }
  if (ncol(x) != ncol(data)){
    stop("Dimensions of the arguments \"x\" and \"data\" should coincide")
  }
  if (nrow(data) < 2){
    stop("Too few data points")
  }
  if (!is.numeric(beta) || length(beta) > 1 || beta < 1){
    stop("Argument \"beta\" should be a real value >= 1")
  }
  if (!is.numeric(Lp.p) || length(Lp.p) > 1 || Lp.p < 0){
    stop("Argument \"Lp.p\" should be a real value >= 0")
  }
  if (toupper(distance) == "MAHALANOBIS"){
    distance.code <- 5L
    if(toupper(mah.estimate) == "NONE"){
      sigma = diag(ncol(data))
    } else {
      if(toupper(mah.estimate) == "MOMENT"){
        cov <- cov(data)
      } else if(toupper(mah.estimate) == "MCD"){
        cov <- covMcd(data, mah.parMcd)$cov
      } else {stop("Wrong argument \"mah.estimate\", should be one of \"moment\", \"MCD\", \"none\"")}
      if(sum(is.na(cov)) == 0){
        sigma <- solve(cov)
      } else{
        sigma = diag(ncol(data))
        warning("Covariance estimate not found, no affine-invariance-adjustment")
      }
    }
  }else{
    sigma = 0;
    if (toupper(distance) == "LP"){
      distance.code <- 4L
      if (Lp.p == 1){
        distance.code <- 1L
      }
      if (Lp.p == 2){
        distance.code <- 2L
      }
      if (is.infinite(Lp.p) && Lp.p > 0){
        distance.code <- 3L
      }
    } else {stop("Argument \"distance\" should be either \"Lp\" or \"Mahalanobis\"")}
  }

  points <- as.vector(t(data))
  objects <- as.vector(t(x))
  ds <- .C("BetaSkeletonDepth", 
           as.double(points), 
           as.double(objects), 
           as.integer(nrow(data)), 
           as.integer(nrow(x)), 
           as.integer(ncol(data)), 
           as.double(beta), 
           as.integer(distance.code), 
           as.double(Lp.p), 
           as.double(sigma), 
           depths=double(nrow(x)))$depths
  
  return (ds)
}

Try the ddalpha package in your browser

Any scripts or data that you put into this service are public.

ddalpha documentation built on March 23, 2022, 9:07 a.m.