R/bicop_plot.R

Defines functions contour.bicop_dist plot.bicop_dist

Documented in contour.bicop_dist plot.bicop_dist

#' Plotting tools for `bicop_dist` and `bicop` objects
#'
#' There are several options for plotting bicop_dist objects. The density of a
#' bivariate copula density can be visualized as surface/perspective or contour
#' plot. Optionally, the density can be coupled with standard normal margins
#' (default for contour plots).
#'
#' @method plot bicop_dist
#'
#' @aliases plot.bicop_dist contour.bicop_dist plot.bicop contour.bicop
#' @param x \code{bicop_dist object.}
#' @param type plot type; either \code{"surface"} or \code{"contour"}.
#' @param margins options are: \code{"unif"} for the original copula density,
#'   \code{"norm"} for the transformed density with standard normal margins,
#'   \code{"exp"} with standard exponential margins, and  \code{"flexp"} with
#'   flipped exponential margins. Default is \code{"norm"} for \code{type =
#'   "contour"}, and \code{"unif"} for \code{type = "surface"}.
#' @param size integer; the plot is based on values on a `size x size` grid,
#'   default is 100.
#' @param \dots optional arguments passed to \code{\link{contour}} or
#'   \code{\link{wireframe}}.
#' @seealso [bicop_dist()], [graphics::contour()], [wireframe()]
#' @keywords plot
#' @examples
#'
#' ## construct bicop_dist object for a student t copula
#' obj <- bicop_dist(family = "t", rotation = 0, parameters = c(0.7, 4))
#'
#' ## plots
#' plot(obj) # surface plot of copula density
#' contour(obj) # contour plot with standard normal margins
#' contour(obj, margins = "unif") # contour plot of copula density
#' @importFrom graphics contour plot
#' @importFrom grDevices rgb col2rgb colorRampPalette
#' @importFrom lattice wireframe
#' @importFrom stats pnorm qnorm dnorm pexp qexp dexp
#' @importFrom utils modifyList
#' @export
plot.bicop_dist <- function(x, type = "surface", margins, size, ...) {
  ## partial matching and sanity check for type
  allowed_types <- c("contour", "surface")
  type <- allowed_types[pmatch(type, allowed_types)]
  assert_that(in_set(type, allowed_types))

  ## choose margins if missing, else partial matching and sanity check
  if (missing(margins)) {
    margins <- switch(type, "contour" = "norm", "surface" = "unif")
  } else {
    assert_that(is.string(margins))
    allowed_margins <- c("norm", "unif", "exp", "flexp")
    margins <- allowed_margins[pmatch(margins, allowed_margins)]
    assert_that(in_set(margins, allowed_margins))
  }

  ## choose size if missing and sanity check
  if (missing(size)) {
    size <- switch(type, "contour" = 100L, "surface" = 40L)
  }
  assert_that(is.number(size))
  size <- round(size)
  if (size < 5) {
    warning("size too small, set to 5")
    size <- 5
  }

  if (is.null(list(...)$xlim) & is.null(list(...)$ylim)) {
    xylim <- switch(margins,
      "unif" = c(1e-2, 1 - 1e-2),
      "norm" = c(-3, 3),
      "exp" = c(0, 6),
      "flexp" = c(-6, 0)
    )
  } else {
    xylim <- range(c(list(...)$xlim, list(...)$ylim))
  }

  ## prepare for plotting with selected margins
  if (margins == "unif") {
    points <- switch(type,
      "contour" = seq(1e-5, 1 - 1e-5, length.out = size),
      "surface" = 1:size / (size + 1)
    )
    g <- as.matrix(expand.grid(points, points))
    points <- g[1L:size, 1L]
    adj <- 1
    gu <- g[, 1L]
    gv <- g[, 2L]
    levels <- c(0.2, 0.6, 1, 1.5, 2, 3, 5, 10, 20)
    xlim <- ylim <- xylim
    at <- c(seq(0, 3, length.out = 50), seq(5, 100, length.out = 50))
  } else if (margins == "norm") {
    points <- pnorm(seq(xylim[1L], xylim[2L], length.out = size))
    g <- as.matrix(expand.grid(points, points))
    points <- qnorm(g[1L:size, 1L])
    adj <- tcrossprod(dnorm(points))
    levels <- c(0.01, 0.025, 0.05, 0.1, 0.15, 0.2, 0.3, 0.4, 0.5)
    gu <- qnorm(g[, 1L])
    gv <- qnorm(g[, 2L])
    xlim <- ylim <- xylim
    # at <- c(seq(0, 0.3, length.out = 50), seq(0.3, 100, length.out = 50))
  } else if (margins == "exp") {
    ll <- ifelse(type == "contour", 1e-2, 1e-1)
    points <- pexp(seq(ll, xylim[2L], length.out = size))
    g <- as.matrix(expand.grid(points, points))
    points <- qexp(g[1L:size, 1L])
    adj <- tcrossprod(dexp(points))
    levels <- c(0.005, 0.01, 0.025, 0.05, 0.1, 0.2, 0.3, 0.5)
    gu <- qexp(g[, 1L])
    gv <- qexp(g[, 2L])
    xlim <- ylim <- xylim
    # at <- c(0, exp(seq(-10, 0, length.out = 79)), seq(1, 10, length.out = 20))
  } else if (margins == "flexp") {
    ll <- ifelse(type == "contour", 1e-2, 1e-1)
    points <- pexp(-seq(xylim[1L], -ll, length.out = size))
    g <- as.matrix(expand.grid(points, points))
    points <- -qexp(g[1L:size, 1L])
    adj <- tcrossprod(dexp(qexp(g[1L:size, 1L])))
    levels <- c(0.005, 0.01, 0.025, 0.05, 0.1, 0.2, 0.3, 0.5)
    gu <- -qexp(g[, 1L])
    gv <- -qexp(g[, 2L])
    g <- 1 - g
    xlim <- ylim <- xylim
    # at <- c(0, exp(seq(-10, 0, length.out = 79)), seq(1, 10, length.out = 20))
  }

  ## evaluate on grid
  vt <- x$var_types
  x$var_types <- c("c", "c")
  vals <- dbicop(g, x)
  x$var_types <- vt
  cop <- matrix(vals, size, size)

  ## adjust for margins
  dens <- cop * adj
  if (length(unique(c(dens))) == 1) {
    dens[1] <- 1.000001 * dens[1]
  }

  if (margins %in% c("exp", "flexp")) {
    dens <- pmin(dens, 6)
  }

  jet.colors <- colorRampPalette(c(
    "#00007F", "blue", "#007FFF", "cyan",
    "#7FFF7F", "yellow", "#FF7F00", "red",
    "#7F0000"
  ),
  bias = 2
  )
  ## actual plotting
  if (type == "contour") {
    # set default parameters
    pars <- list(
      x = points,
      y = points,
      z = dens,
      levels = levels,
      xlim = xlim,
      ylim = ylim,
      xlab = switch(margins,
        "unif" = expression(u[1]),
        "norm" = expression(z[1]),
        "exp" = expression(e[1]),
        "flexp" = expression(e[1])
      ),
      ylab = switch(margins,
        "unif" = expression(u[2]),
        "norm" = expression(z[2]),
        "exp" = expression(e[2]),
        "flexp" = expression(e[2])
      )
    )
  } else if (type == "surface") {
    # list with coordinates
    lst <- list(u = gu, v = gv, c = as.vector(dens))

    # set default parameters
    pars <- list(
      x = c ~ u * v,
      data = lst,
      scales = list(arrows = FALSE),
      drape = TRUE, colorkey = FALSE,
      screen = list(z = 25, x = -55),
      shade = FALSE,
      aspect = c(1, 1),
      light.source = c(10, 0, 10),
      zoom = 0.85,
      par.settings = list(axis.line = list(col = "transparent")),
      col = NA,
      col.regions = jet.colors(100),
      xlab = switch(margins,
        "unif" = expression(u[1]),
        "norm" = expression(z[1]),
        "exp" = expression(e[1]),
        "flexp" = expression(e[1])
      ),
      ylab = switch(margins,
        "unif" = expression(u[2]),
        "norm" = expression(z[2]),
        "exp" = expression(e[2]),
        "flexp" = expression(e[2])
      ),
      zlab = "",
      zlim = switch(margins,
        "unif" = c(0, max(3, 1.1 * max(lst$c))),
        "norm" = c(0, max(0.4, 1.1 * max(lst$c))),
        "exp" = c(0, max(1, 1.1 * max(lst$c))),
        "flexp" = c(0, max(1, 1.1 * max(lst$c)))
      )
    )
  }

  pars <- modifyList(pars, list(...))
  plot_obj <- switch(type,
    contour = do.call(contour, pars),
    surface = print(do.call(wireframe, pars))
  )
  invisible(plot_obj)
}

#' @rdname plot.bicop_dist
#' @export
plot.bicop <- plot.bicop_dist

#' @method contour bicop_dist
#' @rdname plot.bicop_dist
#' @export
contour.bicop_dist <- function(x, margins = "norm", size = 100L, ...) {
  plot(x, type = "contour", margins = margins, size = size, ...)
}

#' @rdname plot.bicop_dist
#' @export
contour.bicop <- contour.bicop_dist

Try the rvinecopulib package in your browser

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

rvinecopulib documentation built on March 7, 2023, 6:20 p.m.