R/extract.R

Defines functions transform_ind Extract

Documented in Extract

################################################################################

# Transform negative or boolean indices to positive indices
transform_ind <- function(k, lim) {
  
  if (missing(k))
    return(seq_len(lim))
  
  if (is.character(k))
    stop2("Character subsetting is not allowed.")
  
  res <- seq_len(lim)[k]
  
  if (any(is.na(res)))
    stop2("Error when subsetting (missing values? out of bounds?)")
  
  res
}

################################################################################

#' Create an Implementation of [ For Custom Matrix-Like Types
#'
#' `extract` is a function that converts different index types such as negative
#' integer vectors or logical vectors passed to the `[` function as `i`
#' (e.g. \code{X[i]}) or `i` and `j` (e.g. \code{X[i, j]}) into positive
#' integer vectors. The converted indices are provided as the `i` parameter of
#' `extract_vector` or `i` and `j` parameters of `extract_matrix` to facilitate
#' implementing the extraction mechanism for custom matrix-like types.
#'
#' The custom type must implement methods for `dim` for this function
#' to work. Implementing methods for `nrow` and `ncol` is not necessary
#' as the default method of those generics calls `dim` internally.
#'
#' **This idea initially comes from [package crochet](https://goo.gl/3RDNQG).**
#'
#' @param extract_vector A function in the form of `function(x, i)` that takes
#' a subset of `x` based on a single vector of indices `i` and returns a vector.
#' @param extract_matrix A function in the form of `function(x, i, j)` that
#' takes a subset of `x` based on two vectors of indices `i` and `j` and returns
#' a matrix.
#'
#' @return A function in the form of `function(x, i, j, ..., drop = TRUE)` that
#' is meant to be used as a method for `[` for a custom type.
#'
Extract <- function(extract_vector, extract_matrix) {
  
  function(x, i, j, ..., drop = TRUE) {
    
    n <- nrow(x)
    m <- ncol(x)
    
    nargs <- nargs() - !missing(drop)
    
    if (nargs == 2) {  # only i
      
      if (missing(i)) {
        nargs <- 3  # x[] is the same as x[,]
      } else {
        if (is.logical(i))
          stop2("Logical vector subsetting is not allowed")
        if (!isTRUE(all(i > 0)))
          stop2("Only positive vector subsetting is allowed")
        if (!is.matrix(i))
          stop2("Vector subsetting is not allowed")
        
        return(extract_vector(x, i))
      }
      
    }
    
    if (nargs == 3) {
      
      res <- extract_matrix(x, transform_ind(i, n), transform_ind(j, m))
      
      return(`if`(drop, drop(res), res))
      
    }
    
  }
  
}

################################################################################

Try the mmapcharr package in your browser

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

mmapcharr documentation built on May 2, 2019, 2:04 a.m.