R/range.R

Defines functions bind_rng lower upper level is_range bt `$.range` `[.range` c.range print.range format.range is.na.range duplicated.range unique.range rep.range as.data.frame.range

Documented in bind_rng bt is_range level lower upper

#' Construct ranges
#'
#' @param lower,upper A numeric vector of values for lower and upper limits.
#' @param level Default `NULL` does not include 'level'. Otherwise values of 
#' length 1 or as length of `lower`, expected between 0 and 100.
#'
#' @return A "range" object
#' @examples
#' bind_rng(lower = rnorm(10), upper = rnorm(10) + 5, level = 95L)
#'
#' @export
bind_rng <- function(lower, upper, level = NULL) {
  if (any(vapply(list(lower, upper), is.null, logical(1)))) {
    stop("no default for `lower` or `upper`.", call. = FALSE)
  }
  if (any(upper < lower, na.rm = TRUE)) {
    stop("'upper' can't be lower than 'lower'.", call. = FALSE)
  }
  len <- length(lower)
  if (is.null(level)) {
    return(as_range(Map(list, lower = lower, upper = upper)))
  } else if (any(level < 0 | level > 100, na.rm = TRUE)) {
    stop("'level' can't be negative or greater than 100.", call. = FALSE)
  } else if (!(length(level) %in% c(1, len))) {
    stop(gettextf("'level' should be of length 1 or %d.", len), call. = FALSE)
  } else {
    as_range(Map(list, lower = lower, upper = upper, level = level))
  }
}

#' Helpers for "range"
#'
#' @param x A "range" object.
#' @rdname helper
#'
#' @export
lower <- function(x) {
  stopifnot(is_range(x))
  x$lower
}

#' @rdname helper
#' @export
upper <- function(x) {
  stopifnot(is_range(x))
  x$upper
}

#' @rdname helper
#' @export
level <- function(x) {
  stopifnot(is_range(x))
  x$level
}

#' @rdname helper
#' @export
is_range <- function(x) {
  inherits(x, "range")
}

#' Validate whether values fall in the ranges
#'
#' @param x A numeric vector of values.
#' @param range A vector of ranges.
#'
#' @details "b/t" is short for "between". To achieve faster performance, one
#' could do `dplyr::between(x, lower(rng), upper(rng))`.
#'
#' @examples
#' rng <- bind_rng(lower = rnorm(10), upper = rnorm(10) + 5)
#' bt(0.2017, rng)
#'
#' @export
bt <- function(x, range) {
  stopifnot(is.numeric(x) || is_range(range))
  x >= range$lower & x <= range$upper
}

#' @export
`$.range` <- function(x, name) {
  fast_unlist(lapply(x, "[[", name))
}

#' @export
`[.range` <- function(x, ..., drop = TRUE) {
  as_range(NextMethod())
}

#' @export
c.range <- function(...) {
  as_range(fast_unlist(lapply(list(...), `[`)))
}

#' @export
print.range <- function(x, ..., digits = NULL) {
  cat(format(x, digits = digits), sep = "\n")
  invisible(x)
}

#' @export
format.range <- function(x, digits = NULL, ...) {
  format(compact_range(x, digits = digits), ...)
}

#' @export
is.na.range <- function(x) {
  # both lower and upper are NA's
  rowSums(is.na(matrix(c(x$lower, x$upper), ncol = 2))) == 2
}

#' @export
duplicated.range <- function(x, incomparables = FALSE, fromLast = FALSE, ...) {
  mat <- matrix(c(x$lower, x$upper, x$level), ncol = 3)
  duplicated(mat, incomparables = incomparables, fromLast = fromLast, ...)
}

#' @export
unique.range <- function(x, incomparables = FALSE, ...) {
  x[!duplicated(x, incomparables = incomparables, ...)]
}

#' @export
rep.range <- function(x, ...) {
  as_range(NextMethod())
}

#' @export
as.data.frame.range <- function(
  x, row.names = NULL, optional = FALSE, ..., 
  nm = paste(deparse(substitute(x), width.cutoff = 500L), collapse = " ")
) {
  as.data.frame.vector(
    x, row.names = row.names, optional = optional, ..., 
    nm = nm
  )
}
earowang/hilo documentation built on May 18, 2019, 8:07 p.m.