blob: ba011625678b997a26e349d2680a2dd42744271d [file] [log] [blame]
# File src/library/graphics/R/image.R
# Part of the R package, https://www.R-project.org
#
# Copyright (C) 1995-2014 The R Core Team
#
# This program is free software; you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation; either version 2 of the License, or
# (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# A copy of the GNU General Public License is available at
# https://www.R-project.org/Licenses/
image <- function(x, ...) UseMethod("image")
image.default <- function (x = seq(0, 1, length.out = nrow(z)),
y = seq(0, 1, length.out = ncol(z)),
z,
zlim = range(z[is.finite(z)]),
xlim = range(x),
ylim = range(y),
col = hcl.colors(12, "YlOrRd", rev = TRUE),
add = FALSE,
xaxs = "i", yaxs = "i", xlab, ylab,
breaks, oldstyle = FALSE,
useRaster, ...)
{
if (missing(z)) {
if (!missing(x)) {
if (is.list(x)) {
z <- x$z; y <- x$y; x <- x$x
} else {
if(is.null(dim(x)))
stop("argument must be matrix-like")
z <- x
x <- seq.int(0, 1, length.out = nrow(z))
}
if (missing(xlab)) xlab <- ""
if (missing(ylab)) ylab <- ""
} else stop("no 'z' matrix specified")
} else if (is.list(x)) {
xn <- deparse(substitute(x))
if (missing(xlab)) xlab <- paste0(xn, "$x")
if (missing(ylab)) ylab <- paste0(xn, "$y")
y <- x$y
x <- x$x
} else {
if (missing(xlab))
xlab <- if (missing(x)) "" else deparse(substitute(x))
if (missing(ylab))
ylab <- if (missing(y)) "" else deparse(substitute(y))
}
if (any(!is.finite(x)) || any(!is.finite(y)))
stop("'x' and 'y' values must be finite and non-missing")
if (any(diff(x) <= 0) || any(diff(y) <= 0))
stop("increasing 'x' and 'y' values expected")
if (!is.matrix(z))
stop("'z' must be a matrix")
if (!typeof(z) %in% c("logical", "integer", "double"))
stop("'z' must be numeric or logical")
if (length(x) > 1 && length(x) == nrow(z)) { # midpoints
dx <- 0.5*diff(x)
x <- c(x[1L] - dx[1L], x[-length(x)] + dx,
x[length(x)] + dx[length(x)-1])
}
if (length(y) > 1 && length(y) == ncol(z)) { # midpoints
dy <- 0.5*diff(y)
y <- c(y[1L] - dy[1L], y[-length(y)] + dy,
y[length(y)] + dy[length(y)-1L])
}
if (missing(breaks)) {
nc <- length(col)
if (!missing(zlim) && (any(!is.finite(zlim)) || diff(zlim) < 0))
stop("invalid z limits")
if (diff(zlim) == 0)
zlim <- if (zlim[1L] == 0) c(-1, 1)
else zlim[1L] + c(-.4, .4)*abs(zlim[1L])
z <- (z - zlim[1L])/diff(zlim)
zi <- if (oldstyle) floor((nc - 1) * z + 0.5)
else floor((nc - 1e-5) * z + 1e-7)
zi[zi < 0 | zi >= nc] <- NA
} else {
if (length(breaks) != length(col) + 1)
stop("must have one more break than colour")
if (any(!is.finite(breaks)))
stop("'breaks' must all be finite")
if (is.unsorted(breaks)) {
warning("unsorted 'breaks' will be sorted before use")
breaks <- sort(breaks)
}
## spatstat passes a factor matrix here, but .bincode converts to double
zi <- .bincode(z, breaks, TRUE, TRUE) - 1L
}
if (!add) # use xlim, ylim here to get dispatch on Axis.
plot(xlim, ylim, xlim = xlim, ylim = ylim, type = "n", xaxs = xaxs,
yaxs = yaxs, xlab = xlab, ylab = ylab, ...)
## need plot set up before we do this
if (length(x) <= 1) x <- par("usr")[1L:2]
if (length(y) <= 1) y <- par("usr")[3:4]
if (length(x) != nrow(z)+1 || length(y) != ncol(z)+1)
stop("dimensions of z are not length(x)(-1) times length(y)(-1)")
check_irregular <- function(x, y)
{
# check that the grid is regular
dx <- diff(x)
dy <- diff(y)
(length(dx) && !isTRUE(all.equal(dx, rep(dx[1], length(dx))))) ||
(length(dy) && !isTRUE(all.equal(dy, rep(dy[1], length(dy)))))
}
if (missing(useRaster)) {
useRaster <- getOption("preferRaster", FALSE)
if (useRaster && check_irregular(x, y)) useRaster <- FALSE
if (useRaster) {
useRaster <- FALSE
ras <- dev.capabilities("rasterImage")$rasterImage
if(identical(ras, "yes")) useRaster <- TRUE
if(identical(ras, "non-missing")) useRaster <- all(!is.na(zi))
}
}
if (useRaster) {
if(check_irregular(x,y))
stop(gettextf("%s can only be used with a regular grid",
sQuote("useRaster = TRUE")),
domain = NA)
# this should be mostly equivalent to RGBpar3 with bg = R_TRANWHITE
if (!is.character(col)) {
col <- as.integer(col)
if (any(!is.na(col) & col < 0L))
stop("integer colors must be non-negative")
col[col < 1L] <- NA_integer_
p <- palette()
col <- p[((col - 1L) %% length(p)) + 1L]
}
zc <- col[zi + 1L]
dim(zc) <- dim(z)
zc <- t(zc)[ncol(zc):1L,, drop = FALSE]
rasterImage(as.raster(zc),
min(x), min(y), max(x), max(y),
interpolate = FALSE)
} else .External.graphics(C_image, x, y, zi, col)
invisible()
}