2025-01-12 04:36:52 +08:00

391 lines
16 KiB
Plaintext

% File src/library/grid/vignettes/frame.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}
%\VignetteIndexEntry{Frames and packing grobs}
%\VignettePackage{grid}
\newcommand{\code}[1]{\texttt{#1}}
\newcommand{\pkg}[1]{{\normalfont\fontseries{b}\selectfont #1}}
\newcommand{\grid}{\pkg{grid}}
\newcommand{\R}{{\sffamily R}}
\newcommand{\I}[1]{#1}
\setlength{\parindent}{0in}
\setlength{\parskip}{.1in}
\setlength{\textwidth}{140mm}
\setlength{\oddsidemargin}{10mm}
\title{A GUI-Builder Approach to \grid{} Graphics}
\author{Paul Murrell}
\begin{document}
\maketitle
<<echo=FALSE, results=hide>>=
library(grDevices)
library(stats) # for runif
library(grid)
ps.options(pointsize = 12)
options(width = 60)
@
Grid contains a lot of support for locating, sizing, and arranging
graphical components on a device and with respect to each other.
However, most of this support relies on \emph{either} the parent
object dictating both location and size (layouts) \emph{or} the child
dictating both location and size.
Some sorts of arrangements are more conveniently handled by having the
parent dictate the location, but letting the child dictate the size.
This is the situation for GUI-builders (software which forms
arrangements of GUI components or widgets). The approach taken by
(many ?) GUI-builders is to allow the user to create a parent
\emph{frame} and then \emph{pack} widgets into this frame. The frame
locates and arranges the children with the help of hints such as
``place this widget at the bottom of the frame'', and the children
dictate what size they would like to be.
This document describes a first attempt at such an interface for
arranging Grid graphical objects.
\section*{The \code{"frame"} grob}
You can create a \code{"frame"} graphical object using the function
\code{frameGrob()}. You must assign the result to a variable in order
to pack grobs into it.
<<results=hide>>=
gf <- frameGrob()
@
\section*{The \code{packGrob()} function}
Having created a frame, you can then pack other graphical objects into
it using the \code{packGrob()} function. This function has a complex
interface which allows for a variety of methods of packing graphical
objects. The required arguments are:
\begin{description}
\item[\code{frame}] is a \code{"frame"} object created by \code{grid.frame}.
\item[\code{grob}] is the grob to pack into the frame.
\end{description}
The remaining arguments specify where the grob is located in the frame
and possibly how much space the grob should occupy. The frame is effectively
just a layout; you can add grobs to existing rows and columns
or you can append the grob in a new row and/or column. If the
grob is added to an existing row then the height of
that row becomes the maximum of the new height and
the previous height. If the row is new then it just
gets the specified height. Similar rules apply for column widths.
\begin{description}
\item[\code{col}] is the column to put the grob in. This can be 1 greater than
the existing number of columns (in which case a new column is added).
\item[\code{row}] is like \code{col} but for rows.
\item[\code{col.after}] specifies that the grob should be put in a
new column inserted between \code{col.after} and \code{col.after + 1}.
\item[\code{col.before}] specifies that the grob should be put in a
new column inserted between \code{col.before} and \code{col.before + 1}.
\item[\code{row.after} and \code{row.before}] do what you would expect.
\item[\code{side}] specifies which side to append the new grob to.
The valid values are \code{"left"}, \code{"right"}, \code{"bottom"},
and \code{"top"}.
\item[\code{width}] is the width of the row that the grob is being packed
into. If this is not given then the grob supplies the width.
\item[\code{height}] is like \code{width} but for rows.
\end{description}
It is possible to modify this default behaviour. For example, it is possible
to add a grob to a row and force that row to have the specified height
by setting \code{force.height=TRUE} (and similarly for column widths).
It is also possible to pack a graphical object into several rows or
columns at once (although you cannot simultaneously affect the heights or
widths of those rows and columns).
The result of this function is the modified frame, so you must
assign the result to a variable.
<<results=hide>>=
gf <- packGrob(gf, textGrob("Hello frame!"))
@
\section*{\code{"grobwidth"} and \code{"grobheight"} units}
A \code{"frame"} object allows a grob to specify its size by making
use of \code{"grobwidth"} and \code{"grobheight"} units.
These units may, of course, be used outside of frames too so
their use is described here.
Consider a simple example where I want to draw a rectangle around
a piece of text. I can get the size of the piece of text from the
\code{"text"} grob as follows:
<<frame1, include=FALSE, width=4, height=2, fig=TRUE, results=hide>>=
st <- grid.text("some text")
grid.rect(width = unit(1, "grobwidth", st),
height = unit(1, "grobheight", st))
@
\begin{center}
\includegraphics[height=1in, width=2in]{frame-frame1}
\end{center}
@
You could do the same thing with simple \code{"strwidth"}
and \code{"strheight"} units, but \code{"grobwidth"} and \code{"grobheight"}
give you a lot more power. The biggest gain is that you
can get the size of other objects besides pieces of text (more on that
soon). Another thing you can do is provide a ``reference'' to a grob
rather than the grob itself; you do this by giving the name of a grob.
What this does is make the unit ``dynamic'' so that changes in the
grob affect the unit. The following is a dynamic version of the
previous example.
<<frame1, include=FALSE, width=4, height=2, fig=TRUE, results=hide>>=
grid.text("some text", name = "st")
grid.rect(width = unit(1, "grobwidth", "st"),
height = unit(1, "grobheight", "st"))
@
Now watch what happens if I modify the text grob named \code{"st"}:
<<results=hide, eval=FALSE>>=
grid.edit("st", gp = gpar(fontsize = 20))
<<frame2, echo=FALSE, include=FALSE, width=4, height=2, fig=TRUE, results=hide>>=
my.text <- textGrob("some text")
my.text <- editGrob(my.text, gp = gpar(fontsize = 20))
my.rect <- rectGrob(width = unit(1, "grobwidth", my.text),
height = unit(1, "grobheight", my.text))
grid.draw(my.text)
grid.draw(my.rect)
@
\begin{center}
\includegraphics[height=1in, width=2in]{frame-frame2}
\end{center}
@
Similarly, I can change the text itself:
<<results=hide, eval=FALSE>>=
grid.edit("st", label="some different text")
<<frame3, echo=FALSE, include=FALSE, width=4, height=2, fig=TRUE, results=hide>>=
my.text <- textGrob("some text")
my.text <- editGrob(my.text, gp = gpar(fontsize = 20))
my.text <- editGrob(my.text, label = "some different text")
my.rect <- rectGrob(width = unit(1, "grobwidth", my.text),
height = unit(1, "grobheight", my.text))
grid.draw(my.text)
grid.draw(my.rect)
@
\begin{center}
\includegraphics[height=1in, width=2in]{frame-frame3}
\end{center}
@
\section*{The \code{widthDetails} and \code{heightDetails} generic functions}
The calculation of \code{"grobwidth"} and \code{"grobheight"} units
is a bit complicated, but fortunately most of it is automated.
The simple part is that a grob provides a normal \code{"unit"} object
to express its width or height. The complication comes because that
\code{"unit"} object has to be evaluated in the correct context; in
particular, if the grob has a non-\code{NULL} \code{vp} argument then
those viewports have to be pushed so that the size of the grob is
the size it would be when it is drawn. This is achieved by calling the
\code{preDrawDetails()} function for the grob and in most cases what
happens by default will be correct. The thing to avoid is having
any viewport operations in a \code{drawDetails()} method for your
grob; they should go in a \code{preDrawDetails()} method.
All that needs to be written (usually) are the functions that
provide the \code{"unit"} objects. These functions need to be
\code{widthDetails} and \code{heightDetails} methods.
The default methods return \code{unit(1, "null")}
so your grob will be of this size unless you write your own methods.
The classic example methods are those for \code{"text"} grobs; these
return \code{unit(1, "mystrwidth", <text grob label>)} and
\code{unit(1, "mystrheight", <text grob label>)} respectively.
The other very important examples of these methods are those for
\code{"frame"} grobs. These return the \code{sum} of the
widths (heights) of the columns (rows) of the layout that has been
built up by packing grobs into the frame. This means that
when a \code{"frame"} grob is packed within another \code{"frame"} grob
the parent automatically leaves enough room for the child.
Another useful pair of examples are those for \code{"rect"} grobs.
These methods make use of the \code{absolute.size} function.
When a grob is asked to specify its size, it makes sense to respond
with the \I{grob}'s width and height if the grob has an ``absolute'' size
(e.g., \code{"inches"}, \code{"cm"}, \code{"lines"}, etc;
i.e., the grob knows exactly how big itself is).
On the other hand, it does not make sense to respond with the \I{grob}'s
width and height if the grob has a ``relative'' size (e.g., \code{"npc"} or
\code{"native"}; i.e.,
the grob needs to know about it's parent's size before it can figure
out its own). The \code{absolute.size} function leaves absolute
units alone, but converts relative units to \code{"null"} units
(i.e., the child says to the parent, ``you decide how big I should be''), so
you can return something like \code{absolute.size(width(<grob>))} in order
to always give a sensible answer.
@
\section*{Examples}
The original motivating example for this GUI-builder approach was
to be able to produce a quite general-purpose legend grob.
A legend consists of data symbols and associated textual descriptions.
In order to be quite general, it would be nice to allow, for example,
multiple lines of text per data symbol. Rather than having to look at
the text supplied for the legend in order to determine the arrangement
of the legend, it would be nice to be able to simply specify the
composition of the legend and let it figure out the arrangement for
us. The code below defines just such a legend grob, using the new
\code{"frame"} grob and \code{packGrob()} function. Some points to
note are:
\begin{itemize}
\item The use of \code{border}s to create space around the legend
components.
\item The size of the data symbol component is specified, but the
size of the text components are taken from the \code{"text"} grobs.
\item The heights of the rows in the legend will be the maximum of
$\code{vgap} + \code{unit(1, "lines")}$ and $\code{vgap} +
\code{unit(1, "grobheight", <text grob>)}$.
\item We have two functions, one for generating a grob and one
for producing output.
\end{itemize}
<<>>=
legendGrob <- function(pch, labels, frame = TRUE,
hgap = unit(1, "lines"), vgap = unit(1, "lines"),
default.units = "lines",
vp = NULL) {
nkeys <- length(labels)
gf <- frameGrob(vp = vp)
for (i in 1:nkeys) {
if (i == 1) {
symbol.border <- unit.c(vgap, hgap, vgap, hgap)
text.border <- unit.c(vgap, unit(0, "npc"), vgap,
hgap)
} else {
symbol.border <- unit.c(vgap, hgap, unit(0, "npc"), hgap)
text.border <- unit.c(vgap, unit(0, "npc"), unit(0, "npc"), hgap)
}
gf <- packGrob(gf, pointsGrob(0.5, 0.5, pch = pch[i]),
col = 1, row = i, border = symbol.border,
width = unit(1, "lines"),
height = unit(1, "lines"), force.width = TRUE)
gf <- packGrob(gf, textGrob(labels[i], x = 0, y = 0.5,
just = c("left", "centre")),
col = 2, row = i, border = text.border)
}
gf
}
grid.legend <- function(pch, labels, frame = TRUE,
hgap = unit(1, "lines"), vgap = unit(1, "lines"),
default.units = "lines", draw = TRUE,
vp = NULL) {
gf <- legendGrob(pch, labels, frame, hgap, vgap, default.units, vp)
if (draw) grid.draw(gf)
gf
}
@
The next piece of code shows the \code{grid.legend()} function being
used procedurally; the output is shown below the code.
<<legend, include=FALSE, width=4, height=2, fig=TRUE, results=hide>>=
grid.legend(1:3, c("one line", "two\nlines", "three\nlines\nof text"))
@
\begin{center}
\includegraphics[height=2in, width=4in]{frame-legend}
\end{center}
@
The legend example might not seem too difficult to do by hand rather
than using frames and packing, but the next example shows how
useful it can be.
Suppose you want to arrange a legend next to a plot. This requires
leaving enough space for the legend and then filling the remaining
space with the plot. This requires figuring out how much space the
legend needs, and that is a task that is neither trivial nor
easy to cater for in the general case. Ideally, we want to know
as little as possible about the legend.
With the GUI-builder
approach this becomes extremely simple. The code below
shows how the construction of such a scene might be performed; the
output from the code is again shown below.
The following points are noteworthy:
\begin{itemize}
\item We don't need to know anything about how the legend was
constructed; it could be any sort of grob.
\item We specify the height of the legend to be \code{unit(1, "null")}
so that it will occupy the full height of the plot. If we
did not do this then the plot would be forced to be the height of
the legend (because of the way that \code{"null"} units interact
with other units).
\item The width of the legend is calculated from the contents of the
legend because the legend is a \code{"frame"} grob.
\item The dimensions of the ``plot'' default to \code{unit(1, "null")}
because \code{"collection"} grobs have no width or height methods, which
means that the plot fills up whatever space remains once the
legend has been accommodated.
\end{itemize}
<<plot, echo=FALSE, include=FALSE, fig=TRUE, results=hide>>=
top.vp <- viewport(width = 0.8, height = 0.8)
pushViewport(top.vp)
x <- runif(10)
y1 <- runif(10)
y2 <- runif(10)
pch <- 1:3
labels <- c("Girls", "Boys", "Other")
gf <- frameGrob()
plt <- gTree(children = gList(rectGrob(),
pointsGrob(x, y1, pch = 1),
pointsGrob(x, y2, pch = 2),
xaxisGrob(),
yaxisGrob()))
gf <- packGrob(gf, plt)
gf <- packGrob(gf, legendGrob(pch, labels),
height = unit(1, "null"), side = "right")
grid.rect(gp = gpar(col = "grey"))
grid.draw(gf)
popViewport()
grid.rect(gp = gpar(lty = "dashed"), width = .99, height = .99)
@
\begin{center}
\includegraphics{frame-plot}
\end{center}
@
\section*{Notes}
\begin{enumerate}
\item There are \code{grid.frame()} and \code{grid.pack()} equivalents
for these functions, but these are really only useful to see the
changes in the frame as each packing operation takes place.
\item This frame-and-packing stuff is easier to use, \emph{but}
(in almost all cases) it is less efficient than specifying the
arrangement by hand. There is consequently a penalty to pay in
terms of memory (inconsequential I think) and in terms of speed
(noticeably slower).
Perhaps one sensible use of these functions is to
build an image interactively using the simple arguments,
which will be slow, then attempt to speed up the drawing
by exploring some of the more advanced arguments.
One way to speed things up a bit is to specify the layout when
the frame is initially created and then use the \code{placeGrob()} function
to put grobs into existing rows and columns.
The speed penalty in the cases I have seen are mostly due to the time
taken to generate the (sometimes very) complicated unit objects that
express the heights and widths of the rows and columns of the frame
layout. Future effort will be put into speeding up the creation of
unit objects.
\end{enumerate}
\end{document}