# 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 <- deparse1(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 deparse1(substitute(x)) if (missing(ylab)) ylab <- if (missing(y)) "" else deparse1(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() }