% File src/library/grid/vignettes/displaylist.Rnw % Part of the R package, https://www.R-project.org % Copyright 2001-13 Paul Murrell and the R Core Team % Distributed under GPL 2 or later \documentclass[a4paper]{article} \usepackage{Rd} % \VignetteIndexEntry{Display Lists in grid} % \VignettePackage{grid} % \VignetteDepends{grDevices, graphics, stats} \newcommand{\grid}{\pkg{grid}} \newcommand{\gridBase}{\CRANpkg{gridBase}} \newcommand{\lattice}{\CRANpkg{lattice}} \setlength{\parindent}{0in} \setlength{\parskip}{.1in} \setlength{\textwidth}{140mm} \setlength{\oddsidemargin}{10mm} \title{Display Lists in \grid{}} \author{Paul Murrell} \begin{document} \maketitle <<echo=FALSE, results=hide>>= library(grDevices) library(graphics) # for plot() library(stats) # for runif() library(grid) ps.options(pointsize = 12) options(width = 60) @ A display list is a record of drawing operations. It is used to redraw graphics output when a graphics window is resized, when graphics output is copied from one device to another, and when graphics output is edited (via \code{grid.edit}). There are two display lists that can be used when working with \grid{}. \R{}'s graphics engine maintains a display list and \grid{} maintains its own display list. The former is maintained at the C code level and records both base graphics output and \grid{} graphics output. The latter is maintained at the R code level and only records \grid{} output. In standard usage, the graphics engine's display list is used to redraw when a window is resized and when copying between devices. \grid{}'s display list is used for redrawing when editing \grid{} output. There are two main problems with this standard usage: \begin{enumerate} \item The graphics engine display list only records graphics output; none of the calculations leading up to producing the output are recorded. This particularly impacts on plots which perform calculations based on the physical dimensions of the device -- an example is the legend function which performs calculations in order to arrange the elements of the legend. The effect can be seen from any example which uses the legend function. Try running \code{example(legend)} then resize the device (make it quite tall and thin or quite wide and fat); the legend will start to look pretty sick. {\bf NOTE:} that this is a problem with the graphics engine display list -- it is not specific to \grid{}. In fact, much of \grid{}'s behaviour is protected from this problem because things like \grid{} units are ``declarative'' and will be re-evaluated on each redraw. However, there are situations where \grid{} output can be afflicted, in particular, whenever the \code{convertUnit()} function (or one of its variants) is used (the help file for \code{convertUnit()} gives an example). A situation where this problem becomes very relevant for \grid{} output is when the \gridBase{} package is used. This is a situation where lots of calculations are performed in order to align base and grid output, but these calculations are not recorded on the graphics engine display list, so if the device is resized the output will become very yukky. \item \grid{}'s display list does not record base graphics output\footnote{This is not quite true; it is possible to include base graphics output on the \grid{} display list as we will see later.} so if both base and \grid{} output appear on the same device then the result of editing will not redraw the base output. The following code provides a simple example: <<guts1, eval=FALSE>>= plot(1:10) par(new = TRUE) grid.rect(width = 0.5, height = 0.5, gp = gpar(lwd = 3), name = "gr") <<ex1, echo=FALSE, fig=TRUE, width=4, height=3, include=FALSE>>= <<guts1>> grid.rect(width = 0.99, height = 0.99, gp = gpar(lty = "dashed")) @ \includegraphics[width=4in, height=3in]{displaylist-ex1} <<ex2, echo=FALSE, fig=TRUE, width=4, height=3, include=FALSE>>= grid.rect(width = 0.5, height = 0.5, gp = gpar(col = "red", lwd = 3)) grid.rect(width = 0.99, height = 0.99, gp = gpar(lty = "dashed")) <<eval=FALSE>>= grid.edit("gr", gp = gpar(col = "red", lwd = 3)) @ \includegraphics[width=4in, height=3in]{displaylist-ex2} After the \code{grid.edit}, the rectangle has been redrawn, but the base plot has not. \end{enumerate} \section*{Saving calculations on the graphics engine display list\\ and saving base graphics on the \grid{} display list} Both of the problems described in the previous section can be avoided by using a \code{drawDetails()} method in \grid{}. When a \grid{} grob is drawn, the \code{drawDetails} method for that grob is called; if calculations are put within a \code{drawDetails} method, then the calculations will be performed every time the grob is drawn. This means that it is possible, for example, to use \code{convertUnit()} and have the result consistent across device resizes or copies\footnote{In each of the examples that follow, you should execute the example code, resize the device to see any inconsistency, then close the device before trying the next example.}. This next piece of code is an example where the output becomes inconsistent when the device is resized. We specify a width for the rectangle in inches, but convert it (gratuitously) to \abbr{NPC} coordinates -- when the device is resized, the \abbr{NPC} coordinates will no longer correspond to 1''. <<results=hide>>= grid.rect(width = convertWidth(unit(1, "inches"), "npc")) @ The next piece of code demonstrates that, if we place the calculations within a \code{drawDetails} method, then the output remains consistent across device resizes and copies. <<results=hide>>= drawDetails.myrect <- function(x, recording) { gr <- rectGrob(width = convertWidth(unit(1, "inches"), "npc")) grid.draw(gr) } grid.draw(grob(cl = "myrect")) @ The next example shows that a \code{drawDetails()} method can also be used to save base graphics output on the \grid{} display list. This example uses \gridBase{} to combine base and \grid{} graphics output. Here I replicate the last example from the \gridBase{} vignette -- a set of base pie charts within \grid{} viewports within a base plot. In this case, I can produce all of the grobs required in the normal manner -- their locations and sizes are not based on special calculations\footnote{The example is wrapped inside a check for whether the \CRANpkg{gridBase} package is installed so that the code will still ``run'' on systems without \CRANpkg{gridBase}.}. <<results=hide>>= x <- c(0.88, 1.00, 0.67, 0.34) y <- c(0.87, 0.43, 0.04, 0.94) z <- matrix(runif(4*2), ncol = 2) maxpiesize <- unit(1, "inches") totals <- apply(z, 1, sum) sizemult <- totals/max(totals) gs <- segmentsGrob(x0 = unit(c(rep(0, 4), x), rep(c("npc", "native"), each = 4)), x1 = unit(c(x, x), rep("native", 8)), y0 = unit(c(y, rep(0, 4)), rep(c("native", "npc"), each = 4)), y1 = unit(c(y, y), rep("native", 8)), gp = gpar(lty = "dashed", col = "grey")) gr <- rectGrob(gp = gpar(col = "grey", fill = "white", lty = "dashed")) @ What is important is that I place the calls to the \gridBase{} functions within the \code{drawDetails} method so that they are performed every time the grob is drawn {\em and} the calls to the base graphics functions are in here too so that they are called for every redraw. <<results=hide>>= drawDetails.pieplot <- function(x, recording) { plot(x$x, x$y, xlim = c(-0.2, 1.2), ylim = c(-0.2, 1.2), type = "n") vps <- baseViewports() pushViewport(vps$inner, vps$figure, vps$plot, recording = FALSE) grid.draw(x$gs, recording = FALSE) for (i in 1:4) { pushViewport(viewport(x = unit(x$x[i], "native"), y = unit(x$y[i], "native"), width = x$sizemult[i]*x$maxpiesize, height = x$sizemult[i]*x$maxpiesize), recording = FALSE) grid.draw(x$gr, recording = FALSE) par(plt = gridPLT(), new = TRUE) pie(x$z[i, ], radius = 1, labels = rep("", 2)) popViewport(recording = FALSE) } popViewport(3, recording = FALSE) } @ The ``pie plot'' is created by assembling the component grobs into a collective grob of the appropriate class; the \code{drawDetails} method takes care of actually producing the output. % produce figure, but don't include % (to avoid par setting contamination between code segments) <<results=hide, fig=TRUE, width=6, height=6, include=FALSE>>= if (suppressWarnings(require("gridBase", quietly = TRUE))) { grid.draw(grob(x = x, y = y, z = z, maxpiesize = maxpiesize, sizemult = sizemult, gs = gs, gr = gr, cl = "pieplot")) } @ The output from this example can be resized safely; \grid{} handles all of the redrawing, and performs all of the actions within the \code{drawDetails} method for each redraw, including redrawing the base graphics output! As a final example, we will harness the \grid{} display list purely to achieve consistency in base graphics output. The following reproduces the last example from the \code{legend()} help page, but produces output which can be resized without the legend going crazy. % produce figure, but don't include % (to avoid par setting contamination between code segments) <<results=hide, fig=TRUE, include=FALSE>>= drawDetails.mylegend <- function(x, recording) { x <- 0:64/64 y <- sin(3*pi*x) plot(x, y, type = "l", col = "blue", main = "points with bg & legend(*, pt.bg)") points(x, y, pch = 21, bg = "white") legend(.4,1, "sin(c x)", pch = 21, pt.bg = "white", lty = 1, col = "blue") } grid.draw(grob(cl = "mylegend")) @ \end{document}