1816 lines
1.5 MiB
HTML
1816 lines
1.5 MiB
HTML
|
<!DOCTYPE html>
|
|||
|
|
|||
|
<html>
|
|||
|
|
|||
|
<head>
|
|||
|
|
|||
|
<meta charset="utf-8" />
|
|||
|
<meta name="generator" content="pandoc" />
|
|||
|
<meta http-equiv="X-UA-Compatible" content="IE=EDGE" />
|
|||
|
|
|||
|
<meta name="viewport" content="width=device-width, initial-scale=1" />
|
|||
|
|
|||
|
|
|||
|
|
|||
|
<title>Extending ggplot2</title>
|
|||
|
|
|||
|
<script>// Pandoc 2.9 adds attributes on both header and div. We remove the former (to
|
|||
|
// be compatible with the behavior of Pandoc < 2.8).
|
|||
|
document.addEventListener('DOMContentLoaded', function(e) {
|
|||
|
var hs = document.querySelectorAll("div.section[class*='level'] > :first-child");
|
|||
|
var i, h, a;
|
|||
|
for (i = 0; i < hs.length; i++) {
|
|||
|
h = hs[i];
|
|||
|
if (!/^h[1-6]$/i.test(h.tagName)) continue; // it should be a header h1-h6
|
|||
|
a = h.attributes;
|
|||
|
while (a.length > 0) h.removeAttribute(a[0].name);
|
|||
|
}
|
|||
|
});
|
|||
|
</script>
|
|||
|
|
|||
|
<style type="text/css">
|
|||
|
code{white-space: pre-wrap;}
|
|||
|
span.smallcaps{font-variant: small-caps;}
|
|||
|
span.underline{text-decoration: underline;}
|
|||
|
div.column{display: inline-block; vertical-align: top; width: 50%;}
|
|||
|
div.hanging-indent{margin-left: 1.5em; text-indent: -1.5em;}
|
|||
|
ul.task-list{list-style: none;}
|
|||
|
</style>
|
|||
|
|
|||
|
|
|||
|
|
|||
|
<style type="text/css">
|
|||
|
code {
|
|||
|
white-space: pre;
|
|||
|
}
|
|||
|
.sourceCode {
|
|||
|
overflow: visible;
|
|||
|
}
|
|||
|
</style>
|
|||
|
<style type="text/css" data-origin="pandoc">
|
|||
|
pre > code.sourceCode { white-space: pre; position: relative; }
|
|||
|
pre > code.sourceCode > span { display: inline-block; line-height: 1.25; }
|
|||
|
pre > code.sourceCode > span:empty { height: 1.2em; }
|
|||
|
.sourceCode { overflow: visible; }
|
|||
|
code.sourceCode > span { color: inherit; text-decoration: inherit; }
|
|||
|
div.sourceCode { margin: 1em 0; }
|
|||
|
pre.sourceCode { margin: 0; }
|
|||
|
@media screen {
|
|||
|
div.sourceCode { overflow: auto; }
|
|||
|
}
|
|||
|
@media print {
|
|||
|
pre > code.sourceCode { white-space: pre-wrap; }
|
|||
|
pre > code.sourceCode > span { text-indent: -5em; padding-left: 5em; }
|
|||
|
}
|
|||
|
pre.numberSource code
|
|||
|
{ counter-reset: source-line 0; }
|
|||
|
pre.numberSource code > span
|
|||
|
{ position: relative; left: -4em; counter-increment: source-line; }
|
|||
|
pre.numberSource code > span > a:first-child::before
|
|||
|
{ content: counter(source-line);
|
|||
|
position: relative; left: -1em; text-align: right; vertical-align: baseline;
|
|||
|
border: none; display: inline-block;
|
|||
|
-webkit-touch-callout: none; -webkit-user-select: none;
|
|||
|
-khtml-user-select: none; -moz-user-select: none;
|
|||
|
-ms-user-select: none; user-select: none;
|
|||
|
padding: 0 4px; width: 4em;
|
|||
|
color: #aaaaaa;
|
|||
|
}
|
|||
|
pre.numberSource { margin-left: 3em; border-left: 1px solid #aaaaaa; padding-left: 4px; }
|
|||
|
div.sourceCode
|
|||
|
{ }
|
|||
|
@media screen {
|
|||
|
pre > code.sourceCode > span > a:first-child::before { text-decoration: underline; }
|
|||
|
}
|
|||
|
code span.al { color: #ff0000; font-weight: bold; }
|
|||
|
code span.an { color: #60a0b0; font-weight: bold; font-style: italic; }
|
|||
|
code span.at { color: #7d9029; }
|
|||
|
code span.bn { color: #40a070; }
|
|||
|
code span.bu { color: #008000; }
|
|||
|
code span.cf { color: #007020; font-weight: bold; }
|
|||
|
code span.ch { color: #4070a0; }
|
|||
|
code span.cn { color: #880000; }
|
|||
|
code span.co { color: #60a0b0; font-style: italic; }
|
|||
|
code span.cv { color: #60a0b0; font-weight: bold; font-style: italic; }
|
|||
|
code span.do { color: #ba2121; font-style: italic; }
|
|||
|
code span.dt { color: #902000; }
|
|||
|
code span.dv { color: #40a070; }
|
|||
|
code span.er { color: #ff0000; font-weight: bold; }
|
|||
|
code span.ex { }
|
|||
|
code span.fl { color: #40a070; }
|
|||
|
code span.fu { color: #06287e; }
|
|||
|
code span.im { color: #008000; font-weight: bold; }
|
|||
|
code span.in { color: #60a0b0; font-weight: bold; font-style: italic; }
|
|||
|
code span.kw { color: #007020; font-weight: bold; }
|
|||
|
code span.op { color: #666666; }
|
|||
|
code span.ot { color: #007020; }
|
|||
|
code span.pp { color: #bc7a00; }
|
|||
|
code span.sc { color: #4070a0; }
|
|||
|
code span.ss { color: #bb6688; }
|
|||
|
code span.st { color: #4070a0; }
|
|||
|
code span.va { color: #19177c; }
|
|||
|
code span.vs { color: #4070a0; }
|
|||
|
code span.wa { color: #60a0b0; font-weight: bold; font-style: italic; }
|
|||
|
</style>
|
|||
|
<script>
|
|||
|
// apply pandoc div.sourceCode style to pre.sourceCode instead
|
|||
|
(function() {
|
|||
|
var sheets = document.styleSheets;
|
|||
|
for (var i = 0; i < sheets.length; i++) {
|
|||
|
if (sheets[i].ownerNode.dataset["origin"] !== "pandoc") continue;
|
|||
|
try { var rules = sheets[i].cssRules; } catch (e) { continue; }
|
|||
|
var j = 0;
|
|||
|
while (j < rules.length) {
|
|||
|
var rule = rules[j];
|
|||
|
// check if there is a div.sourceCode rule
|
|||
|
if (rule.type !== rule.STYLE_RULE || rule.selectorText !== "div.sourceCode") {
|
|||
|
j++;
|
|||
|
continue;
|
|||
|
}
|
|||
|
var style = rule.style.cssText;
|
|||
|
// check if color or background-color is set
|
|||
|
if (rule.style.color === '' && rule.style.backgroundColor === '') {
|
|||
|
j++;
|
|||
|
continue;
|
|||
|
}
|
|||
|
// replace div.sourceCode by a pre.sourceCode rule
|
|||
|
sheets[i].deleteRule(j);
|
|||
|
sheets[i].insertRule('pre.sourceCode{' + style + '}', j);
|
|||
|
}
|
|||
|
}
|
|||
|
})();
|
|||
|
</script>
|
|||
|
|
|||
|
|
|||
|
|
|||
|
|
|||
|
<style type="text/css">body {
|
|||
|
background-color: #fff;
|
|||
|
margin: 1em auto;
|
|||
|
max-width: 700px;
|
|||
|
overflow: visible;
|
|||
|
padding-left: 2em;
|
|||
|
padding-right: 2em;
|
|||
|
font-family: "Open Sans", "Helvetica Neue", Helvetica, Arial, sans-serif;
|
|||
|
font-size: 14px;
|
|||
|
line-height: 1.35;
|
|||
|
}
|
|||
|
#TOC {
|
|||
|
clear: both;
|
|||
|
margin: 0 0 10px 10px;
|
|||
|
padding: 4px;
|
|||
|
width: 400px;
|
|||
|
border: 1px solid #CCCCCC;
|
|||
|
border-radius: 5px;
|
|||
|
background-color: #f6f6f6;
|
|||
|
font-size: 13px;
|
|||
|
line-height: 1.3;
|
|||
|
}
|
|||
|
#TOC .toctitle {
|
|||
|
font-weight: bold;
|
|||
|
font-size: 15px;
|
|||
|
margin-left: 5px;
|
|||
|
}
|
|||
|
#TOC ul {
|
|||
|
padding-left: 40px;
|
|||
|
margin-left: -1.5em;
|
|||
|
margin-top: 5px;
|
|||
|
margin-bottom: 5px;
|
|||
|
}
|
|||
|
#TOC ul ul {
|
|||
|
margin-left: -2em;
|
|||
|
}
|
|||
|
#TOC li {
|
|||
|
line-height: 16px;
|
|||
|
}
|
|||
|
table {
|
|||
|
margin: 1em auto;
|
|||
|
border-width: 1px;
|
|||
|
border-color: #DDDDDD;
|
|||
|
border-style: outset;
|
|||
|
border-collapse: collapse;
|
|||
|
}
|
|||
|
table th {
|
|||
|
border-width: 2px;
|
|||
|
padding: 5px;
|
|||
|
border-style: inset;
|
|||
|
}
|
|||
|
table td {
|
|||
|
border-width: 1px;
|
|||
|
border-style: inset;
|
|||
|
line-height: 18px;
|
|||
|
padding: 5px 5px;
|
|||
|
}
|
|||
|
table, table th, table td {
|
|||
|
border-left-style: none;
|
|||
|
border-right-style: none;
|
|||
|
}
|
|||
|
table thead, table tr.even {
|
|||
|
background-color: #f7f7f7;
|
|||
|
}
|
|||
|
p {
|
|||
|
margin: 0.5em 0;
|
|||
|
}
|
|||
|
blockquote {
|
|||
|
background-color: #f6f6f6;
|
|||
|
padding: 0.25em 0.75em;
|
|||
|
}
|
|||
|
hr {
|
|||
|
border-style: solid;
|
|||
|
border: none;
|
|||
|
border-top: 1px solid #777;
|
|||
|
margin: 28px 0;
|
|||
|
}
|
|||
|
dl {
|
|||
|
margin-left: 0;
|
|||
|
}
|
|||
|
dl dd {
|
|||
|
margin-bottom: 13px;
|
|||
|
margin-left: 13px;
|
|||
|
}
|
|||
|
dl dt {
|
|||
|
font-weight: bold;
|
|||
|
}
|
|||
|
ul {
|
|||
|
margin-top: 0;
|
|||
|
}
|
|||
|
ul li {
|
|||
|
list-style: circle outside;
|
|||
|
}
|
|||
|
ul ul {
|
|||
|
margin-bottom: 0;
|
|||
|
}
|
|||
|
pre, code {
|
|||
|
background-color: #f7f7f7;
|
|||
|
border-radius: 3px;
|
|||
|
color: #333;
|
|||
|
white-space: pre-wrap;
|
|||
|
}
|
|||
|
pre {
|
|||
|
border-radius: 3px;
|
|||
|
margin: 5px 0px 10px 0px;
|
|||
|
padding: 10px;
|
|||
|
}
|
|||
|
pre:not([class]) {
|
|||
|
background-color: #f7f7f7;
|
|||
|
}
|
|||
|
code {
|
|||
|
font-family: Consolas, Monaco, 'Courier New', monospace;
|
|||
|
font-size: 85%;
|
|||
|
}
|
|||
|
p > code, li > code {
|
|||
|
padding: 2px 0px;
|
|||
|
}
|
|||
|
div.figure {
|
|||
|
text-align: center;
|
|||
|
}
|
|||
|
img {
|
|||
|
background-color: #FFFFFF;
|
|||
|
padding: 2px;
|
|||
|
border: 1px solid #DDDDDD;
|
|||
|
border-radius: 3px;
|
|||
|
border: 1px solid #CCCCCC;
|
|||
|
margin: 0 5px;
|
|||
|
}
|
|||
|
h1 {
|
|||
|
margin-top: 0;
|
|||
|
font-size: 35px;
|
|||
|
line-height: 40px;
|
|||
|
}
|
|||
|
h2 {
|
|||
|
border-bottom: 4px solid #f7f7f7;
|
|||
|
padding-top: 10px;
|
|||
|
padding-bottom: 2px;
|
|||
|
font-size: 145%;
|
|||
|
}
|
|||
|
h3 {
|
|||
|
border-bottom: 2px solid #f7f7f7;
|
|||
|
padding-top: 10px;
|
|||
|
font-size: 120%;
|
|||
|
}
|
|||
|
h4 {
|
|||
|
border-bottom: 1px solid #f7f7f7;
|
|||
|
margin-left: 8px;
|
|||
|
font-size: 105%;
|
|||
|
}
|
|||
|
h5, h6 {
|
|||
|
border-bottom: 1px solid #ccc;
|
|||
|
font-size: 105%;
|
|||
|
}
|
|||
|
a {
|
|||
|
color: #0033dd;
|
|||
|
text-decoration: none;
|
|||
|
}
|
|||
|
a:hover {
|
|||
|
color: #6666ff; }
|
|||
|
a:visited {
|
|||
|
color: #800080; }
|
|||
|
a:visited:hover {
|
|||
|
color: #BB00BB; }
|
|||
|
a[href^="http:"] {
|
|||
|
text-decoration: underline; }
|
|||
|
a[href^="https:"] {
|
|||
|
text-decoration: underline; }
|
|||
|
|
|||
|
code > span.kw { color: #555; font-weight: bold; }
|
|||
|
code > span.dt { color: #902000; }
|
|||
|
code > span.dv { color: #40a070; }
|
|||
|
code > span.bn { color: #d14; }
|
|||
|
code > span.fl { color: #d14; }
|
|||
|
code > span.ch { color: #d14; }
|
|||
|
code > span.st { color: #d14; }
|
|||
|
code > span.co { color: #888888; font-style: italic; }
|
|||
|
code > span.ot { color: #007020; }
|
|||
|
code > span.al { color: #ff0000; font-weight: bold; }
|
|||
|
code > span.fu { color: #900; font-weight: bold; }
|
|||
|
code > span.er { color: #a61717; background-color: #e3d2d2; }
|
|||
|
</style>
|
|||
|
|
|||
|
|
|||
|
|
|||
|
|
|||
|
</head>
|
|||
|
|
|||
|
<body>
|
|||
|
|
|||
|
|
|||
|
|
|||
|
|
|||
|
<h1 class="title toc-ignore">Extending ggplot2</h1>
|
|||
|
|
|||
|
|
|||
|
|
|||
|
<p>This vignette documents the official extension mechanism provided in
|
|||
|
ggplot2 2.0.0. This vignette is a high-level adjunct to the low-level
|
|||
|
details found in <code>?Stat</code>, <code>?Geom</code> and
|
|||
|
<code>?theme</code>. You’ll learn how to extend ggplot2 by creating a
|
|||
|
new stat, geom, or theme.</p>
|
|||
|
<p>As you read this document, you’ll see many things that will make you
|
|||
|
scratch your head and wonder why on earth is it designed this way?
|
|||
|
Mostly it’s historical accident - I wasn’t a terribly good R programmer
|
|||
|
when I started writing ggplot2 and I made a lot of questionable
|
|||
|
decisions. We cleaned up as many of those issues as possible in the
|
|||
|
2.0.0 release, but some fixes simply weren’t worth the effort.</p>
|
|||
|
<div id="ggproto" class="section level2">
|
|||
|
<h2>ggproto</h2>
|
|||
|
<p>All ggplot2 objects are built using the ggproto system of object
|
|||
|
oriented programming. This OO system is used only in one place: ggplot2.
|
|||
|
This is mostly historical accident: ggplot2 started off using <a href="https://cran.r-project.org/package=proto">proto</a> because I
|
|||
|
needed mutable objects. This was well before the creation of (the
|
|||
|
briefly lived) <a href="http://vita.had.co.nz/papers/mutatr.html">mutatr</a>, reference
|
|||
|
classes and R6: proto was the only game in town.</p>
|
|||
|
<p>But why ggproto? Well when we turned to add an official extension
|
|||
|
mechanism to ggplot2, we found a major problem that caused problems when
|
|||
|
proto objects were extended in a different package (methods were
|
|||
|
evaluated in ggplot2, not the package where the extension was added). We
|
|||
|
tried converting to R6, but it was a poor fit for the needs of ggplot2.
|
|||
|
We could’ve modified proto, but that would’ve first involved
|
|||
|
understanding exactly how proto worked, and secondly making sure that
|
|||
|
the changes didn’t affect other users of proto.</p>
|
|||
|
<p>It’s strange to say, but this is a case where inventing a new OO
|
|||
|
system was actually the right answer to the problem! Fortunately Winston
|
|||
|
is now very good at creating OO systems, so it only took him a day to
|
|||
|
come up with ggproto: it maintains all the features of proto that
|
|||
|
ggplot2 needs, while allowing cross package inheritance to work.</p>
|
|||
|
<p>Here’s a quick demo of ggproto in action:</p>
|
|||
|
<div class="sourceCode" id="cb1"><pre class="sourceCode r"><code class="sourceCode r"><span id="cb1-1"><a href="#cb1-1" tabindex="-1"></a>A <span class="ot"><-</span> <span class="fu">ggproto</span>(<span class="st">"A"</span>, <span class="cn">NULL</span>,</span>
|
|||
|
<span id="cb1-2"><a href="#cb1-2" tabindex="-1"></a> <span class="at">x =</span> <span class="dv">1</span>,</span>
|
|||
|
<span id="cb1-3"><a href="#cb1-3" tabindex="-1"></a> <span class="at">inc =</span> <span class="cf">function</span>(self) {</span>
|
|||
|
<span id="cb1-4"><a href="#cb1-4" tabindex="-1"></a> self<span class="sc">$</span>x <span class="ot"><-</span> self<span class="sc">$</span>x <span class="sc">+</span> <span class="dv">1</span></span>
|
|||
|
<span id="cb1-5"><a href="#cb1-5" tabindex="-1"></a> }</span>
|
|||
|
<span id="cb1-6"><a href="#cb1-6" tabindex="-1"></a>)</span>
|
|||
|
<span id="cb1-7"><a href="#cb1-7" tabindex="-1"></a>A<span class="sc">$</span>x</span>
|
|||
|
<span id="cb1-8"><a href="#cb1-8" tabindex="-1"></a><span class="co">#> [1] 1</span></span>
|
|||
|
<span id="cb1-9"><a href="#cb1-9" tabindex="-1"></a>A<span class="sc">$</span><span class="fu">inc</span>()</span>
|
|||
|
<span id="cb1-10"><a href="#cb1-10" tabindex="-1"></a>A<span class="sc">$</span>x</span>
|
|||
|
<span id="cb1-11"><a href="#cb1-11" tabindex="-1"></a><span class="co">#> [1] 2</span></span>
|
|||
|
<span id="cb1-12"><a href="#cb1-12" tabindex="-1"></a>A<span class="sc">$</span><span class="fu">inc</span>()</span>
|
|||
|
<span id="cb1-13"><a href="#cb1-13" tabindex="-1"></a>A<span class="sc">$</span><span class="fu">inc</span>()</span>
|
|||
|
<span id="cb1-14"><a href="#cb1-14" tabindex="-1"></a>A<span class="sc">$</span>x</span>
|
|||
|
<span id="cb1-15"><a href="#cb1-15" tabindex="-1"></a><span class="co">#> [1] 4</span></span></code></pre></div>
|
|||
|
<p>The majority of ggplot2 classes are immutable and static: the methods
|
|||
|
neither use nor modify state in the class. They’re mostly used as a
|
|||
|
convenient way of bundling related methods together.</p>
|
|||
|
<p>To create a new geom or stat, you will just create a new ggproto that
|
|||
|
inherits from <code>Stat</code>, <code>Geom</code> and override the
|
|||
|
methods described below.</p>
|
|||
|
</div>
|
|||
|
<div id="creating-a-new-stat" class="section level2">
|
|||
|
<h2>Creating a new stat</h2>
|
|||
|
<div id="the-simplest-stat" class="section level3">
|
|||
|
<h3>The simplest stat</h3>
|
|||
|
<p>We’ll start by creating a very simple stat: one that gives the convex
|
|||
|
hull (the <em>c</em> hull) of a set of points. First we create a new
|
|||
|
ggproto object that inherits from <code>Stat</code>:</p>
|
|||
|
<div class="sourceCode" id="cb2"><pre class="sourceCode r"><code class="sourceCode r"><span id="cb2-1"><a href="#cb2-1" tabindex="-1"></a>StatChull <span class="ot"><-</span> <span class="fu">ggproto</span>(<span class="st">"StatChull"</span>, Stat,</span>
|
|||
|
<span id="cb2-2"><a href="#cb2-2" tabindex="-1"></a> <span class="at">compute_group =</span> <span class="cf">function</span>(data, scales) {</span>
|
|||
|
<span id="cb2-3"><a href="#cb2-3" tabindex="-1"></a> data[<span class="fu">chull</span>(data<span class="sc">$</span>x, data<span class="sc">$</span>y), , drop <span class="ot">=</span> <span class="cn">FALSE</span>]</span>
|
|||
|
<span id="cb2-4"><a href="#cb2-4" tabindex="-1"></a> },</span>
|
|||
|
<span id="cb2-5"><a href="#cb2-5" tabindex="-1"></a> </span>
|
|||
|
<span id="cb2-6"><a href="#cb2-6" tabindex="-1"></a> <span class="at">required_aes =</span> <span class="fu">c</span>(<span class="st">"x"</span>, <span class="st">"y"</span>)</span>
|
|||
|
<span id="cb2-7"><a href="#cb2-7" tabindex="-1"></a>)</span></code></pre></div>
|
|||
|
<p>The two most important components are the
|
|||
|
<code>compute_group()</code> method (which does the computation), and
|
|||
|
the <code>required_aes</code> field, which lists which aesthetics must
|
|||
|
be present in order for the stat to work.</p>
|
|||
|
<p>Next we write a layer function. Unfortunately, due to an early design
|
|||
|
mistake I called these either <code>stat_()</code> or
|
|||
|
<code>geom_()</code>. A better decision would have been to call them
|
|||
|
<code>layer_()</code> functions: that’s a more accurate description
|
|||
|
because every layer involves a stat <em>and</em> a geom.</p>
|
|||
|
<p>All layer functions follow the same form - you specify defaults in
|
|||
|
the function arguments and then call the <code>layer()</code> function,
|
|||
|
sending <code>...</code> into the <code>params</code> argument. The
|
|||
|
arguments in <code>...</code> will either be arguments for the geom (if
|
|||
|
you’re making a stat wrapper), arguments for the stat (if you’re making
|
|||
|
a geom wrapper), or aesthetics to be set. <code>layer()</code> takes
|
|||
|
care of teasing the different parameters apart and making sure they’re
|
|||
|
stored in the right place:</p>
|
|||
|
<div class="sourceCode" id="cb3"><pre class="sourceCode r"><code class="sourceCode r"><span id="cb3-1"><a href="#cb3-1" tabindex="-1"></a>stat_chull <span class="ot"><-</span> <span class="cf">function</span>(<span class="at">mapping =</span> <span class="cn">NULL</span>, <span class="at">data =</span> <span class="cn">NULL</span>, <span class="at">geom =</span> <span class="st">"polygon"</span>,</span>
|
|||
|
<span id="cb3-2"><a href="#cb3-2" tabindex="-1"></a> <span class="at">position =</span> <span class="st">"identity"</span>, <span class="at">na.rm =</span> <span class="cn">FALSE</span>, <span class="at">show.legend =</span> <span class="cn">NA</span>, </span>
|
|||
|
<span id="cb3-3"><a href="#cb3-3" tabindex="-1"></a> <span class="at">inherit.aes =</span> <span class="cn">TRUE</span>, ...) {</span>
|
|||
|
<span id="cb3-4"><a href="#cb3-4" tabindex="-1"></a> <span class="fu">layer</span>(</span>
|
|||
|
<span id="cb3-5"><a href="#cb3-5" tabindex="-1"></a> <span class="at">stat =</span> StatChull, <span class="at">data =</span> data, <span class="at">mapping =</span> mapping, <span class="at">geom =</span> geom, </span>
|
|||
|
<span id="cb3-6"><a href="#cb3-6" tabindex="-1"></a> <span class="at">position =</span> position, <span class="at">show.legend =</span> show.legend, <span class="at">inherit.aes =</span> inherit.aes,</span>
|
|||
|
<span id="cb3-7"><a href="#cb3-7" tabindex="-1"></a> <span class="at">params =</span> <span class="fu">list</span>(<span class="at">na.rm =</span> na.rm, ...)</span>
|
|||
|
<span id="cb3-8"><a href="#cb3-8" tabindex="-1"></a> )</span>
|
|||
|
<span id="cb3-9"><a href="#cb3-9" tabindex="-1"></a>}</span></code></pre></div>
|
|||
|
<p>(Note that if you’re writing this in your own package, you’ll either
|
|||
|
need to call <code>ggplot2::layer()</code> explicitly, or import the
|
|||
|
<code>layer()</code> function into your package namespace.)</p>
|
|||
|
<p>Once we have a layer function we can try our new stat:</p>
|
|||
|
<div class="sourceCode" id="cb4"><pre class="sourceCode r"><code class="sourceCode r"><span id="cb4-1"><a href="#cb4-1" tabindex="-1"></a><span class="fu">ggplot</span>(mpg, <span class="fu">aes</span>(displ, hwy)) <span class="sc">+</span> </span>
|
|||
|
<span id="cb4-2"><a href="#cb4-2" tabindex="-1"></a> <span class="fu">geom_point</span>() <span class="sc">+</span> </span>
|
|||
|
<span id="cb4-3"><a href="#cb4-3" tabindex="-1"></a> <span class="fu">stat_chull</span>(<span class="at">fill =</span> <span class="cn">NA</span>, <span class="at">colour =</span> <span class="st">"black"</span>)</span></code></pre></div>
|
|||
|
<p><img src="
|
|||
|
<p>(We’ll see later how to change the defaults of the geom so that you
|
|||
|
don’t need to specify <code>fill = NA</code> every time.)</p>
|
|||
|
<p>Once we’ve written this basic object, ggplot2 gives a lot for free.
|
|||
|
For example, ggplot2 automatically preserves aesthetics that are
|
|||
|
constant within each group:</p>
|
|||
|
<div class="sourceCode" id="cb5"><pre class="sourceCode r"><code class="sourceCode r"><span id="cb5-1"><a href="#cb5-1" tabindex="-1"></a><span class="fu">ggplot</span>(mpg, <span class="fu">aes</span>(displ, hwy, <span class="at">colour =</span> drv)) <span class="sc">+</span> </span>
|
|||
|
<span id="cb5-2"><a href="#cb5-2" tabindex="-1"></a> <span class="fu">geom_point</span>() <span class="sc">+</span> </span>
|
|||
|
<span id="cb5-3"><a href="#cb5-3" tabindex="-1"></a> <span class="fu">stat_chull</span>(<span class="at">fill =</span> <span class="cn">NA</span>)</span></code></pre></div>
|
|||
|
<p><img src="
|
|||
|
<p>We can also override the default geom to display the convex hull in a
|
|||
|
different way:</p>
|
|||
|
<div class="sourceCode" id="cb6"><pre class="sourceCode r"><code class="sourceCode r"><span id="cb6-1"><a href="#cb6-1" tabindex="-1"></a><span class="fu">ggplot</span>(mpg, <span class="fu">aes</span>(displ, hwy)) <span class="sc">+</span> </span>
|
|||
|
<span id="cb6-2"><a href="#cb6-2" tabindex="-1"></a> <span class="fu">stat_chull</span>(<span class="at">geom =</span> <span class="st">"point"</span>, <span class="at">size =</span> <span class="dv">4</span>, <span class="at">colour =</span> <span class="st">"red"</span>) <span class="sc">+</span></span>
|
|||
|
<span id="cb6-3"><a href="#cb6-3" tabindex="-1"></a> <span class="fu">geom_point</span>()</span></code></pre></div>
|
|||
|
<p><img src="
|
|||
|
</div>
|
|||
|
<div id="stat-parameters" class="section level3">
|
|||
|
<h3>Stat parameters</h3>
|
|||
|
<p>A more complex stat will do some computation. Let’s implement a
|
|||
|
simple version of <code>geom_smooth()</code> that adds a line of best
|
|||
|
fit to a plot. We create a <code>StatLm</code> that inherits from
|
|||
|
<code>Stat</code> and a layer function, <code>stat_lm()</code>:</p>
|
|||
|
<div class="sourceCode" id="cb7"><pre class="sourceCode r"><code class="sourceCode r"><span id="cb7-1"><a href="#cb7-1" tabindex="-1"></a>StatLm <span class="ot"><-</span> <span class="fu">ggproto</span>(<span class="st">"StatLm"</span>, Stat, </span>
|
|||
|
<span id="cb7-2"><a href="#cb7-2" tabindex="-1"></a> <span class="at">required_aes =</span> <span class="fu">c</span>(<span class="st">"x"</span>, <span class="st">"y"</span>),</span>
|
|||
|
<span id="cb7-3"><a href="#cb7-3" tabindex="-1"></a> </span>
|
|||
|
<span id="cb7-4"><a href="#cb7-4" tabindex="-1"></a> <span class="at">compute_group =</span> <span class="cf">function</span>(data, scales) {</span>
|
|||
|
<span id="cb7-5"><a href="#cb7-5" tabindex="-1"></a> rng <span class="ot"><-</span> <span class="fu">range</span>(data<span class="sc">$</span>x, <span class="at">na.rm =</span> <span class="cn">TRUE</span>)</span>
|
|||
|
<span id="cb7-6"><a href="#cb7-6" tabindex="-1"></a> grid <span class="ot"><-</span> <span class="fu">data.frame</span>(<span class="at">x =</span> rng)</span>
|
|||
|
<span id="cb7-7"><a href="#cb7-7" tabindex="-1"></a> </span>
|
|||
|
<span id="cb7-8"><a href="#cb7-8" tabindex="-1"></a> mod <span class="ot"><-</span> <span class="fu">lm</span>(y <span class="sc">~</span> x, <span class="at">data =</span> data)</span>
|
|||
|
<span id="cb7-9"><a href="#cb7-9" tabindex="-1"></a> grid<span class="sc">$</span>y <span class="ot"><-</span> <span class="fu">predict</span>(mod, <span class="at">newdata =</span> grid)</span>
|
|||
|
<span id="cb7-10"><a href="#cb7-10" tabindex="-1"></a> </span>
|
|||
|
<span id="cb7-11"><a href="#cb7-11" tabindex="-1"></a> grid</span>
|
|||
|
<span id="cb7-12"><a href="#cb7-12" tabindex="-1"></a> }</span>
|
|||
|
<span id="cb7-13"><a href="#cb7-13" tabindex="-1"></a>)</span>
|
|||
|
<span id="cb7-14"><a href="#cb7-14" tabindex="-1"></a></span>
|
|||
|
<span id="cb7-15"><a href="#cb7-15" tabindex="-1"></a>stat_lm <span class="ot"><-</span> <span class="cf">function</span>(<span class="at">mapping =</span> <span class="cn">NULL</span>, <span class="at">data =</span> <span class="cn">NULL</span>, <span class="at">geom =</span> <span class="st">"line"</span>,</span>
|
|||
|
<span id="cb7-16"><a href="#cb7-16" tabindex="-1"></a> <span class="at">position =</span> <span class="st">"identity"</span>, <span class="at">na.rm =</span> <span class="cn">FALSE</span>, <span class="at">show.legend =</span> <span class="cn">NA</span>, </span>
|
|||
|
<span id="cb7-17"><a href="#cb7-17" tabindex="-1"></a> <span class="at">inherit.aes =</span> <span class="cn">TRUE</span>, ...) {</span>
|
|||
|
<span id="cb7-18"><a href="#cb7-18" tabindex="-1"></a> <span class="fu">layer</span>(</span>
|
|||
|
<span id="cb7-19"><a href="#cb7-19" tabindex="-1"></a> <span class="at">stat =</span> StatLm, <span class="at">data =</span> data, <span class="at">mapping =</span> mapping, <span class="at">geom =</span> geom, </span>
|
|||
|
<span id="cb7-20"><a href="#cb7-20" tabindex="-1"></a> <span class="at">position =</span> position, <span class="at">show.legend =</span> show.legend, <span class="at">inherit.aes =</span> inherit.aes,</span>
|
|||
|
<span id="cb7-21"><a href="#cb7-21" tabindex="-1"></a> <span class="at">params =</span> <span class="fu">list</span>(<span class="at">na.rm =</span> na.rm, ...)</span>
|
|||
|
<span id="cb7-22"><a href="#cb7-22" tabindex="-1"></a> )</span>
|
|||
|
<span id="cb7-23"><a href="#cb7-23" tabindex="-1"></a>}</span>
|
|||
|
<span id="cb7-24"><a href="#cb7-24" tabindex="-1"></a></span>
|
|||
|
<span id="cb7-25"><a href="#cb7-25" tabindex="-1"></a><span class="fu">ggplot</span>(mpg, <span class="fu">aes</span>(displ, hwy)) <span class="sc">+</span> </span>
|
|||
|
<span id="cb7-26"><a href="#cb7-26" tabindex="-1"></a> <span class="fu">geom_point</span>() <span class="sc">+</span> </span>
|
|||
|
<span id="cb7-27"><a href="#cb7-27" tabindex="-1"></a> <span class="fu">stat_lm</span>()</span></code></pre></div>
|
|||
|
<p><img src="
|
|||
|
<p><code>StatLm</code> is inflexible because it has no parameters. We
|
|||
|
might want to allow the user to control the model formula and the number
|
|||
|
of points used to generate the grid. To do so, we add arguments to the
|
|||
|
<code>compute_group()</code> method and our wrapper function:</p>
|
|||
|
<div class="sourceCode" id="cb8"><pre class="sourceCode r"><code class="sourceCode r"><span id="cb8-1"><a href="#cb8-1" tabindex="-1"></a>StatLm <span class="ot"><-</span> <span class="fu">ggproto</span>(<span class="st">"StatLm"</span>, Stat, </span>
|
|||
|
<span id="cb8-2"><a href="#cb8-2" tabindex="-1"></a> <span class="at">required_aes =</span> <span class="fu">c</span>(<span class="st">"x"</span>, <span class="st">"y"</span>),</span>
|
|||
|
<span id="cb8-3"><a href="#cb8-3" tabindex="-1"></a> </span>
|
|||
|
<span id="cb8-4"><a href="#cb8-4" tabindex="-1"></a> <span class="at">compute_group =</span> <span class="cf">function</span>(data, scales, params, <span class="at">n =</span> <span class="dv">100</span>, <span class="at">formula =</span> y <span class="sc">~</span> x) {</span>
|
|||
|
<span id="cb8-5"><a href="#cb8-5" tabindex="-1"></a> rng <span class="ot"><-</span> <span class="fu">range</span>(data<span class="sc">$</span>x, <span class="at">na.rm =</span> <span class="cn">TRUE</span>)</span>
|
|||
|
<span id="cb8-6"><a href="#cb8-6" tabindex="-1"></a> grid <span class="ot"><-</span> <span class="fu">data.frame</span>(<span class="at">x =</span> <span class="fu">seq</span>(rng[<span class="dv">1</span>], rng[<span class="dv">2</span>], <span class="at">length =</span> n))</span>
|
|||
|
<span id="cb8-7"><a href="#cb8-7" tabindex="-1"></a> </span>
|
|||
|
<span id="cb8-8"><a href="#cb8-8" tabindex="-1"></a> mod <span class="ot"><-</span> <span class="fu">lm</span>(formula, <span class="at">data =</span> data)</span>
|
|||
|
<span id="cb8-9"><a href="#cb8-9" tabindex="-1"></a> grid<span class="sc">$</span>y <span class="ot"><-</span> <span class="fu">predict</span>(mod, <span class="at">newdata =</span> grid)</span>
|
|||
|
<span id="cb8-10"><a href="#cb8-10" tabindex="-1"></a> </span>
|
|||
|
<span id="cb8-11"><a href="#cb8-11" tabindex="-1"></a> grid</span>
|
|||
|
<span id="cb8-12"><a href="#cb8-12" tabindex="-1"></a> }</span>
|
|||
|
<span id="cb8-13"><a href="#cb8-13" tabindex="-1"></a>)</span>
|
|||
|
<span id="cb8-14"><a href="#cb8-14" tabindex="-1"></a></span>
|
|||
|
<span id="cb8-15"><a href="#cb8-15" tabindex="-1"></a>stat_lm <span class="ot"><-</span> <span class="cf">function</span>(<span class="at">mapping =</span> <span class="cn">NULL</span>, <span class="at">data =</span> <span class="cn">NULL</span>, <span class="at">geom =</span> <span class="st">"line"</span>,</span>
|
|||
|
<span id="cb8-16"><a href="#cb8-16" tabindex="-1"></a> <span class="at">position =</span> <span class="st">"identity"</span>, <span class="at">na.rm =</span> <span class="cn">FALSE</span>, <span class="at">show.legend =</span> <span class="cn">NA</span>, </span>
|
|||
|
<span id="cb8-17"><a href="#cb8-17" tabindex="-1"></a> <span class="at">inherit.aes =</span> <span class="cn">TRUE</span>, <span class="at">n =</span> <span class="dv">50</span>, <span class="at">formula =</span> y <span class="sc">~</span> x, </span>
|
|||
|
<span id="cb8-18"><a href="#cb8-18" tabindex="-1"></a> ...) {</span>
|
|||
|
<span id="cb8-19"><a href="#cb8-19" tabindex="-1"></a> <span class="fu">layer</span>(</span>
|
|||
|
<span id="cb8-20"><a href="#cb8-20" tabindex="-1"></a> <span class="at">stat =</span> StatLm, <span class="at">data =</span> data, <span class="at">mapping =</span> mapping, <span class="at">geom =</span> geom, </span>
|
|||
|
<span id="cb8-21"><a href="#cb8-21" tabindex="-1"></a> <span class="at">position =</span> position, <span class="at">show.legend =</span> show.legend, <span class="at">inherit.aes =</span> inherit.aes,</span>
|
|||
|
<span id="cb8-22"><a href="#cb8-22" tabindex="-1"></a> <span class="at">params =</span> <span class="fu">list</span>(<span class="at">n =</span> n, <span class="at">formula =</span> formula, <span class="at">na.rm =</span> na.rm, ...)</span>
|
|||
|
<span id="cb8-23"><a href="#cb8-23" tabindex="-1"></a> )</span>
|
|||
|
<span id="cb8-24"><a href="#cb8-24" tabindex="-1"></a>}</span>
|
|||
|
<span id="cb8-25"><a href="#cb8-25" tabindex="-1"></a></span>
|
|||
|
<span id="cb8-26"><a href="#cb8-26" tabindex="-1"></a><span class="fu">ggplot</span>(mpg, <span class="fu">aes</span>(displ, hwy)) <span class="sc">+</span> </span>
|
|||
|
<span id="cb8-27"><a href="#cb8-27" tabindex="-1"></a> <span class="fu">geom_point</span>() <span class="sc">+</span> </span>
|
|||
|
<span id="cb8-28"><a href="#cb8-28" tabindex="-1"></a> <span class="fu">stat_lm</span>(<span class="at">formula =</span> y <span class="sc">~</span> <span class="fu">poly</span>(x, <span class="dv">10</span>)) <span class="sc">+</span> </span>
|
|||
|
<span id="cb8-29"><a href="#cb8-29" tabindex="-1"></a> <span class="fu">stat_lm</span>(<span class="at">formula =</span> y <span class="sc">~</span> <span class="fu">poly</span>(x, <span class="dv">10</span>), <span class="at">geom =</span> <span class="st">"point"</span>, <span class="at">colour =</span> <span class="st">"red"</span>, <span class="at">n =</span> <span class="dv">20</span>)</span></code></pre></div>
|
|||
|
<p><img src="
|
|||
|
<p>Note that we don’t <em>have</em> to explicitly include the new
|
|||
|
parameters in the arguments for the layer, <code>...</code> will get
|
|||
|
passed to the right place anyway. But you’ll need to document them
|
|||
|
somewhere so the user knows about them. Here’s a brief example. Note
|
|||
|
<code>@inheritParams ggplot2::stat_identity</code>: that will
|
|||
|
automatically inherit documentation for all the parameters also defined
|
|||
|
for <code>stat_identity()</code>.</p>
|
|||
|
<div class="sourceCode" id="cb9"><pre class="sourceCode r"><code class="sourceCode r"><span id="cb9-1"><a href="#cb9-1" tabindex="-1"></a><span class="co">#' @export</span></span>
|
|||
|
<span id="cb9-2"><a href="#cb9-2" tabindex="-1"></a><span class="co">#' @inheritParams ggplot2::stat_identity</span></span>
|
|||
|
<span id="cb9-3"><a href="#cb9-3" tabindex="-1"></a><span class="co">#' @param formula The modelling formula passed to \code{lm}. Should only </span></span>
|
|||
|
<span id="cb9-4"><a href="#cb9-4" tabindex="-1"></a><span class="co">#' involve \code{y} and \code{x}</span></span>
|
|||
|
<span id="cb9-5"><a href="#cb9-5" tabindex="-1"></a><span class="co">#' @param n Number of points used for interpolation.</span></span>
|
|||
|
<span id="cb9-6"><a href="#cb9-6" tabindex="-1"></a>stat_lm <span class="ot"><-</span> <span class="cf">function</span>(<span class="at">mapping =</span> <span class="cn">NULL</span>, <span class="at">data =</span> <span class="cn">NULL</span>, <span class="at">geom =</span> <span class="st">"line"</span>,</span>
|
|||
|
<span id="cb9-7"><a href="#cb9-7" tabindex="-1"></a> <span class="at">position =</span> <span class="st">"identity"</span>, <span class="at">na.rm =</span> <span class="cn">FALSE</span>, <span class="at">show.legend =</span> <span class="cn">NA</span>, </span>
|
|||
|
<span id="cb9-8"><a href="#cb9-8" tabindex="-1"></a> <span class="at">inherit.aes =</span> <span class="cn">TRUE</span>, <span class="at">n =</span> <span class="dv">50</span>, <span class="at">formula =</span> y <span class="sc">~</span> x, </span>
|
|||
|
<span id="cb9-9"><a href="#cb9-9" tabindex="-1"></a> ...) {</span>
|
|||
|
<span id="cb9-10"><a href="#cb9-10" tabindex="-1"></a> <span class="fu">layer</span>(</span>
|
|||
|
<span id="cb9-11"><a href="#cb9-11" tabindex="-1"></a> <span class="at">stat =</span> StatLm, <span class="at">data =</span> data, <span class="at">mapping =</span> mapping, <span class="at">geom =</span> geom, </span>
|
|||
|
<span id="cb9-12"><a href="#cb9-12" tabindex="-1"></a> <span class="at">position =</span> position, <span class="at">show.legend =</span> show.legend, <span class="at">inherit.aes =</span> inherit.aes,</span>
|
|||
|
<span id="cb9-13"><a href="#cb9-13" tabindex="-1"></a> <span class="at">params =</span> <span class="fu">list</span>(<span class="at">n =</span> n, <span class="at">formula =</span> formula, <span class="at">na.rm =</span> na.rm, ...)</span>
|
|||
|
<span id="cb9-14"><a href="#cb9-14" tabindex="-1"></a> )</span>
|
|||
|
<span id="cb9-15"><a href="#cb9-15" tabindex="-1"></a>}</span></code></pre></div>
|
|||
|
<p><code>stat_lm()</code> must be exported if you want other people to
|
|||
|
use it. You could also consider exporting <code>StatLm</code> if you
|
|||
|
want people to extend the underlying object; this should be done with
|
|||
|
care.</p>
|
|||
|
</div>
|
|||
|
<div id="picking-defaults" class="section level3">
|
|||
|
<h3>Picking defaults</h3>
|
|||
|
<p>Sometimes you have calculations that should be performed once for the
|
|||
|
complete dataset, not once for each group. This is useful for picking
|
|||
|
sensible default values. For example, if we want to do a density
|
|||
|
estimate, it’s reasonable to pick one bandwidth for the whole plot. The
|
|||
|
following Stat creates a variation of the <code>stat_density()</code>
|
|||
|
that picks one bandwidth for all groups by choosing the mean of the
|
|||
|
“best” bandwidth for each group (I have no theoretical justification for
|
|||
|
this, but it doesn’t seem unreasonable).</p>
|
|||
|
<p>To do this we override the <code>setup_params()</code> method. It’s
|
|||
|
passed the data and a list of params, and returns an updated list.</p>
|
|||
|
<div class="sourceCode" id="cb10"><pre class="sourceCode r"><code class="sourceCode r"><span id="cb10-1"><a href="#cb10-1" tabindex="-1"></a>StatDensityCommon <span class="ot"><-</span> <span class="fu">ggproto</span>(<span class="st">"StatDensityCommon"</span>, Stat, </span>
|
|||
|
<span id="cb10-2"><a href="#cb10-2" tabindex="-1"></a> <span class="at">required_aes =</span> <span class="st">"x"</span>,</span>
|
|||
|
<span id="cb10-3"><a href="#cb10-3" tabindex="-1"></a> </span>
|
|||
|
<span id="cb10-4"><a href="#cb10-4" tabindex="-1"></a> <span class="at">setup_params =</span> <span class="cf">function</span>(data, params) {</span>
|
|||
|
<span id="cb10-5"><a href="#cb10-5" tabindex="-1"></a> <span class="cf">if</span> (<span class="sc">!</span><span class="fu">is.null</span>(params<span class="sc">$</span>bandwidth))</span>
|
|||
|
<span id="cb10-6"><a href="#cb10-6" tabindex="-1"></a> <span class="fu">return</span>(params)</span>
|
|||
|
<span id="cb10-7"><a href="#cb10-7" tabindex="-1"></a> </span>
|
|||
|
<span id="cb10-8"><a href="#cb10-8" tabindex="-1"></a> xs <span class="ot"><-</span> <span class="fu">split</span>(data<span class="sc">$</span>x, data<span class="sc">$</span>group)</span>
|
|||
|
<span id="cb10-9"><a href="#cb10-9" tabindex="-1"></a> bws <span class="ot"><-</span> <span class="fu">vapply</span>(xs, bw.nrd0, <span class="fu">numeric</span>(<span class="dv">1</span>))</span>
|
|||
|
<span id="cb10-10"><a href="#cb10-10" tabindex="-1"></a> bw <span class="ot"><-</span> <span class="fu">mean</span>(bws)</span>
|
|||
|
<span id="cb10-11"><a href="#cb10-11" tabindex="-1"></a> <span class="fu">message</span>(<span class="st">"Picking bandwidth of "</span>, <span class="fu">signif</span>(bw, <span class="dv">3</span>))</span>
|
|||
|
<span id="cb10-12"><a href="#cb10-12" tabindex="-1"></a> </span>
|
|||
|
<span id="cb10-13"><a href="#cb10-13" tabindex="-1"></a> params<span class="sc">$</span>bandwidth <span class="ot"><-</span> bw</span>
|
|||
|
<span id="cb10-14"><a href="#cb10-14" tabindex="-1"></a> params</span>
|
|||
|
<span id="cb10-15"><a href="#cb10-15" tabindex="-1"></a> },</span>
|
|||
|
<span id="cb10-16"><a href="#cb10-16" tabindex="-1"></a> </span>
|
|||
|
<span id="cb10-17"><a href="#cb10-17" tabindex="-1"></a> <span class="at">compute_group =</span> <span class="cf">function</span>(data, scales, <span class="at">bandwidth =</span> <span class="dv">1</span>) {</span>
|
|||
|
<span id="cb10-18"><a href="#cb10-18" tabindex="-1"></a> d <span class="ot"><-</span> <span class="fu">density</span>(data<span class="sc">$</span>x, <span class="at">bw =</span> bandwidth)</span>
|
|||
|
<span id="cb10-19"><a href="#cb10-19" tabindex="-1"></a> <span class="fu">data.frame</span>(<span class="at">x =</span> d<span class="sc">$</span>x, <span class="at">y =</span> d<span class="sc">$</span>y)</span>
|
|||
|
<span id="cb10-20"><a href="#cb10-20" tabindex="-1"></a> } </span>
|
|||
|
<span id="cb10-21"><a href="#cb10-21" tabindex="-1"></a>)</span>
|
|||
|
<span id="cb10-22"><a href="#cb10-22" tabindex="-1"></a></span>
|
|||
|
<span id="cb10-23"><a href="#cb10-23" tabindex="-1"></a>stat_density_common <span class="ot"><-</span> <span class="cf">function</span>(<span class="at">mapping =</span> <span class="cn">NULL</span>, <span class="at">data =</span> <span class="cn">NULL</span>, <span class="at">geom =</span> <span class="st">"line"</span>,</span>
|
|||
|
<span id="cb10-24"><a href="#cb10-24" tabindex="-1"></a> <span class="at">position =</span> <span class="st">"identity"</span>, <span class="at">na.rm =</span> <span class="cn">FALSE</span>, <span class="at">show.legend =</span> <span class="cn">NA</span>, </span>
|
|||
|
<span id="cb10-25"><a href="#cb10-25" tabindex="-1"></a> <span class="at">inherit.aes =</span> <span class="cn">TRUE</span>, <span class="at">bandwidth =</span> <span class="cn">NULL</span>,</span>
|
|||
|
<span id="cb10-26"><a href="#cb10-26" tabindex="-1"></a> ...) {</span>
|
|||
|
<span id="cb10-27"><a href="#cb10-27" tabindex="-1"></a> <span class="fu">layer</span>(</span>
|
|||
|
<span id="cb10-28"><a href="#cb10-28" tabindex="-1"></a> <span class="at">stat =</span> StatDensityCommon, <span class="at">data =</span> data, <span class="at">mapping =</span> mapping, <span class="at">geom =</span> geom, </span>
|
|||
|
<span id="cb10-29"><a href="#cb10-29" tabindex="-1"></a> <span class="at">position =</span> position, <span class="at">show.legend =</span> show.legend, <span class="at">inherit.aes =</span> inherit.aes,</span>
|
|||
|
<span id="cb10-30"><a href="#cb10-30" tabindex="-1"></a> <span class="at">params =</span> <span class="fu">list</span>(<span class="at">bandwidth =</span> bandwidth, <span class="at">na.rm =</span> na.rm, ...)</span>
|
|||
|
<span id="cb10-31"><a href="#cb10-31" tabindex="-1"></a> )</span>
|
|||
|
<span id="cb10-32"><a href="#cb10-32" tabindex="-1"></a>}</span>
|
|||
|
<span id="cb10-33"><a href="#cb10-33" tabindex="-1"></a></span>
|
|||
|
<span id="cb10-34"><a href="#cb10-34" tabindex="-1"></a><span class="fu">ggplot</span>(mpg, <span class="fu">aes</span>(displ, <span class="at">colour =</span> drv)) <span class="sc">+</span> </span>
|
|||
|
<span id="cb10-35"><a href="#cb10-35" tabindex="-1"></a> <span class="fu">stat_density_common</span>()</span>
|
|||
|
<span id="cb10-36"><a href="#cb10-36" tabindex="-1"></a><span class="co">#> Picking bandwidth of 0.345</span></span></code></pre></div>
|
|||
|
<p><img src="
|
|||
|
<div class="sourceCode" id="cb11"><pre class="sourceCode r"><code class="sourceCode r"><span id="cb11-1"><a href="#cb11-1" tabindex="-1"></a></span>
|
|||
|
<span id="cb11-2"><a href="#cb11-2" tabindex="-1"></a><span class="fu">ggplot</span>(mpg, <span class="fu">aes</span>(displ, <span class="at">colour =</span> drv)) <span class="sc">+</span> </span>
|
|||
|
<span id="cb11-3"><a href="#cb11-3" tabindex="-1"></a> <span class="fu">stat_density_common</span>(<span class="at">bandwidth =</span> <span class="fl">0.5</span>)</span></code></pre></div>
|
|||
|
<p><img src="
|
|||
|
<p>I recommend using <code>NULL</code> as a default value. If you pick
|
|||
|
important parameters automatically, it’s a good idea to
|
|||
|
<code>message()</code> to the user (and when printing a floating point
|
|||
|
parameter, using <code>signif()</code> to show only a few significant
|
|||
|
digits).</p>
|
|||
|
</div>
|
|||
|
<div id="variable-names-and-default-aesthetics" class="section level3">
|
|||
|
<h3>Variable names and default aesthetics</h3>
|
|||
|
<p>This stat illustrates another important point. If we want to make
|
|||
|
this stat usable with other geoms, we should return a variable called
|
|||
|
<code>density</code> instead of <code>y</code>. Then we can set up the
|
|||
|
<code>default_aes</code> to automatically map <code>density</code> to
|
|||
|
<code>y</code>, which allows the user to override it to use with
|
|||
|
different geoms:</p>
|
|||
|
<div class="sourceCode" id="cb12"><pre class="sourceCode r"><code class="sourceCode r"><span id="cb12-1"><a href="#cb12-1" tabindex="-1"></a>StatDensityCommon <span class="ot"><-</span> <span class="fu">ggproto</span>(<span class="st">"StatDensity2"</span>, Stat, </span>
|
|||
|
<span id="cb12-2"><a href="#cb12-2" tabindex="-1"></a> <span class="at">required_aes =</span> <span class="st">"x"</span>,</span>
|
|||
|
<span id="cb12-3"><a href="#cb12-3" tabindex="-1"></a> <span class="at">default_aes =</span> <span class="fu">aes</span>(<span class="at">y =</span> <span class="fu">after_stat</span>(density)),</span>
|
|||
|
<span id="cb12-4"><a href="#cb12-4" tabindex="-1"></a></span>
|
|||
|
<span id="cb12-5"><a href="#cb12-5" tabindex="-1"></a> <span class="at">compute_group =</span> <span class="cf">function</span>(data, scales, <span class="at">bandwidth =</span> <span class="dv">1</span>) {</span>
|
|||
|
<span id="cb12-6"><a href="#cb12-6" tabindex="-1"></a> d <span class="ot"><-</span> <span class="fu">density</span>(data<span class="sc">$</span>x, <span class="at">bw =</span> bandwidth)</span>
|
|||
|
<span id="cb12-7"><a href="#cb12-7" tabindex="-1"></a> <span class="fu">data.frame</span>(<span class="at">x =</span> d<span class="sc">$</span>x, <span class="at">density =</span> d<span class="sc">$</span>y)</span>
|
|||
|
<span id="cb12-8"><a href="#cb12-8" tabindex="-1"></a> } </span>
|
|||
|
<span id="cb12-9"><a href="#cb12-9" tabindex="-1"></a>)</span>
|
|||
|
<span id="cb12-10"><a href="#cb12-10" tabindex="-1"></a></span>
|
|||
|
<span id="cb12-11"><a href="#cb12-11" tabindex="-1"></a><span class="fu">ggplot</span>(mpg, <span class="fu">aes</span>(displ, drv, <span class="at">colour =</span> <span class="fu">after_stat</span>(density))) <span class="sc">+</span> </span>
|
|||
|
<span id="cb12-12"><a href="#cb12-12" tabindex="-1"></a> <span class="fu">stat_density_common</span>(<span class="at">bandwidth =</span> <span class="dv">1</span>, <span class="at">geom =</span> <span class="st">"point"</span>)</span></code></pre></div>
|
|||
|
<p><img src="
|
|||
|
<p>However, using this stat with the area geom doesn’t work quite right.
|
|||
|
The areas don’t stack on top of each other:</p>
|
|||
|
<div class="sourceCode" id="cb13"><pre class="sourceCode r"><code class="sourceCode r"><span id="cb13-1"><a href="#cb13-1" tabindex="-1"></a><span class="fu">ggplot</span>(mpg, <span class="fu">aes</span>(displ, <span class="at">fill =</span> drv)) <span class="sc">+</span> </span>
|
|||
|
<span id="cb13-2"><a href="#cb13-2" tabindex="-1"></a> <span class="fu">stat_density_common</span>(<span class="at">bandwidth =</span> <span class="dv">1</span>, <span class="at">geom =</span> <span class="st">"area"</span>, <span class="at">position =</span> <span class="st">"stack"</span>)</span></code></pre></div>
|
|||
|
<p><img src="
|
|||
|
<p>This is because each density is computed independently, and the
|
|||
|
estimated <code>x</code>s don’t line up. We can resolve that issue by
|
|||
|
computing the range of the data once in <code>setup_params()</code>.</p>
|
|||
|
<div class="sourceCode" id="cb14"><pre class="sourceCode r"><code class="sourceCode r"><span id="cb14-1"><a href="#cb14-1" tabindex="-1"></a>StatDensityCommon <span class="ot"><-</span> <span class="fu">ggproto</span>(<span class="st">"StatDensityCommon"</span>, Stat, </span>
|
|||
|
<span id="cb14-2"><a href="#cb14-2" tabindex="-1"></a> <span class="at">required_aes =</span> <span class="st">"x"</span>,</span>
|
|||
|
<span id="cb14-3"><a href="#cb14-3" tabindex="-1"></a> <span class="at">default_aes =</span> <span class="fu">aes</span>(<span class="at">y =</span> <span class="fu">after_stat</span>(density)),</span>
|
|||
|
<span id="cb14-4"><a href="#cb14-4" tabindex="-1"></a></span>
|
|||
|
<span id="cb14-5"><a href="#cb14-5" tabindex="-1"></a> <span class="at">setup_params =</span> <span class="cf">function</span>(data, params) {</span>
|
|||
|
<span id="cb14-6"><a href="#cb14-6" tabindex="-1"></a> min <span class="ot"><-</span> <span class="fu">min</span>(data<span class="sc">$</span>x) <span class="sc">-</span> <span class="dv">3</span> <span class="sc">*</span> params<span class="sc">$</span>bandwidth</span>
|
|||
|
<span id="cb14-7"><a href="#cb14-7" tabindex="-1"></a> max <span class="ot"><-</span> <span class="fu">max</span>(data<span class="sc">$</span>x) <span class="sc">+</span> <span class="dv">3</span> <span class="sc">*</span> params<span class="sc">$</span>bandwidth</span>
|
|||
|
<span id="cb14-8"><a href="#cb14-8" tabindex="-1"></a> </span>
|
|||
|
<span id="cb14-9"><a href="#cb14-9" tabindex="-1"></a> <span class="fu">list</span>(</span>
|
|||
|
<span id="cb14-10"><a href="#cb14-10" tabindex="-1"></a> <span class="at">bandwidth =</span> params<span class="sc">$</span>bandwidth,</span>
|
|||
|
<span id="cb14-11"><a href="#cb14-11" tabindex="-1"></a> <span class="at">min =</span> min,</span>
|
|||
|
<span id="cb14-12"><a href="#cb14-12" tabindex="-1"></a> <span class="at">max =</span> max,</span>
|
|||
|
<span id="cb14-13"><a href="#cb14-13" tabindex="-1"></a> <span class="at">na.rm =</span> params<span class="sc">$</span>na.rm</span>
|
|||
|
<span id="cb14-14"><a href="#cb14-14" tabindex="-1"></a> )</span>
|
|||
|
<span id="cb14-15"><a href="#cb14-15" tabindex="-1"></a> },</span>
|
|||
|
<span id="cb14-16"><a href="#cb14-16" tabindex="-1"></a> </span>
|
|||
|
<span id="cb14-17"><a href="#cb14-17" tabindex="-1"></a> <span class="at">compute_group =</span> <span class="cf">function</span>(data, scales, min, max, <span class="at">bandwidth =</span> <span class="dv">1</span>) {</span>
|
|||
|
<span id="cb14-18"><a href="#cb14-18" tabindex="-1"></a> d <span class="ot"><-</span> <span class="fu">density</span>(data<span class="sc">$</span>x, <span class="at">bw =</span> bandwidth, <span class="at">from =</span> min, <span class="at">to =</span> max)</span>
|
|||
|
<span id="cb14-19"><a href="#cb14-19" tabindex="-1"></a> <span class="fu">data.frame</span>(<span class="at">x =</span> d<span class="sc">$</span>x, <span class="at">density =</span> d<span class="sc">$</span>y)</span>
|
|||
|
<span id="cb14-20"><a href="#cb14-20" tabindex="-1"></a> } </span>
|
|||
|
<span id="cb14-21"><a href="#cb14-21" tabindex="-1"></a>)</span>
|
|||
|
<span id="cb14-22"><a href="#cb14-22" tabindex="-1"></a></span>
|
|||
|
<span id="cb14-23"><a href="#cb14-23" tabindex="-1"></a><span class="fu">ggplot</span>(mpg, <span class="fu">aes</span>(displ, <span class="at">fill =</span> drv)) <span class="sc">+</span> </span>
|
|||
|
<span id="cb14-24"><a href="#cb14-24" tabindex="-1"></a> <span class="fu">stat_density_common</span>(<span class="at">bandwidth =</span> <span class="dv">1</span>, <span class="at">geom =</span> <span class="st">"area"</span>, <span class="at">position =</span> <span class="st">"stack"</span>)</span></code></pre></div>
|
|||
|
<p><img src="
|
|||
|
<div class="sourceCode" id="cb15"><pre class="sourceCode r"><code class="sourceCode r"><span id="cb15-1"><a href="#cb15-1" tabindex="-1"></a><span class="fu">ggplot</span>(mpg, <span class="fu">aes</span>(displ, drv, <span class="at">fill =</span> <span class="fu">after_stat</span>(density))) <span class="sc">+</span> </span>
|
|||
|
<span id="cb15-2"><a href="#cb15-2" tabindex="-1"></a> <span class="fu">stat_density_common</span>(<span class="at">bandwidth =</span> <span class="dv">1</span>, <span class="at">geom =</span> <span class="st">"raster"</span>)</span></code></pre></div>
|
|||
|
<p><img src="
|
|||
|
</div>
|
|||
|
<div id="exercises" class="section level3">
|
|||
|
<h3>Exercises</h3>
|
|||
|
<ol style="list-style-type: decimal">
|
|||
|
<li><p>Extend <code>stat_chull</code> to compute the alpha hull, as from
|
|||
|
the <a href="https://cran.r-project.org/package=alphahull">alphahull</a>
|
|||
|
package. Your new stat should take an <code>alpha</code>
|
|||
|
argument.</p></li>
|
|||
|
<li><p>Modify the final version of <code>StatDensityCommon</code> to
|
|||
|
allow the user to specify the <code>min</code> and <code>max</code>
|
|||
|
parameters. You’ll need to modify both the layer function and the
|
|||
|
<code>compute_group()</code> method.</p>
|
|||
|
<p>Note: be careful when adding parameters to a layer function. The
|
|||
|
following names <em>col</em>, <em>color</em>, <em>pch</em>,
|
|||
|
<em>cex</em>, <em>lty</em>, <em>lwd</em>, <em>srt</em>, <em>adj</em>,
|
|||
|
<em>bg</em>, <em>fg</em>, <em>min</em>, and <em>max</em> are
|
|||
|
intentionally renamed to accommodate base graphical parameter names. For
|
|||
|
example, a value passed as <em>min</em> to a layer appears as
|
|||
|
<em>ymin</em> in the <code>setup_params</code> list of params. It is
|
|||
|
recommended you avoid using these names for layer parameters.</p></li>
|
|||
|
<li><p>Compare and contrast <code>StatLm</code> to
|
|||
|
<code>ggplot2::StatSmooth</code>. What key differences make
|
|||
|
<code>StatSmooth</code> more complex than <code>StatLm</code>?</p></li>
|
|||
|
</ol>
|
|||
|
</div>
|
|||
|
</div>
|
|||
|
<div id="creating-a-new-geom" class="section level2">
|
|||
|
<h2>Creating a new geom</h2>
|
|||
|
<p>It’s harder to create a new geom than a new stat because you also
|
|||
|
need to know some grid. ggplot2 is built on top of grid, so you’ll need
|
|||
|
to know the basics of drawing with grid. If you’re serious about adding
|
|||
|
a new geom, I’d recommend buying <a href="https://www.amazon.com/dp/B00I60M26G/ref=cm_sw_su_dp">R
|
|||
|
graphics</a> by Paul Murrell. It tells you everything you need to know
|
|||
|
about drawing with grid.</p>
|
|||
|
<div id="a-simple-geom" class="section level3">
|
|||
|
<h3>A simple geom</h3>
|
|||
|
<p>It’s easiest to start with a simple example. The code below is a
|
|||
|
simplified version of <code>geom_point()</code>:</p>
|
|||
|
<div class="sourceCode" id="cb16"><pre class="sourceCode r"><code class="sourceCode r"><span id="cb16-1"><a href="#cb16-1" tabindex="-1"></a>GeomSimplePoint <span class="ot"><-</span> <span class="fu">ggproto</span>(<span class="st">"GeomSimplePoint"</span>, Geom,</span>
|
|||
|
<span id="cb16-2"><a href="#cb16-2" tabindex="-1"></a> <span class="at">required_aes =</span> <span class="fu">c</span>(<span class="st">"x"</span>, <span class="st">"y"</span>),</span>
|
|||
|
<span id="cb16-3"><a href="#cb16-3" tabindex="-1"></a> <span class="at">default_aes =</span> <span class="fu">aes</span>(<span class="at">shape =</span> <span class="dv">19</span>, <span class="at">colour =</span> <span class="st">"black"</span>),</span>
|
|||
|
<span id="cb16-4"><a href="#cb16-4" tabindex="-1"></a> <span class="at">draw_key =</span> draw_key_point,</span>
|
|||
|
<span id="cb16-5"><a href="#cb16-5" tabindex="-1"></a></span>
|
|||
|
<span id="cb16-6"><a href="#cb16-6" tabindex="-1"></a> <span class="at">draw_panel =</span> <span class="cf">function</span>(data, panel_params, coord) {</span>
|
|||
|
<span id="cb16-7"><a href="#cb16-7" tabindex="-1"></a> coords <span class="ot"><-</span> coord<span class="sc">$</span><span class="fu">transform</span>(data, panel_params)</span>
|
|||
|
<span id="cb16-8"><a href="#cb16-8" tabindex="-1"></a> grid<span class="sc">::</span><span class="fu">pointsGrob</span>(</span>
|
|||
|
<span id="cb16-9"><a href="#cb16-9" tabindex="-1"></a> coords<span class="sc">$</span>x, coords<span class="sc">$</span>y,</span>
|
|||
|
<span id="cb16-10"><a href="#cb16-10" tabindex="-1"></a> <span class="at">pch =</span> coords<span class="sc">$</span>shape,</span>
|
|||
|
<span id="cb16-11"><a href="#cb16-11" tabindex="-1"></a> <span class="at">gp =</span> grid<span class="sc">::</span><span class="fu">gpar</span>(<span class="at">col =</span> coords<span class="sc">$</span>colour)</span>
|
|||
|
<span id="cb16-12"><a href="#cb16-12" tabindex="-1"></a> )</span>
|
|||
|
<span id="cb16-13"><a href="#cb16-13" tabindex="-1"></a> }</span>
|
|||
|
<span id="cb16-14"><a href="#cb16-14" tabindex="-1"></a>)</span>
|
|||
|
<span id="cb16-15"><a href="#cb16-15" tabindex="-1"></a></span>
|
|||
|
<span id="cb16-16"><a href="#cb16-16" tabindex="-1"></a>geom_simple_point <span class="ot"><-</span> <span class="cf">function</span>(<span class="at">mapping =</span> <span class="cn">NULL</span>, <span class="at">data =</span> <span class="cn">NULL</span>, <span class="at">stat =</span> <span class="st">"identity"</span>,</span>
|
|||
|
<span id="cb16-17"><a href="#cb16-17" tabindex="-1"></a> <span class="at">position =</span> <span class="st">"identity"</span>, <span class="at">na.rm =</span> <span class="cn">FALSE</span>, <span class="at">show.legend =</span> <span class="cn">NA</span>, </span>
|
|||
|
<span id="cb16-18"><a href="#cb16-18" tabindex="-1"></a> <span class="at">inherit.aes =</span> <span class="cn">TRUE</span>, ...) {</span>
|
|||
|
<span id="cb16-19"><a href="#cb16-19" tabindex="-1"></a> <span class="fu">layer</span>(</span>
|
|||
|
<span id="cb16-20"><a href="#cb16-20" tabindex="-1"></a> <span class="at">geom =</span> GeomSimplePoint, <span class="at">mapping =</span> mapping, <span class="at">data =</span> data, <span class="at">stat =</span> stat, </span>
|
|||
|
<span id="cb16-21"><a href="#cb16-21" tabindex="-1"></a> <span class="at">position =</span> position, <span class="at">show.legend =</span> show.legend, <span class="at">inherit.aes =</span> inherit.aes,</span>
|
|||
|
<span id="cb16-22"><a href="#cb16-22" tabindex="-1"></a> <span class="at">params =</span> <span class="fu">list</span>(<span class="at">na.rm =</span> na.rm, ...)</span>
|
|||
|
<span id="cb16-23"><a href="#cb16-23" tabindex="-1"></a> )</span>
|
|||
|
<span id="cb16-24"><a href="#cb16-24" tabindex="-1"></a>}</span>
|
|||
|
<span id="cb16-25"><a href="#cb16-25" tabindex="-1"></a></span>
|
|||
|
<span id="cb16-26"><a href="#cb16-26" tabindex="-1"></a><span class="fu">ggplot</span>(mpg, <span class="fu">aes</span>(displ, hwy)) <span class="sc">+</span> </span>
|
|||
|
<span id="cb16-27"><a href="#cb16-27" tabindex="-1"></a> <span class="fu">geom_simple_point</span>()</span></code></pre></div>
|
|||
|
<p><img src="
|
|||
|
<p>This is very similar to defining a new stat. You always need to
|
|||
|
provide fields/methods for the four pieces shown above:</p>
|
|||
|
<ul>
|
|||
|
<li><p><code>required_aes</code> is a character vector which lists all
|
|||
|
the aesthetics that the user must provide.</p></li>
|
|||
|
<li><p><code>default_aes</code> lists the aesthetics that have default
|
|||
|
values.</p></li>
|
|||
|
<li><p><code>draw_key</code> provides the function used to draw the key
|
|||
|
in the legend. You can see a list of all the build in key functions in
|
|||
|
<code>?draw_key</code></p></li>
|
|||
|
<li><p><code>draw_panel()</code> is where the magic happens. This
|
|||
|
function takes three arguments and returns a grid grob. It is called
|
|||
|
once for each panel. It’s the most complicated part and is described in
|
|||
|
more detail below.</p></li>
|
|||
|
</ul>
|
|||
|
<p><code>draw_panel()</code> has three arguments:</p>
|
|||
|
<ul>
|
|||
|
<li><p><code>data</code>: a data frame with one column for each
|
|||
|
aesthetic.</p></li>
|
|||
|
<li><p><code>panel_params</code>: a list of per-panel parameters
|
|||
|
generated by the coord. You should consider this an opaque data
|
|||
|
structure: don’t look inside it, just pass along to <code>coord</code>
|
|||
|
methods.</p></li>
|
|||
|
<li><p><code>coord</code>: an object describing the coordinate
|
|||
|
system.</p></li>
|
|||
|
</ul>
|
|||
|
<p>You need to use <code>panel_params</code> and <code>coord</code>
|
|||
|
together to transform the data
|
|||
|
<code>coords <- coord$transform(data, panel_params)</code>. This
|
|||
|
creates a data frame where position variables are scaled to the range
|
|||
|
0–1. You then take this data and call a grid grob function.
|
|||
|
(Transforming for non-Cartesian coordinate systems is quite complex -
|
|||
|
you’re best off transforming your data to the form accepted by an
|
|||
|
existing ggplot2 geom and passing it.)</p>
|
|||
|
</div>
|
|||
|
<div id="collective-geoms" class="section level3">
|
|||
|
<h3>Collective geoms</h3>
|
|||
|
<p>Overriding <code>draw_panel()</code> is most appropriate if there is
|
|||
|
one graphic element per row. In other cases, you want graphic element
|
|||
|
per group. For example, take polygons: each row gives one vertex of a
|
|||
|
polygon. In this case, you should instead override
|
|||
|
<code>draw_group()</code>.</p>
|
|||
|
<p>The following code makes a simplified version of
|
|||
|
<code>GeomPolygon</code>:</p>
|
|||
|
<div class="sourceCode" id="cb17"><pre class="sourceCode r"><code class="sourceCode r"><span id="cb17-1"><a href="#cb17-1" tabindex="-1"></a>GeomSimplePolygon <span class="ot"><-</span> <span class="fu">ggproto</span>(<span class="st">"GeomPolygon"</span>, Geom,</span>
|
|||
|
<span id="cb17-2"><a href="#cb17-2" tabindex="-1"></a> <span class="at">required_aes =</span> <span class="fu">c</span>(<span class="st">"x"</span>, <span class="st">"y"</span>),</span>
|
|||
|
<span id="cb17-3"><a href="#cb17-3" tabindex="-1"></a> </span>
|
|||
|
<span id="cb17-4"><a href="#cb17-4" tabindex="-1"></a> <span class="at">default_aes =</span> <span class="fu">aes</span>(</span>
|
|||
|
<span id="cb17-5"><a href="#cb17-5" tabindex="-1"></a> <span class="at">colour =</span> <span class="cn">NA</span>, <span class="at">fill =</span> <span class="st">"grey20"</span>, <span class="at">linewidth =</span> <span class="fl">0.5</span>,</span>
|
|||
|
<span id="cb17-6"><a href="#cb17-6" tabindex="-1"></a> <span class="at">linetype =</span> <span class="dv">1</span>, <span class="at">alpha =</span> <span class="dv">1</span></span>
|
|||
|
<span id="cb17-7"><a href="#cb17-7" tabindex="-1"></a> ),</span>
|
|||
|
<span id="cb17-8"><a href="#cb17-8" tabindex="-1"></a></span>
|
|||
|
<span id="cb17-9"><a href="#cb17-9" tabindex="-1"></a> <span class="at">draw_key =</span> draw_key_polygon,</span>
|
|||
|
<span id="cb17-10"><a href="#cb17-10" tabindex="-1"></a></span>
|
|||
|
<span id="cb17-11"><a href="#cb17-11" tabindex="-1"></a> <span class="at">draw_group =</span> <span class="cf">function</span>(data, panel_params, coord) {</span>
|
|||
|
<span id="cb17-12"><a href="#cb17-12" tabindex="-1"></a> n <span class="ot"><-</span> <span class="fu">nrow</span>(data)</span>
|
|||
|
<span id="cb17-13"><a href="#cb17-13" tabindex="-1"></a> <span class="cf">if</span> (n <span class="sc"><=</span> <span class="dv">2</span>) <span class="fu">return</span>(grid<span class="sc">::</span><span class="fu">nullGrob</span>())</span>
|
|||
|
<span id="cb17-14"><a href="#cb17-14" tabindex="-1"></a></span>
|
|||
|
<span id="cb17-15"><a href="#cb17-15" tabindex="-1"></a> coords <span class="ot"><-</span> coord<span class="sc">$</span><span class="fu">transform</span>(data, panel_params)</span>
|
|||
|
<span id="cb17-16"><a href="#cb17-16" tabindex="-1"></a> <span class="co"># A polygon can only have a single colour, fill, etc, so take from first row</span></span>
|
|||
|
<span id="cb17-17"><a href="#cb17-17" tabindex="-1"></a> first_row <span class="ot"><-</span> coords[<span class="dv">1</span>, , drop <span class="ot">=</span> <span class="cn">FALSE</span>]</span>
|
|||
|
<span id="cb17-18"><a href="#cb17-18" tabindex="-1"></a></span>
|
|||
|
<span id="cb17-19"><a href="#cb17-19" tabindex="-1"></a> grid<span class="sc">::</span><span class="fu">polygonGrob</span>(</span>
|
|||
|
<span id="cb17-20"><a href="#cb17-20" tabindex="-1"></a> coords<span class="sc">$</span>x, coords<span class="sc">$</span>y, </span>
|
|||
|
<span id="cb17-21"><a href="#cb17-21" tabindex="-1"></a> <span class="at">default.units =</span> <span class="st">"native"</span>,</span>
|
|||
|
<span id="cb17-22"><a href="#cb17-22" tabindex="-1"></a> <span class="at">gp =</span> grid<span class="sc">::</span><span class="fu">gpar</span>(</span>
|
|||
|
<span id="cb17-23"><a href="#cb17-23" tabindex="-1"></a> <span class="at">col =</span> first_row<span class="sc">$</span>colour,</span>
|
|||
|
<span id="cb17-24"><a href="#cb17-24" tabindex="-1"></a> <span class="at">fill =</span> scales<span class="sc">::</span><span class="fu">alpha</span>(first_row<span class="sc">$</span>fill, first_row<span class="sc">$</span>alpha),</span>
|
|||
|
<span id="cb17-25"><a href="#cb17-25" tabindex="-1"></a> <span class="at">lwd =</span> first_row<span class="sc">$</span>linewidth <span class="sc">*</span> .pt,</span>
|
|||
|
<span id="cb17-26"><a href="#cb17-26" tabindex="-1"></a> <span class="at">lty =</span> first_row<span class="sc">$</span>linetype</span>
|
|||
|
<span id="cb17-27"><a href="#cb17-27" tabindex="-1"></a> )</span>
|
|||
|
<span id="cb17-28"><a href="#cb17-28" tabindex="-1"></a> )</span>
|
|||
|
<span id="cb17-29"><a href="#cb17-29" tabindex="-1"></a> }</span>
|
|||
|
<span id="cb17-30"><a href="#cb17-30" tabindex="-1"></a>)</span>
|
|||
|
<span id="cb17-31"><a href="#cb17-31" tabindex="-1"></a>geom_simple_polygon <span class="ot"><-</span> <span class="cf">function</span>(<span class="at">mapping =</span> <span class="cn">NULL</span>, <span class="at">data =</span> <span class="cn">NULL</span>, <span class="at">stat =</span> <span class="st">"chull"</span>,</span>
|
|||
|
<span id="cb17-32"><a href="#cb17-32" tabindex="-1"></a> <span class="at">position =</span> <span class="st">"identity"</span>, <span class="at">na.rm =</span> <span class="cn">FALSE</span>, <span class="at">show.legend =</span> <span class="cn">NA</span>, </span>
|
|||
|
<span id="cb17-33"><a href="#cb17-33" tabindex="-1"></a> <span class="at">inherit.aes =</span> <span class="cn">TRUE</span>, ...) {</span>
|
|||
|
<span id="cb17-34"><a href="#cb17-34" tabindex="-1"></a> <span class="fu">layer</span>(</span>
|
|||
|
<span id="cb17-35"><a href="#cb17-35" tabindex="-1"></a> <span class="at">geom =</span> GeomSimplePolygon, <span class="at">mapping =</span> mapping, <span class="at">data =</span> data, <span class="at">stat =</span> stat, </span>
|
|||
|
<span id="cb17-36"><a href="#cb17-36" tabindex="-1"></a> <span class="at">position =</span> position, <span class="at">show.legend =</span> show.legend, <span class="at">inherit.aes =</span> inherit.aes,</span>
|
|||
|
<span id="cb17-37"><a href="#cb17-37" tabindex="-1"></a> <span class="at">params =</span> <span class="fu">list</span>(<span class="at">na.rm =</span> na.rm, ...)</span>
|
|||
|
<span id="cb17-38"><a href="#cb17-38" tabindex="-1"></a> )</span>
|
|||
|
<span id="cb17-39"><a href="#cb17-39" tabindex="-1"></a>}</span>
|
|||
|
<span id="cb17-40"><a href="#cb17-40" tabindex="-1"></a></span>
|
|||
|
<span id="cb17-41"><a href="#cb17-41" tabindex="-1"></a><span class="fu">ggplot</span>(mpg, <span class="fu">aes</span>(displ, hwy)) <span class="sc">+</span> </span>
|
|||
|
<span id="cb17-42"><a href="#cb17-42" tabindex="-1"></a> <span class="fu">geom_point</span>() <span class="sc">+</span> </span>
|
|||
|
<span id="cb17-43"><a href="#cb17-43" tabindex="-1"></a> <span class="fu">geom_simple_polygon</span>(<span class="fu">aes</span>(<span class="at">colour =</span> class), <span class="at">fill =</span> <span class="cn">NA</span>)</span></code></pre></div>
|
|||
|
<p><img src="
|
|||
|
<p>There are a few things to note here:</p>
|
|||
|
<ul>
|
|||
|
<li><p>We override <code>draw_group()</code> instead of
|
|||
|
<code>draw_panel()</code> because we want one polygon per group, not one
|
|||
|
polygon per row.</p></li>
|
|||
|
<li><p>If the data contains two or fewer points, there’s no point trying
|
|||
|
to draw a polygon, so we return a <code>nullGrob()</code>. This is the
|
|||
|
graphical equivalent of <code>NULL</code>: it’s a grob that doesn’t draw
|
|||
|
anything and doesn’t take up any space.</p></li>
|
|||
|
<li><p>Note the units: <code>x</code> and <code>y</code> should always
|
|||
|
be drawn in “native” units. (The default units for
|
|||
|
<code>pointGrob()</code> is a native, so we didn’t need to change it
|
|||
|
there). <code>lwd</code> is measured in points, but ggplot2 uses mm, so
|
|||
|
we need to multiply it by the adjustment factor
|
|||
|
<code>.pt</code>.</p></li>
|
|||
|
</ul>
|
|||
|
<p>You might want to compare this to the real <code>GeomPolygon</code>.
|
|||
|
You’ll see it overrides <code>draw_panel()</code> because it uses some
|
|||
|
tricks to make <code>polygonGrob()</code> produce multiple polygons in
|
|||
|
one call. This is considerably more complicated, but gives better
|
|||
|
performance.</p>
|
|||
|
</div>
|
|||
|
<div id="inheriting-from-an-existing-geom" class="section level3">
|
|||
|
<h3>Inheriting from an existing Geom</h3>
|
|||
|
<p>Sometimes you just want to make a small modification to an existing
|
|||
|
geom. In this case, rather than inheriting from <code>Geom</code> you
|
|||
|
can inherit from an existing subclass. For example, we might want to
|
|||
|
change the defaults for <code>GeomPolygon</code> to work better with
|
|||
|
<code>StatChull</code>:</p>
|
|||
|
<div class="sourceCode" id="cb18"><pre class="sourceCode r"><code class="sourceCode r"><span id="cb18-1"><a href="#cb18-1" tabindex="-1"></a>GeomPolygonHollow <span class="ot"><-</span> <span class="fu">ggproto</span>(<span class="st">"GeomPolygonHollow"</span>, GeomPolygon,</span>
|
|||
|
<span id="cb18-2"><a href="#cb18-2" tabindex="-1"></a> <span class="at">default_aes =</span> <span class="fu">aes</span>(<span class="at">colour =</span> <span class="st">"black"</span>, <span class="at">fill =</span> <span class="cn">NA</span>, <span class="at">linewidth =</span> <span class="fl">0.5</span>, <span class="at">linetype =</span> <span class="dv">1</span>,</span>
|
|||
|
<span id="cb18-3"><a href="#cb18-3" tabindex="-1"></a> <span class="at">alpha =</span> <span class="cn">NA</span>)</span>
|
|||
|
<span id="cb18-4"><a href="#cb18-4" tabindex="-1"></a> )</span>
|
|||
|
<span id="cb18-5"><a href="#cb18-5" tabindex="-1"></a>geom_chull <span class="ot"><-</span> <span class="cf">function</span>(<span class="at">mapping =</span> <span class="cn">NULL</span>, <span class="at">data =</span> <span class="cn">NULL</span>, </span>
|
|||
|
<span id="cb18-6"><a href="#cb18-6" tabindex="-1"></a> <span class="at">position =</span> <span class="st">"identity"</span>, <span class="at">na.rm =</span> <span class="cn">FALSE</span>, <span class="at">show.legend =</span> <span class="cn">NA</span>, </span>
|
|||
|
<span id="cb18-7"><a href="#cb18-7" tabindex="-1"></a> <span class="at">inherit.aes =</span> <span class="cn">TRUE</span>, ...) {</span>
|
|||
|
<span id="cb18-8"><a href="#cb18-8" tabindex="-1"></a> <span class="fu">layer</span>(</span>
|
|||
|
<span id="cb18-9"><a href="#cb18-9" tabindex="-1"></a> <span class="at">stat =</span> StatChull, <span class="at">geom =</span> GeomPolygonHollow, <span class="at">data =</span> data, <span class="at">mapping =</span> mapping,</span>
|
|||
|
<span id="cb18-10"><a href="#cb18-10" tabindex="-1"></a> <span class="at">position =</span> position, <span class="at">show.legend =</span> show.legend, <span class="at">inherit.aes =</span> inherit.aes,</span>
|
|||
|
<span id="cb18-11"><a href="#cb18-11" tabindex="-1"></a> <span class="at">params =</span> <span class="fu">list</span>(<span class="at">na.rm =</span> na.rm, ...)</span>
|
|||
|
<span id="cb18-12"><a href="#cb18-12" tabindex="-1"></a> )</span>
|
|||
|
<span id="cb18-13"><a href="#cb18-13" tabindex="-1"></a>}</span>
|
|||
|
<span id="cb18-14"><a href="#cb18-14" tabindex="-1"></a></span>
|
|||
|
<span id="cb18-15"><a href="#cb18-15" tabindex="-1"></a><span class="fu">ggplot</span>(mpg, <span class="fu">aes</span>(displ, hwy)) <span class="sc">+</span> </span>
|
|||
|
<span id="cb18-16"><a href="#cb18-16" tabindex="-1"></a> <span class="fu">geom_point</span>() <span class="sc">+</span> </span>
|
|||
|
<span id="cb18-17"><a href="#cb18-17" tabindex="-1"></a> <span class="fu">geom_chull</span>()</span></code></pre></div>
|
|||
|
<p><img src="
|
|||
|
<p>This doesn’t allow you to use different geoms with the stat, but that
|
|||
|
seems appropriate here since the convex hull is primarily a polygonal
|
|||
|
feature.</p>
|
|||
|
</div>
|
|||
|
<div id="exercises-1" class="section level3">
|
|||
|
<h3>Exercises</h3>
|
|||
|
<ol style="list-style-type: decimal">
|
|||
|
<li><p>Compare and contrast <code>GeomPoint</code> with
|
|||
|
<code>GeomSimplePoint</code>.</p></li>
|
|||
|
<li><p>Compare and contrast <code>GeomPolygon</code> with
|
|||
|
<code>GeomSimplePolygon</code>.</p></li>
|
|||
|
</ol>
|
|||
|
</div>
|
|||
|
</div>
|
|||
|
<div id="geoms-and-stats-with-multiple-orientation" class="section level2">
|
|||
|
<h2>Geoms and Stats with multiple orientation</h2>
|
|||
|
<p>Some layers have a specific orientation. <code>geom_bar()</code>
|
|||
|
e.g. have the bars along one axis, <code>geom_line()</code> will sort
|
|||
|
the input by one axis, etc. The original approach to using these geoms
|
|||
|
in the other orientation was to add <code>coord_flip()</code> to the
|
|||
|
plot to switch the position of the x and y axes. Following ggplot2 v3.3
|
|||
|
all the geoms will natively work in both orientations without
|
|||
|
<code>coord_flip()</code>. The mechanism is that the layer will try to
|
|||
|
guess the orientation from the mapped data, or take direction from the
|
|||
|
user using the <code>orientation</code> parameter. To replicate this
|
|||
|
functionality in new stats and geoms there’s a few steps to take. We
|
|||
|
will look at the boxplot layer as an example instead of creating a new
|
|||
|
one from scratch.</p>
|
|||
|
<div id="omnidirectional-stats" class="section level3">
|
|||
|
<h3>Omnidirectional stats</h3>
|
|||
|
<p>The actual guessing of orientation will happen in
|
|||
|
<code>setup_params()</code> using the <code>has_flipped_aes()</code>
|
|||
|
helper:</p>
|
|||
|
<div class="sourceCode" id="cb19"><pre class="sourceCode r"><code class="sourceCode r"><span id="cb19-1"><a href="#cb19-1" tabindex="-1"></a>StatBoxplot<span class="sc">$</span>setup_params</span>
|
|||
|
<span id="cb19-2"><a href="#cb19-2" tabindex="-1"></a><span class="co">#> <ggproto method></span></span>
|
|||
|
<span id="cb19-3"><a href="#cb19-3" tabindex="-1"></a><span class="co">#> <Wrapper function></span></span>
|
|||
|
<span id="cb19-4"><a href="#cb19-4" tabindex="-1"></a><span class="co">#> function (...) </span></span>
|
|||
|
<span id="cb19-5"><a href="#cb19-5" tabindex="-1"></a><span class="co">#> setup_params(..., self = self)</span></span>
|
|||
|
<span id="cb19-6"><a href="#cb19-6" tabindex="-1"></a><span class="co">#> </span></span>
|
|||
|
<span id="cb19-7"><a href="#cb19-7" tabindex="-1"></a><span class="co">#> <Inner function (f)></span></span>
|
|||
|
<span id="cb19-8"><a href="#cb19-8" tabindex="-1"></a><span class="co">#> function (self, data, params) </span></span>
|
|||
|
<span id="cb19-9"><a href="#cb19-9" tabindex="-1"></a><span class="co">#> {</span></span>
|
|||
|
<span id="cb19-10"><a href="#cb19-10" tabindex="-1"></a><span class="co">#> params$flipped_aes <- has_flipped_aes(data, params, main_is_orthogonal = TRUE, </span></span>
|
|||
|
<span id="cb19-11"><a href="#cb19-11" tabindex="-1"></a><span class="co">#> group_has_equal = TRUE, main_is_optional = TRUE)</span></span>
|
|||
|
<span id="cb19-12"><a href="#cb19-12" tabindex="-1"></a><span class="co">#> data <- flip_data(data, params$flipped_aes)</span></span>
|
|||
|
<span id="cb19-13"><a href="#cb19-13" tabindex="-1"></a><span class="co">#> has_x <- !(is.null(data$x) && is.null(params$x))</span></span>
|
|||
|
<span id="cb19-14"><a href="#cb19-14" tabindex="-1"></a><span class="co">#> has_y <- !(is.null(data$y) && is.null(params$y))</span></span>
|
|||
|
<span id="cb19-15"><a href="#cb19-15" tabindex="-1"></a><span class="co">#> if (!has_x && !has_y) {</span></span>
|
|||
|
<span id="cb19-16"><a href="#cb19-16" tabindex="-1"></a><span class="co">#> cli::cli_abort("{.fn {snake_class(self)}} requires an {.field x} or {.field y} aesthetic.")</span></span>
|
|||
|
<span id="cb19-17"><a href="#cb19-17" tabindex="-1"></a><span class="co">#> }</span></span>
|
|||
|
<span id="cb19-18"><a href="#cb19-18" tabindex="-1"></a><span class="co">#> params$width <- params$width %||% (resolution(data$x %||% </span></span>
|
|||
|
<span id="cb19-19"><a href="#cb19-19" tabindex="-1"></a><span class="co">#> 0, discrete = TRUE) * 0.75)</span></span>
|
|||
|
<span id="cb19-20"><a href="#cb19-20" tabindex="-1"></a><span class="co">#> if (!is_mapped_discrete(data$x) && is.double(data$x) && !has_groups(data) && </span></span>
|
|||
|
<span id="cb19-21"><a href="#cb19-21" tabindex="-1"></a><span class="co">#> any(data$x != data$x[1L])) {</span></span>
|
|||
|
<span id="cb19-22"><a href="#cb19-22" tabindex="-1"></a><span class="co">#> cli::cli_warn(c("Continuous {.field {flipped_names(params$flipped_aes)$x}} aesthetic", </span></span>
|
|||
|
<span id="cb19-23"><a href="#cb19-23" tabindex="-1"></a><span class="co">#> i = "did you forget {.code aes(group = ...)}?"))</span></span>
|
|||
|
<span id="cb19-24"><a href="#cb19-24" tabindex="-1"></a><span class="co">#> }</span></span>
|
|||
|
<span id="cb19-25"><a href="#cb19-25" tabindex="-1"></a><span class="co">#> params</span></span>
|
|||
|
<span id="cb19-26"><a href="#cb19-26" tabindex="-1"></a><span class="co">#> }</span></span></code></pre></div>
|
|||
|
<p>Following this is a call to <code>flip_data()</code> which will make
|
|||
|
sure the data is in horizontal orientation. The rest of the code can
|
|||
|
then simply assume that the data is in a specific orientation. The same
|
|||
|
thing happens in <code>setup_data()</code>:</p>
|
|||
|
<div class="sourceCode" id="cb20"><pre class="sourceCode r"><code class="sourceCode r"><span id="cb20-1"><a href="#cb20-1" tabindex="-1"></a>StatBoxplot<span class="sc">$</span>setup_data</span>
|
|||
|
<span id="cb20-2"><a href="#cb20-2" tabindex="-1"></a><span class="co">#> <ggproto method></span></span>
|
|||
|
<span id="cb20-3"><a href="#cb20-3" tabindex="-1"></a><span class="co">#> <Wrapper function></span></span>
|
|||
|
<span id="cb20-4"><a href="#cb20-4" tabindex="-1"></a><span class="co">#> function (...) </span></span>
|
|||
|
<span id="cb20-5"><a href="#cb20-5" tabindex="-1"></a><span class="co">#> setup_data(..., self = self)</span></span>
|
|||
|
<span id="cb20-6"><a href="#cb20-6" tabindex="-1"></a><span class="co">#> </span></span>
|
|||
|
<span id="cb20-7"><a href="#cb20-7" tabindex="-1"></a><span class="co">#> <Inner function (f)></span></span>
|
|||
|
<span id="cb20-8"><a href="#cb20-8" tabindex="-1"></a><span class="co">#> function (self, data, params) </span></span>
|
|||
|
<span id="cb20-9"><a href="#cb20-9" tabindex="-1"></a><span class="co">#> {</span></span>
|
|||
|
<span id="cb20-10"><a href="#cb20-10" tabindex="-1"></a><span class="co">#> data <- flip_data(data, params$flipped_aes)</span></span>
|
|||
|
<span id="cb20-11"><a href="#cb20-11" tabindex="-1"></a><span class="co">#> data$x <- data$x %||% 0</span></span>
|
|||
|
<span id="cb20-12"><a href="#cb20-12" tabindex="-1"></a><span class="co">#> data <- remove_missing(data, na.rm = params$na.rm, vars = "x", </span></span>
|
|||
|
<span id="cb20-13"><a href="#cb20-13" tabindex="-1"></a><span class="co">#> name = "stat_boxplot")</span></span>
|
|||
|
<span id="cb20-14"><a href="#cb20-14" tabindex="-1"></a><span class="co">#> flip_data(data, params$flipped_aes)</span></span>
|
|||
|
<span id="cb20-15"><a href="#cb20-15" tabindex="-1"></a><span class="co">#> }</span></span></code></pre></div>
|
|||
|
<p>The data is flipped (if needed), manipulated, and flipped back as it
|
|||
|
is returned.</p>
|
|||
|
<p>During the computation, this sandwiching between
|
|||
|
<code>flip_data()</code> is used as well, but right before the data is
|
|||
|
returned it will also get a <code>flipped_aes</code> column denoting if
|
|||
|
the data is flipped or not. This allows the stat to communicate to the
|
|||
|
geom that orientation has already been determined.</p>
|
|||
|
</div>
|
|||
|
<div id="omnidirecitonal-geoms" class="section level3">
|
|||
|
<h3>Omnidirecitonal geoms</h3>
|
|||
|
<p>The setup for geoms is pretty much the same, with a few twists.
|
|||
|
<code>has_flipped_aes()</code> is also used in
|
|||
|
<code>setup_params()</code>, where it will usually be picked up from the
|
|||
|
<code>flipped_aes</code> column given by the stat. In
|
|||
|
<code>setup_data()</code> you will often see that
|
|||
|
<code>flipped_aes</code> is reassigned, to make sure it exist prior to
|
|||
|
position adjustment. This is needed if the geom is used together with a
|
|||
|
stat that doesn’t handle orientation (often
|
|||
|
<code>stat_identity()</code>):</p>
|
|||
|
<div class="sourceCode" id="cb21"><pre class="sourceCode r"><code class="sourceCode r"><span id="cb21-1"><a href="#cb21-1" tabindex="-1"></a>GeomBoxplot<span class="sc">$</span>setup_data</span>
|
|||
|
<span id="cb21-2"><a href="#cb21-2" tabindex="-1"></a><span class="co">#> <ggproto method></span></span>
|
|||
|
<span id="cb21-3"><a href="#cb21-3" tabindex="-1"></a><span class="co">#> <Wrapper function></span></span>
|
|||
|
<span id="cb21-4"><a href="#cb21-4" tabindex="-1"></a><span class="co">#> function (...) </span></span>
|
|||
|
<span id="cb21-5"><a href="#cb21-5" tabindex="-1"></a><span class="co">#> setup_data(...)</span></span>
|
|||
|
<span id="cb21-6"><a href="#cb21-6" tabindex="-1"></a><span class="co">#> </span></span>
|
|||
|
<span id="cb21-7"><a href="#cb21-7" tabindex="-1"></a><span class="co">#> <Inner function (f)></span></span>
|
|||
|
<span id="cb21-8"><a href="#cb21-8" tabindex="-1"></a><span class="co">#> function (data, params) </span></span>
|
|||
|
<span id="cb21-9"><a href="#cb21-9" tabindex="-1"></a><span class="co">#> {</span></span>
|
|||
|
<span id="cb21-10"><a href="#cb21-10" tabindex="-1"></a><span class="co">#> data$flipped_aes <- params$flipped_aes</span></span>
|
|||
|
<span id="cb21-11"><a href="#cb21-11" tabindex="-1"></a><span class="co">#> data <- flip_data(data, params$flipped_aes)</span></span>
|
|||
|
<span id="cb21-12"><a href="#cb21-12" tabindex="-1"></a><span class="co">#> data$width <- data$width %||% params$width %||% (resolution(data$x, </span></span>
|
|||
|
<span id="cb21-13"><a href="#cb21-13" tabindex="-1"></a><span class="co">#> FALSE, TRUE) * 0.9)</span></span>
|
|||
|
<span id="cb21-14"><a href="#cb21-14" tabindex="-1"></a><span class="co">#> if (isFALSE(params$outliers)) {</span></span>
|
|||
|
<span id="cb21-15"><a href="#cb21-15" tabindex="-1"></a><span class="co">#> data$outliers <- NULL</span></span>
|
|||
|
<span id="cb21-16"><a href="#cb21-16" tabindex="-1"></a><span class="co">#> }</span></span>
|
|||
|
<span id="cb21-17"><a href="#cb21-17" tabindex="-1"></a><span class="co">#> if (!is.null(data$outliers)) {</span></span>
|
|||
|
<span id="cb21-18"><a href="#cb21-18" tabindex="-1"></a><span class="co">#> suppressWarnings({</span></span>
|
|||
|
<span id="cb21-19"><a href="#cb21-19" tabindex="-1"></a><span class="co">#> out_min <- vapply(data$outliers, min, numeric(1))</span></span>
|
|||
|
<span id="cb21-20"><a href="#cb21-20" tabindex="-1"></a><span class="co">#> out_max <- vapply(data$outliers, max, numeric(1))</span></span>
|
|||
|
<span id="cb21-21"><a href="#cb21-21" tabindex="-1"></a><span class="co">#> })</span></span>
|
|||
|
<span id="cb21-22"><a href="#cb21-22" tabindex="-1"></a><span class="co">#> data$ymin_final <- pmin(out_min, data$ymin)</span></span>
|
|||
|
<span id="cb21-23"><a href="#cb21-23" tabindex="-1"></a><span class="co">#> data$ymax_final <- pmax(out_max, data$ymax)</span></span>
|
|||
|
<span id="cb21-24"><a href="#cb21-24" tabindex="-1"></a><span class="co">#> }</span></span>
|
|||
|
<span id="cb21-25"><a href="#cb21-25" tabindex="-1"></a><span class="co">#> if (is.null(params) || is.null(params$varwidth) || !params$varwidth || </span></span>
|
|||
|
<span id="cb21-26"><a href="#cb21-26" tabindex="-1"></a><span class="co">#> is.null(data$relvarwidth)) {</span></span>
|
|||
|
<span id="cb21-27"><a href="#cb21-27" tabindex="-1"></a><span class="co">#> data$xmin <- data$x - data$width/2</span></span>
|
|||
|
<span id="cb21-28"><a href="#cb21-28" tabindex="-1"></a><span class="co">#> data$xmax <- data$x + data$width/2</span></span>
|
|||
|
<span id="cb21-29"><a href="#cb21-29" tabindex="-1"></a><span class="co">#> }</span></span>
|
|||
|
<span id="cb21-30"><a href="#cb21-30" tabindex="-1"></a><span class="co">#> else {</span></span>
|
|||
|
<span id="cb21-31"><a href="#cb21-31" tabindex="-1"></a><span class="co">#> data$relvarwidth <- data$relvarwidth/max(data$relvarwidth)</span></span>
|
|||
|
<span id="cb21-32"><a href="#cb21-32" tabindex="-1"></a><span class="co">#> data$xmin <- data$x - data$relvarwidth * data$width/2</span></span>
|
|||
|
<span id="cb21-33"><a href="#cb21-33" tabindex="-1"></a><span class="co">#> data$xmax <- data$x + data$relvarwidth * data$width/2</span></span>
|
|||
|
<span id="cb21-34"><a href="#cb21-34" tabindex="-1"></a><span class="co">#> }</span></span>
|
|||
|
<span id="cb21-35"><a href="#cb21-35" tabindex="-1"></a><span class="co">#> data$width <- NULL</span></span>
|
|||
|
<span id="cb21-36"><a href="#cb21-36" tabindex="-1"></a><span class="co">#> if (!is.null(data$relvarwidth)) </span></span>
|
|||
|
<span id="cb21-37"><a href="#cb21-37" tabindex="-1"></a><span class="co">#> data$relvarwidth <- NULL</span></span>
|
|||
|
<span id="cb21-38"><a href="#cb21-38" tabindex="-1"></a><span class="co">#> flip_data(data, params$flipped_aes)</span></span>
|
|||
|
<span id="cb21-39"><a href="#cb21-39" tabindex="-1"></a><span class="co">#> }</span></span></code></pre></div>
|
|||
|
<p>In the <code>draw_*()</code> method you will once again sandwich any
|
|||
|
data manipulation between <code>flip_data()</code> calls. It is
|
|||
|
important to make sure that the data is flipped back prior to creating
|
|||
|
the grob or calling draw methods from other geoms.</p>
|
|||
|
</div>
|
|||
|
<div id="dealing-with-required-aesthetics" class="section level3">
|
|||
|
<h3>Dealing with required aesthetics</h3>
|
|||
|
<p>Omnidirectional layers usually have two different sets of required
|
|||
|
aesthetics. Which set is used is often how it knows the orientation. To
|
|||
|
handle this gracefully the <code>required_aes</code> field of
|
|||
|
<code>Stat</code> and <code>Geom</code> classes understands the
|
|||
|
<code>|</code> (or) operator. Looking at <code>GeomBoxplot</code> we can
|
|||
|
see how it is used:</p>
|
|||
|
<div class="sourceCode" id="cb22"><pre class="sourceCode r"><code class="sourceCode r"><span id="cb22-1"><a href="#cb22-1" tabindex="-1"></a>GeomBoxplot<span class="sc">$</span>required_aes</span>
|
|||
|
<span id="cb22-2"><a href="#cb22-2" tabindex="-1"></a><span class="co">#> [1] "x|y" "lower|xlower" "upper|xupper" "middle|xmiddle"</span></span>
|
|||
|
<span id="cb22-3"><a href="#cb22-3" tabindex="-1"></a><span class="co">#> [5] "ymin|xmin" "ymax|xmax"</span></span></code></pre></div>
|
|||
|
<p>This tells ggplot2 that either all the aesthetics before
|
|||
|
<code>|</code> are required or all the aesthetics after are
|
|||
|
required.</p>
|
|||
|
</div>
|
|||
|
<div id="ambiguous-layers" class="section level3">
|
|||
|
<h3>Ambiguous layers</h3>
|
|||
|
<p>Some layers will not have a clear interpretation of their data in
|
|||
|
terms of orientation. A classic example is <code>geom_line()</code>
|
|||
|
which just by convention runs along the x-axis. There is nothing in the
|
|||
|
data itself that indicates that. For these geoms the user must indicate
|
|||
|
a flipped orientation by setting <code>orientation = "y"</code>. The
|
|||
|
stat or geom will then call <code>has_flipped_aes()</code> with
|
|||
|
<code>ambiguous = TRUE</code> to cancel any guessing based on data
|
|||
|
format. As an example we can see the <code>setup_params()</code> method
|
|||
|
of <code>GeomLine</code>:</p>
|
|||
|
<div class="sourceCode" id="cb23"><pre class="sourceCode r"><code class="sourceCode r"><span id="cb23-1"><a href="#cb23-1" tabindex="-1"></a>GeomLine<span class="sc">$</span>setup_params</span>
|
|||
|
<span id="cb23-2"><a href="#cb23-2" tabindex="-1"></a><span class="co">#> <ggproto method></span></span>
|
|||
|
<span id="cb23-3"><a href="#cb23-3" tabindex="-1"></a><span class="co">#> <Wrapper function></span></span>
|
|||
|
<span id="cb23-4"><a href="#cb23-4" tabindex="-1"></a><span class="co">#> function (...) </span></span>
|
|||
|
<span id="cb23-5"><a href="#cb23-5" tabindex="-1"></a><span class="co">#> setup_params(...)</span></span>
|
|||
|
<span id="cb23-6"><a href="#cb23-6" tabindex="-1"></a><span class="co">#> </span></span>
|
|||
|
<span id="cb23-7"><a href="#cb23-7" tabindex="-1"></a><span class="co">#> <Inner function (f)></span></span>
|
|||
|
<span id="cb23-8"><a href="#cb23-8" tabindex="-1"></a><span class="co">#> function (data, params) </span></span>
|
|||
|
<span id="cb23-9"><a href="#cb23-9" tabindex="-1"></a><span class="co">#> {</span></span>
|
|||
|
<span id="cb23-10"><a href="#cb23-10" tabindex="-1"></a><span class="co">#> params$flipped_aes <- has_flipped_aes(data, params, ambiguous = TRUE)</span></span>
|
|||
|
<span id="cb23-11"><a href="#cb23-11" tabindex="-1"></a><span class="co">#> params</span></span>
|
|||
|
<span id="cb23-12"><a href="#cb23-12" tabindex="-1"></a><span class="co">#> }</span></span></code></pre></div>
|
|||
|
</div>
|
|||
|
</div>
|
|||
|
<div id="creating-your-own-theme" class="section level2">
|
|||
|
<h2>Creating your own theme</h2>
|
|||
|
<p>If you’re going to create your own complete theme, there are a few
|
|||
|
things you need to know:</p>
|
|||
|
<ul>
|
|||
|
<li>Overriding existing elements, rather than modifying them</li>
|
|||
|
<li>The four global elements that affect (almost) every other theme
|
|||
|
element</li>
|
|||
|
<li>Complete vs. incomplete elements</li>
|
|||
|
</ul>
|
|||
|
<div id="overriding-elements" class="section level3">
|
|||
|
<h3>Overriding elements</h3>
|
|||
|
<p>By default, when you add a new theme element, it inherits values from
|
|||
|
the existing theme. For example, the following code sets the key colour
|
|||
|
to red, but it inherits the existing fill colour:</p>
|
|||
|
<div class="sourceCode" id="cb24"><pre class="sourceCode r"><code class="sourceCode r"><span id="cb24-1"><a href="#cb24-1" tabindex="-1"></a><span class="fu">theme_grey</span>()<span class="sc">$</span>legend.key</span>
|
|||
|
<span id="cb24-2"><a href="#cb24-2" tabindex="-1"></a><span class="co">#> NULL</span></span>
|
|||
|
<span id="cb24-3"><a href="#cb24-3" tabindex="-1"></a></span>
|
|||
|
<span id="cb24-4"><a href="#cb24-4" tabindex="-1"></a>new_theme <span class="ot"><-</span> <span class="fu">theme_grey</span>() <span class="sc">+</span> <span class="fu">theme</span>(<span class="at">legend.key =</span> <span class="fu">element_rect</span>(<span class="at">colour =</span> <span class="st">"red"</span>))</span>
|
|||
|
<span id="cb24-5"><a href="#cb24-5" tabindex="-1"></a>new_theme<span class="sc">$</span>legend.key</span>
|
|||
|
<span id="cb24-6"><a href="#cb24-6" tabindex="-1"></a><span class="co">#> List of 5</span></span>
|
|||
|
<span id="cb24-7"><a href="#cb24-7" tabindex="-1"></a><span class="co">#> $ fill : NULL</span></span>
|
|||
|
<span id="cb24-8"><a href="#cb24-8" tabindex="-1"></a><span class="co">#> $ colour : chr "red"</span></span>
|
|||
|
<span id="cb24-9"><a href="#cb24-9" tabindex="-1"></a><span class="co">#> $ linewidth : NULL</span></span>
|
|||
|
<span id="cb24-10"><a href="#cb24-10" tabindex="-1"></a><span class="co">#> $ linetype : NULL</span></span>
|
|||
|
<span id="cb24-11"><a href="#cb24-11" tabindex="-1"></a><span class="co">#> $ inherit.blank: logi FALSE</span></span>
|
|||
|
<span id="cb24-12"><a href="#cb24-12" tabindex="-1"></a><span class="co">#> - attr(*, "class")= chr [1:2] "element_rect" "element"</span></span></code></pre></div>
|
|||
|
<p>To override it completely, use <code>%+replace%</code> instead of
|
|||
|
<code>+</code>:</p>
|
|||
|
<div class="sourceCode" id="cb25"><pre class="sourceCode r"><code class="sourceCode r"><span id="cb25-1"><a href="#cb25-1" tabindex="-1"></a>new_theme <span class="ot"><-</span> <span class="fu">theme_grey</span>() <span class="sc">%+replace%</span> <span class="fu">theme</span>(<span class="at">legend.key =</span> <span class="fu">element_rect</span>(<span class="at">colour =</span> <span class="st">"red"</span>))</span>
|
|||
|
<span id="cb25-2"><a href="#cb25-2" tabindex="-1"></a>new_theme<span class="sc">$</span>legend.key</span>
|
|||
|
<span id="cb25-3"><a href="#cb25-3" tabindex="-1"></a><span class="co">#> List of 5</span></span>
|
|||
|
<span id="cb25-4"><a href="#cb25-4" tabindex="-1"></a><span class="co">#> $ fill : NULL</span></span>
|
|||
|
<span id="cb25-5"><a href="#cb25-5" tabindex="-1"></a><span class="co">#> $ colour : chr "red"</span></span>
|
|||
|
<span id="cb25-6"><a href="#cb25-6" tabindex="-1"></a><span class="co">#> $ linewidth : NULL</span></span>
|
|||
|
<span id="cb25-7"><a href="#cb25-7" tabindex="-1"></a><span class="co">#> $ linetype : NULL</span></span>
|
|||
|
<span id="cb25-8"><a href="#cb25-8" tabindex="-1"></a><span class="co">#> $ inherit.blank: logi FALSE</span></span>
|
|||
|
<span id="cb25-9"><a href="#cb25-9" tabindex="-1"></a><span class="co">#> - attr(*, "class")= chr [1:2] "element_rect" "element"</span></span></code></pre></div>
|
|||
|
</div>
|
|||
|
<div id="global-elements" class="section level3">
|
|||
|
<h3>Global elements</h3>
|
|||
|
<p>There are four elements that affect the global appearance of the
|
|||
|
plot:</p>
|
|||
|
<table>
|
|||
|
<colgroup>
|
|||
|
<col width="23%" />
|
|||
|
<col width="33%" />
|
|||
|
<col width="42%" />
|
|||
|
</colgroup>
|
|||
|
<thead>
|
|||
|
<tr class="header">
|
|||
|
<th>Element</th>
|
|||
|
<th>Theme function</th>
|
|||
|
<th>Description</th>
|
|||
|
</tr>
|
|||
|
</thead>
|
|||
|
<tbody>
|
|||
|
<tr class="odd">
|
|||
|
<td>line</td>
|
|||
|
<td><code>element_line()</code></td>
|
|||
|
<td>all line elements</td>
|
|||
|
</tr>
|
|||
|
<tr class="even">
|
|||
|
<td>rect</td>
|
|||
|
<td><code>element_rect()</code></td>
|
|||
|
<td>all rectangular elements</td>
|
|||
|
</tr>
|
|||
|
<tr class="odd">
|
|||
|
<td>text</td>
|
|||
|
<td><code>element_text()</code></td>
|
|||
|
<td>all text</td>
|
|||
|
</tr>
|
|||
|
<tr class="even">
|
|||
|
<td>title</td>
|
|||
|
<td><code>element_text()</code></td>
|
|||
|
<td>all text in title elements (plot, axes & legend)</td>
|
|||
|
</tr>
|
|||
|
</tbody>
|
|||
|
</table>
|
|||
|
<p>These set default properties that are inherited by more specific
|
|||
|
settings. These are most useful for setting an overall “background”
|
|||
|
colour and overall font settings (e.g. family and size).</p>
|
|||
|
<div class="sourceCode" id="cb26"><pre class="sourceCode r"><code class="sourceCode r"><span id="cb26-1"><a href="#cb26-1" tabindex="-1"></a>df <span class="ot"><-</span> <span class="fu">data.frame</span>(<span class="at">x =</span> <span class="dv">1</span><span class="sc">:</span><span class="dv">3</span>, <span class="at">y =</span> <span class="dv">1</span><span class="sc">:</span><span class="dv">3</span>)</span>
|
|||
|
<span id="cb26-2"><a href="#cb26-2" tabindex="-1"></a>base <span class="ot"><-</span> <span class="fu">ggplot</span>(df, <span class="fu">aes</span>(x, y)) <span class="sc">+</span> </span>
|
|||
|
<span id="cb26-3"><a href="#cb26-3" tabindex="-1"></a> <span class="fu">geom_point</span>() <span class="sc">+</span> </span>
|
|||
|
<span id="cb26-4"><a href="#cb26-4" tabindex="-1"></a> <span class="fu">theme_minimal</span>()</span>
|
|||
|
<span id="cb26-5"><a href="#cb26-5" tabindex="-1"></a></span>
|
|||
|
<span id="cb26-6"><a href="#cb26-6" tabindex="-1"></a>base</span></code></pre></div>
|
|||
|
<p><img src="
|
|||
|
<div class="sourceCode" id="cb27"><pre class="sourceCode r"><code class="sourceCode r"><span id="cb27-1"><a href="#cb27-1" tabindex="-1"></a>base <span class="sc">+</span> <span class="fu">theme</span>(<span class="at">text =</span> <span class="fu">element_text</span>(<span class="at">colour =</span> <span class="st">"red"</span>))</span></code></pre></div>
|
|||
|
<p><img src="
|
|||
|
<p>You should generally start creating a theme by modifying these
|
|||
|
values.</p>
|
|||
|
</div>
|
|||
|
<div id="complete-vs-incomplete" class="section level3">
|
|||
|
<h3>Complete vs incomplete</h3>
|
|||
|
<p>It is useful to understand the difference between complete and
|
|||
|
incomplete theme objects. A <em>complete</em> theme object is one
|
|||
|
produced by calling a theme function with the attribute
|
|||
|
<code>complete = TRUE</code>.</p>
|
|||
|
<p>Theme functions <code>theme_grey()</code> and <code>theme_bw()</code>
|
|||
|
are examples of complete theme functions. Calls to <code>theme()</code>
|
|||
|
produce <em>incomplete</em> theme objects, since they represent (local)
|
|||
|
modifications to a theme object rather than returning a complete theme
|
|||
|
object per se. When adding an incomplete theme to a complete one, the
|
|||
|
result is a complete theme.</p>
|
|||
|
<p>Complete and incomplete themes behave somewhat differently when added
|
|||
|
to a ggplot object:</p>
|
|||
|
<ul>
|
|||
|
<li><p>Adding an incomplete theme augments the current theme object,
|
|||
|
replacing only those properties of elements defined in the call to
|
|||
|
<code>theme()</code>.</p></li>
|
|||
|
<li><p>Adding a complete theme wipes away the existing theme and applies
|
|||
|
the new theme.</p></li>
|
|||
|
</ul>
|
|||
|
</div>
|
|||
|
</div>
|
|||
|
<div id="creating-a-new-faceting" class="section level2">
|
|||
|
<h2>Creating a new faceting</h2>
|
|||
|
<p>One of the more daunting exercises in ggplot2 extensions is to create
|
|||
|
a new faceting system. The reason for this is that when creating new
|
|||
|
facetings you take on the responsibility of how (almost) everything is
|
|||
|
drawn on the screen, and many do not have experience with directly using
|
|||
|
<a href="https://cran.r-project.org/package=gtable">gtable</a> and <a href="https://cran.r-project.org/package=grid">grid</a> upon which the
|
|||
|
ggplot2 rendering is built. If you decide to venture into faceting
|
|||
|
extensions, it is highly recommended to gain proficiency with the
|
|||
|
above-mentioned packages.</p>
|
|||
|
<p>The <code>Facet</code> class in ggplot2 is very powerful as it takes
|
|||
|
on responsibility of a wide range of tasks. The main tasks of a
|
|||
|
<code>Facet</code> object are:</p>
|
|||
|
<ul>
|
|||
|
<li><p>Define a layout; that is, a partitioning of the data into
|
|||
|
different plot areas (panels) as well as which panels share position
|
|||
|
scales.</p></li>
|
|||
|
<li><p>Map plot data into the correct panels, potentially duplicating
|
|||
|
data if it should exist in multiple panels (e.g. margins in
|
|||
|
<code>facet_grid()</code>).</p></li>
|
|||
|
<li><p>Assemble all panels into a final gtable, adding axes, strips and
|
|||
|
decorations in the process.</p></li>
|
|||
|
</ul>
|
|||
|
<p>Apart from these three tasks, for which functionality must be
|
|||
|
implemented, there are a couple of additional extension points where
|
|||
|
sensible defaults have been provided. These can generally be ignored,
|
|||
|
but adventurous developers can override them for even more control:</p>
|
|||
|
<ul>
|
|||
|
<li><p>Initialization and training of positional scales for each
|
|||
|
panel.</p></li>
|
|||
|
<li><p>Decoration in front of and behind each panel.</p></li>
|
|||
|
<li><p>Drawing of axis labels</p></li>
|
|||
|
</ul>
|
|||
|
<p>To show how a new faceting class is created we will start simple and
|
|||
|
go through each of the required methods in turn to build up
|
|||
|
<code>facet_duplicate()</code> that simply duplicate our plot into two
|
|||
|
panels. After this we will tinker with it a bit to show some of the more
|
|||
|
powerful possibilities.</p>
|
|||
|
<div id="creating-a-layout-specification" class="section level3">
|
|||
|
<h3>Creating a layout specification</h3>
|
|||
|
<p>A layout in the context of facets is a <code>data.frame</code> that
|
|||
|
defines a mapping between data and the panels it should reside in as
|
|||
|
well as which positional scales should be used. The output should at
|
|||
|
least contain the columns <code>PANEL</code>, <code>SCALE_X</code>, and
|
|||
|
<code>SCALE_Y</code>, but will often contain more to help assign data to
|
|||
|
the correct panel (<code>facet_grid()</code> will e.g. also return the
|
|||
|
faceting variables associated with each panel). Let’s make a function
|
|||
|
that defines a duplicate layout:</p>
|
|||
|
<div class="sourceCode" id="cb28"><pre class="sourceCode r"><code class="sourceCode r"><span id="cb28-1"><a href="#cb28-1" tabindex="-1"></a>layout <span class="ot"><-</span> <span class="cf">function</span>(data, params) {</span>
|
|||
|
<span id="cb28-2"><a href="#cb28-2" tabindex="-1"></a> <span class="fu">data.frame</span>(<span class="at">PANEL =</span> <span class="fu">c</span>(1L, 2L), <span class="at">SCALE_X =</span> 1L, <span class="at">SCALE_Y =</span> 1L)</span>
|
|||
|
<span id="cb28-3"><a href="#cb28-3" tabindex="-1"></a>}</span></code></pre></div>
|
|||
|
<p>This is quite simple as the faceting should just define two panels
|
|||
|
irrespectively of the input data and parameters.</p>
|
|||
|
</div>
|
|||
|
<div id="mapping-data-into-panels" class="section level3">
|
|||
|
<h3>Mapping data into panels</h3>
|
|||
|
<p>In order for ggplot2 to know which data should go where it needs the
|
|||
|
data to be assigned to a panel. The purpose of the mapping step is to
|
|||
|
assign a <code>PANEL</code> column to the layer data identifying which
|
|||
|
panel it belongs to.</p>
|
|||
|
<div class="sourceCode" id="cb29"><pre class="sourceCode r"><code class="sourceCode r"><span id="cb29-1"><a href="#cb29-1" tabindex="-1"></a>mapping <span class="ot"><-</span> <span class="cf">function</span>(data, layout, params) {</span>
|
|||
|
<span id="cb29-2"><a href="#cb29-2" tabindex="-1"></a> <span class="cf">if</span> (<span class="fu">is.null</span>(data) <span class="sc">||</span> <span class="fu">nrow</span>(data) <span class="sc">==</span> <span class="dv">0</span>) {</span>
|
|||
|
<span id="cb29-3"><a href="#cb29-3" tabindex="-1"></a> <span class="fu">return</span>(<span class="fu">cbind</span>(data, <span class="at">PANEL =</span> <span class="fu">integer</span>(<span class="dv">0</span>)))</span>
|
|||
|
<span id="cb29-4"><a href="#cb29-4" tabindex="-1"></a> }</span>
|
|||
|
<span id="cb29-5"><a href="#cb29-5" tabindex="-1"></a> <span class="fu">rbind</span>(</span>
|
|||
|
<span id="cb29-6"><a href="#cb29-6" tabindex="-1"></a> <span class="fu">cbind</span>(data, <span class="at">PANEL =</span> 1L),</span>
|
|||
|
<span id="cb29-7"><a href="#cb29-7" tabindex="-1"></a> <span class="fu">cbind</span>(data, <span class="at">PANEL =</span> 2L)</span>
|
|||
|
<span id="cb29-8"><a href="#cb29-8" tabindex="-1"></a> )</span>
|
|||
|
<span id="cb29-9"><a href="#cb29-9" tabindex="-1"></a>}</span></code></pre></div>
|
|||
|
<p>here we first investigate whether we have gotten an empty
|
|||
|
<code>data.frame</code> and if not we duplicate the data and assign the
|
|||
|
original data to the first panel and the new data to the second
|
|||
|
panel.</p>
|
|||
|
</div>
|
|||
|
<div id="laying-out-the-panels" class="section level3">
|
|||
|
<h3>Laying out the panels</h3>
|
|||
|
<p>While the two functions above have been deceivingly simple, this last
|
|||
|
one is going to take some more work. Our goal is to draw two panels
|
|||
|
beside (or above) each other with axes etc.</p>
|
|||
|
<div class="sourceCode" id="cb30"><pre class="sourceCode r"><code class="sourceCode r"><span id="cb30-1"><a href="#cb30-1" tabindex="-1"></a>render <span class="ot"><-</span> <span class="cf">function</span>(panels, layout, x_scales, y_scales, ranges, coord, data,</span>
|
|||
|
<span id="cb30-2"><a href="#cb30-2" tabindex="-1"></a> theme, params) {</span>
|
|||
|
<span id="cb30-3"><a href="#cb30-3" tabindex="-1"></a> <span class="co"># Place panels according to settings</span></span>
|
|||
|
<span id="cb30-4"><a href="#cb30-4" tabindex="-1"></a> <span class="cf">if</span> (params<span class="sc">$</span>horizontal) {</span>
|
|||
|
<span id="cb30-5"><a href="#cb30-5" tabindex="-1"></a> <span class="co"># Put panels in matrix and convert to a gtable</span></span>
|
|||
|
<span id="cb30-6"><a href="#cb30-6" tabindex="-1"></a> panels <span class="ot"><-</span> <span class="fu">matrix</span>(panels, <span class="at">ncol =</span> <span class="dv">2</span>)</span>
|
|||
|
<span id="cb30-7"><a href="#cb30-7" tabindex="-1"></a> panel_table <span class="ot"><-</span> gtable<span class="sc">::</span><span class="fu">gtable_matrix</span>(<span class="st">"layout"</span>, panels, </span>
|
|||
|
<span id="cb30-8"><a href="#cb30-8" tabindex="-1"></a> <span class="at">widths =</span> <span class="fu">unit</span>(<span class="fu">c</span>(<span class="dv">1</span>, <span class="dv">1</span>), <span class="st">"null"</span>), <span class="at">heights =</span> <span class="fu">unit</span>(<span class="dv">1</span>, <span class="st">"null"</span>), <span class="at">clip =</span> <span class="st">"on"</span>)</span>
|
|||
|
<span id="cb30-9"><a href="#cb30-9" tabindex="-1"></a> <span class="co"># Add spacing according to theme</span></span>
|
|||
|
<span id="cb30-10"><a href="#cb30-10" tabindex="-1"></a> panel_spacing <span class="ot"><-</span> <span class="cf">if</span> (<span class="fu">is.null</span>(theme<span class="sc">$</span>panel.spacing.x)) {</span>
|
|||
|
<span id="cb30-11"><a href="#cb30-11" tabindex="-1"></a> theme<span class="sc">$</span>panel.spacing</span>
|
|||
|
<span id="cb30-12"><a href="#cb30-12" tabindex="-1"></a> } <span class="cf">else</span> {</span>
|
|||
|
<span id="cb30-13"><a href="#cb30-13" tabindex="-1"></a> theme<span class="sc">$</span>panel.spacing.x</span>
|
|||
|
<span id="cb30-14"><a href="#cb30-14" tabindex="-1"></a> }</span>
|
|||
|
<span id="cb30-15"><a href="#cb30-15" tabindex="-1"></a> panel_table <span class="ot"><-</span> gtable<span class="sc">::</span><span class="fu">gtable_add_col_space</span>(panel_table, panel_spacing)</span>
|
|||
|
<span id="cb30-16"><a href="#cb30-16" tabindex="-1"></a> } <span class="cf">else</span> {</span>
|
|||
|
<span id="cb30-17"><a href="#cb30-17" tabindex="-1"></a> panels <span class="ot"><-</span> <span class="fu">matrix</span>(panels, <span class="at">ncol =</span> <span class="dv">1</span>)</span>
|
|||
|
<span id="cb30-18"><a href="#cb30-18" tabindex="-1"></a> panel_table <span class="ot"><-</span> gtable<span class="sc">::</span><span class="fu">gtable_matrix</span>(<span class="st">"layout"</span>, panels, </span>
|
|||
|
<span id="cb30-19"><a href="#cb30-19" tabindex="-1"></a> <span class="at">widths =</span> <span class="fu">unit</span>(<span class="dv">1</span>, <span class="st">"null"</span>), <span class="at">heights =</span> <span class="fu">unit</span>(<span class="fu">c</span>(<span class="dv">1</span>, <span class="dv">1</span>), <span class="st">"null"</span>), <span class="at">clip =</span> <span class="st">"on"</span>)</span>
|
|||
|
<span id="cb30-20"><a href="#cb30-20" tabindex="-1"></a> panel_spacing <span class="ot"><-</span> <span class="cf">if</span> (<span class="fu">is.null</span>(theme<span class="sc">$</span>panel.spacing.y)) {</span>
|
|||
|
<span id="cb30-21"><a href="#cb30-21" tabindex="-1"></a> theme<span class="sc">$</span>panel.spacing</span>
|
|||
|
<span id="cb30-22"><a href="#cb30-22" tabindex="-1"></a> } <span class="cf">else</span> {</span>
|
|||
|
<span id="cb30-23"><a href="#cb30-23" tabindex="-1"></a> theme<span class="sc">$</span>panel.spacing.y</span>
|
|||
|
<span id="cb30-24"><a href="#cb30-24" tabindex="-1"></a> }</span>
|
|||
|
<span id="cb30-25"><a href="#cb30-25" tabindex="-1"></a> panel_table <span class="ot"><-</span> gtable<span class="sc">::</span><span class="fu">gtable_add_row_space</span>(panel_table, panel_spacing)</span>
|
|||
|
<span id="cb30-26"><a href="#cb30-26" tabindex="-1"></a> }</span>
|
|||
|
<span id="cb30-27"><a href="#cb30-27" tabindex="-1"></a> <span class="co"># Name panel grobs so they can be found later</span></span>
|
|||
|
<span id="cb30-28"><a href="#cb30-28" tabindex="-1"></a> panel_table<span class="sc">$</span>layout<span class="sc">$</span>name <span class="ot"><-</span> <span class="fu">paste0</span>(<span class="st">"panel-"</span>, <span class="fu">c</span>(<span class="dv">1</span>, <span class="dv">2</span>))</span>
|
|||
|
<span id="cb30-29"><a href="#cb30-29" tabindex="-1"></a> </span>
|
|||
|
<span id="cb30-30"><a href="#cb30-30" tabindex="-1"></a> <span class="co"># Construct the axes</span></span>
|
|||
|
<span id="cb30-31"><a href="#cb30-31" tabindex="-1"></a> axes <span class="ot"><-</span> <span class="fu">render_axes</span>(ranges[<span class="dv">1</span>], ranges[<span class="dv">1</span>], coord, theme, </span>
|
|||
|
<span id="cb30-32"><a href="#cb30-32" tabindex="-1"></a> <span class="at">transpose =</span> <span class="cn">TRUE</span>)</span>
|
|||
|
<span id="cb30-33"><a href="#cb30-33" tabindex="-1"></a></span>
|
|||
|
<span id="cb30-34"><a href="#cb30-34" tabindex="-1"></a> <span class="co"># Add axes around each panel</span></span>
|
|||
|
<span id="cb30-35"><a href="#cb30-35" tabindex="-1"></a> panel_pos_h <span class="ot"><-</span> <span class="fu">panel_cols</span>(panel_table)<span class="sc">$</span>l</span>
|
|||
|
<span id="cb30-36"><a href="#cb30-36" tabindex="-1"></a> panel_pos_v <span class="ot"><-</span> <span class="fu">panel_rows</span>(panel_table)<span class="sc">$</span>t</span>
|
|||
|
<span id="cb30-37"><a href="#cb30-37" tabindex="-1"></a> axis_width_l <span class="ot"><-</span> <span class="fu">unit</span>(grid<span class="sc">::</span><span class="fu">convertWidth</span>(</span>
|
|||
|
<span id="cb30-38"><a href="#cb30-38" tabindex="-1"></a> grid<span class="sc">::</span><span class="fu">grobWidth</span>(axes<span class="sc">$</span>y<span class="sc">$</span>left[[<span class="dv">1</span>]]), <span class="st">"cm"</span>, <span class="cn">TRUE</span>), <span class="st">"cm"</span>)</span>
|
|||
|
<span id="cb30-39"><a href="#cb30-39" tabindex="-1"></a> axis_width_r <span class="ot"><-</span> <span class="fu">unit</span>(grid<span class="sc">::</span><span class="fu">convertWidth</span>(</span>
|
|||
|
<span id="cb30-40"><a href="#cb30-40" tabindex="-1"></a> grid<span class="sc">::</span><span class="fu">grobWidth</span>(axes<span class="sc">$</span>y<span class="sc">$</span>right[[<span class="dv">1</span>]]), <span class="st">"cm"</span>, <span class="cn">TRUE</span>), <span class="st">"cm"</span>)</span>
|
|||
|
<span id="cb30-41"><a href="#cb30-41" tabindex="-1"></a> <span class="do">## We do it reverse so we don't change the position of panels when we add axes</span></span>
|
|||
|
<span id="cb30-42"><a href="#cb30-42" tabindex="-1"></a> <span class="cf">for</span> (i <span class="cf">in</span> <span class="fu">rev</span>(panel_pos_h)) {</span>
|
|||
|
<span id="cb30-43"><a href="#cb30-43" tabindex="-1"></a> panel_table <span class="ot"><-</span> gtable<span class="sc">::</span><span class="fu">gtable_add_cols</span>(panel_table, axis_width_r, i)</span>
|
|||
|
<span id="cb30-44"><a href="#cb30-44" tabindex="-1"></a> panel_table <span class="ot"><-</span> gtable<span class="sc">::</span><span class="fu">gtable_add_grob</span>(panel_table, </span>
|
|||
|
<span id="cb30-45"><a href="#cb30-45" tabindex="-1"></a> <span class="fu">rep</span>(axes<span class="sc">$</span>y<span class="sc">$</span>right, <span class="fu">length</span>(panel_pos_v)), <span class="at">t =</span> panel_pos_v, <span class="at">l =</span> i <span class="sc">+</span> <span class="dv">1</span>, </span>
|
|||
|
<span id="cb30-46"><a href="#cb30-46" tabindex="-1"></a> <span class="at">clip =</span> <span class="st">"off"</span>)</span>
|
|||
|
<span id="cb30-47"><a href="#cb30-47" tabindex="-1"></a> panel_table <span class="ot"><-</span> gtable<span class="sc">::</span><span class="fu">gtable_add_cols</span>(panel_table, axis_width_l, i <span class="sc">-</span> <span class="dv">1</span>)</span>
|
|||
|
<span id="cb30-48"><a href="#cb30-48" tabindex="-1"></a> panel_table <span class="ot"><-</span> gtable<span class="sc">::</span><span class="fu">gtable_add_grob</span>(panel_table, </span>
|
|||
|
<span id="cb30-49"><a href="#cb30-49" tabindex="-1"></a> <span class="fu">rep</span>(axes<span class="sc">$</span>y<span class="sc">$</span>left, <span class="fu">length</span>(panel_pos_v)), <span class="at">t =</span> panel_pos_v, <span class="at">l =</span> i, </span>
|
|||
|
<span id="cb30-50"><a href="#cb30-50" tabindex="-1"></a> <span class="at">clip =</span> <span class="st">"off"</span>)</span>
|
|||
|
<span id="cb30-51"><a href="#cb30-51" tabindex="-1"></a> }</span>
|
|||
|
<span id="cb30-52"><a href="#cb30-52" tabindex="-1"></a> <span class="do">## Recalculate as gtable has changed</span></span>
|
|||
|
<span id="cb30-53"><a href="#cb30-53" tabindex="-1"></a> panel_pos_h <span class="ot"><-</span> <span class="fu">panel_cols</span>(panel_table)<span class="sc">$</span>l</span>
|
|||
|
<span id="cb30-54"><a href="#cb30-54" tabindex="-1"></a> panel_pos_v <span class="ot"><-</span> <span class="fu">panel_rows</span>(panel_table)<span class="sc">$</span>t</span>
|
|||
|
<span id="cb30-55"><a href="#cb30-55" tabindex="-1"></a> axis_height_t <span class="ot"><-</span> <span class="fu">unit</span>(grid<span class="sc">::</span><span class="fu">convertHeight</span>(</span>
|
|||
|
<span id="cb30-56"><a href="#cb30-56" tabindex="-1"></a> grid<span class="sc">::</span><span class="fu">grobHeight</span>(axes<span class="sc">$</span>x<span class="sc">$</span>top[[<span class="dv">1</span>]]), <span class="st">"cm"</span>, <span class="cn">TRUE</span>), <span class="st">"cm"</span>)</span>
|
|||
|
<span id="cb30-57"><a href="#cb30-57" tabindex="-1"></a> axis_height_b <span class="ot"><-</span> <span class="fu">unit</span>(grid<span class="sc">::</span><span class="fu">convertHeight</span>(</span>
|
|||
|
<span id="cb30-58"><a href="#cb30-58" tabindex="-1"></a> grid<span class="sc">::</span><span class="fu">grobHeight</span>(axes<span class="sc">$</span>x<span class="sc">$</span>bottom[[<span class="dv">1</span>]]), <span class="st">"cm"</span>, <span class="cn">TRUE</span>), <span class="st">"cm"</span>)</span>
|
|||
|
<span id="cb30-59"><a href="#cb30-59" tabindex="-1"></a> <span class="cf">for</span> (i <span class="cf">in</span> <span class="fu">rev</span>(panel_pos_v)) {</span>
|
|||
|
<span id="cb30-60"><a href="#cb30-60" tabindex="-1"></a> panel_table <span class="ot"><-</span> gtable<span class="sc">::</span><span class="fu">gtable_add_rows</span>(panel_table, axis_height_b, i)</span>
|
|||
|
<span id="cb30-61"><a href="#cb30-61" tabindex="-1"></a> panel_table <span class="ot"><-</span> gtable<span class="sc">::</span><span class="fu">gtable_add_grob</span>(panel_table, </span>
|
|||
|
<span id="cb30-62"><a href="#cb30-62" tabindex="-1"></a> <span class="fu">rep</span>(axes<span class="sc">$</span>x<span class="sc">$</span>bottom, <span class="fu">length</span>(panel_pos_h)), <span class="at">t =</span> i <span class="sc">+</span> <span class="dv">1</span>, <span class="at">l =</span> panel_pos_h, </span>
|
|||
|
<span id="cb30-63"><a href="#cb30-63" tabindex="-1"></a> <span class="at">clip =</span> <span class="st">"off"</span>)</span>
|
|||
|
<span id="cb30-64"><a href="#cb30-64" tabindex="-1"></a> panel_table <span class="ot"><-</span> gtable<span class="sc">::</span><span class="fu">gtable_add_rows</span>(panel_table, axis_height_t, i <span class="sc">-</span> <span class="dv">1</span>)</span>
|
|||
|
<span id="cb30-65"><a href="#cb30-65" tabindex="-1"></a> panel_table <span class="ot"><-</span> gtable<span class="sc">::</span><span class="fu">gtable_add_grob</span>(panel_table, </span>
|
|||
|
<span id="cb30-66"><a href="#cb30-66" tabindex="-1"></a> <span class="fu">rep</span>(axes<span class="sc">$</span>x<span class="sc">$</span>top, <span class="fu">length</span>(panel_pos_h)), <span class="at">t =</span> i, <span class="at">l =</span> panel_pos_h, </span>
|
|||
|
<span id="cb30-67"><a href="#cb30-67" tabindex="-1"></a> <span class="at">clip =</span> <span class="st">"off"</span>)</span>
|
|||
|
<span id="cb30-68"><a href="#cb30-68" tabindex="-1"></a> }</span>
|
|||
|
<span id="cb30-69"><a href="#cb30-69" tabindex="-1"></a> panel_table</span>
|
|||
|
<span id="cb30-70"><a href="#cb30-70" tabindex="-1"></a>}</span></code></pre></div>
|
|||
|
</div>
|
|||
|
<div id="assembling-the-facet-class" class="section level3">
|
|||
|
<h3>Assembling the Facet class</h3>
|
|||
|
<p>Usually all methods are defined within the class definition in the
|
|||
|
same way as is done for <code>Geom</code> and <code>Stat</code>. Here we
|
|||
|
have split it out so we could go through each in turn. All that remains
|
|||
|
is to assign our functions to the correct methods as well as making a
|
|||
|
constructor</p>
|
|||
|
<div class="sourceCode" id="cb31"><pre class="sourceCode r"><code class="sourceCode r"><span id="cb31-1"><a href="#cb31-1" tabindex="-1"></a><span class="co"># Constructor: shrink is required to govern whether scales are trained on </span></span>
|
|||
|
<span id="cb31-2"><a href="#cb31-2" tabindex="-1"></a><span class="co"># Stat-transformed data or not.</span></span>
|
|||
|
<span id="cb31-3"><a href="#cb31-3" tabindex="-1"></a>facet_duplicate <span class="ot"><-</span> <span class="cf">function</span>(<span class="at">horizontal =</span> <span class="cn">TRUE</span>, <span class="at">shrink =</span> <span class="cn">TRUE</span>) {</span>
|
|||
|
<span id="cb31-4"><a href="#cb31-4" tabindex="-1"></a> <span class="fu">ggproto</span>(<span class="cn">NULL</span>, FacetDuplicate,</span>
|
|||
|
<span id="cb31-5"><a href="#cb31-5" tabindex="-1"></a> <span class="at">shrink =</span> shrink,</span>
|
|||
|
<span id="cb31-6"><a href="#cb31-6" tabindex="-1"></a> <span class="at">params =</span> <span class="fu">list</span>(</span>
|
|||
|
<span id="cb31-7"><a href="#cb31-7" tabindex="-1"></a> <span class="at">horizontal =</span> horizontal</span>
|
|||
|
<span id="cb31-8"><a href="#cb31-8" tabindex="-1"></a> )</span>
|
|||
|
<span id="cb31-9"><a href="#cb31-9" tabindex="-1"></a> )</span>
|
|||
|
<span id="cb31-10"><a href="#cb31-10" tabindex="-1"></a>}</span>
|
|||
|
<span id="cb31-11"><a href="#cb31-11" tabindex="-1"></a></span>
|
|||
|
<span id="cb31-12"><a href="#cb31-12" tabindex="-1"></a>FacetDuplicate <span class="ot"><-</span> <span class="fu">ggproto</span>(<span class="st">"FacetDuplicate"</span>, Facet,</span>
|
|||
|
<span id="cb31-13"><a href="#cb31-13" tabindex="-1"></a> <span class="at">compute_layout =</span> layout,</span>
|
|||
|
<span id="cb31-14"><a href="#cb31-14" tabindex="-1"></a> <span class="at">map_data =</span> mapping,</span>
|
|||
|
<span id="cb31-15"><a href="#cb31-15" tabindex="-1"></a> <span class="at">draw_panels =</span> render</span>
|
|||
|
<span id="cb31-16"><a href="#cb31-16" tabindex="-1"></a>)</span></code></pre></div>
|
|||
|
<p>Now with everything assembled, lets test it out:</p>
|
|||
|
<div class="sourceCode" id="cb32"><pre class="sourceCode r"><code class="sourceCode r"><span id="cb32-1"><a href="#cb32-1" tabindex="-1"></a>p <span class="ot"><-</span> <span class="fu">ggplot</span>(mtcars, <span class="fu">aes</span>(<span class="at">x =</span> hp, <span class="at">y =</span> mpg)) <span class="sc">+</span> <span class="fu">geom_point</span>()</span>
|
|||
|
<span id="cb32-2"><a href="#cb32-2" tabindex="-1"></a>p</span></code></pre></div>
|
|||
|
<p><img src="
|
|||
|
<div class="sourceCode" id="cb33"><pre class="sourceCode r"><code class="sourceCode r"><span id="cb33-1"><a href="#cb33-1" tabindex="-1"></a>p <span class="sc">+</span> <span class="fu">facet_duplicate</span>()</span></code></pre></div>
|
|||
|
<p><img src="
|
|||
|
</div>
|
|||
|
<div id="doing-more-with-facets" class="section level3">
|
|||
|
<h3>Doing more with facets</h3>
|
|||
|
<p>The example above was pretty useless and we’ll now try to expand on
|
|||
|
it to add some actual usability. We are going to make a faceting that
|
|||
|
adds panels with y-transformed axes:</p>
|
|||
|
<div class="sourceCode" id="cb34"><pre class="sourceCode r"><code class="sourceCode r"><span id="cb34-1"><a href="#cb34-1" tabindex="-1"></a><span class="fu">library</span>(scales)</span>
|
|||
|
<span id="cb34-2"><a href="#cb34-2" tabindex="-1"></a></span>
|
|||
|
<span id="cb34-3"><a href="#cb34-3" tabindex="-1"></a>facet_trans <span class="ot"><-</span> <span class="cf">function</span>(trans, <span class="at">horizontal =</span> <span class="cn">TRUE</span>, <span class="at">shrink =</span> <span class="cn">TRUE</span>) {</span>
|
|||
|
<span id="cb34-4"><a href="#cb34-4" tabindex="-1"></a> <span class="fu">ggproto</span>(<span class="cn">NULL</span>, FacetTrans,</span>
|
|||
|
<span id="cb34-5"><a href="#cb34-5" tabindex="-1"></a> <span class="at">shrink =</span> shrink,</span>
|
|||
|
<span id="cb34-6"><a href="#cb34-6" tabindex="-1"></a> <span class="at">params =</span> <span class="fu">list</span>(</span>
|
|||
|
<span id="cb34-7"><a href="#cb34-7" tabindex="-1"></a> <span class="at">trans =</span> scales<span class="sc">::</span><span class="fu">as.transform</span>(trans),</span>
|
|||
|
<span id="cb34-8"><a href="#cb34-8" tabindex="-1"></a> <span class="at">horizontal =</span> horizontal</span>
|
|||
|
<span id="cb34-9"><a href="#cb34-9" tabindex="-1"></a> )</span>
|
|||
|
<span id="cb34-10"><a href="#cb34-10" tabindex="-1"></a> )</span>
|
|||
|
<span id="cb34-11"><a href="#cb34-11" tabindex="-1"></a>}</span>
|
|||
|
<span id="cb34-12"><a href="#cb34-12" tabindex="-1"></a></span>
|
|||
|
<span id="cb34-13"><a href="#cb34-13" tabindex="-1"></a>FacetTrans <span class="ot"><-</span> <span class="fu">ggproto</span>(<span class="st">"FacetTrans"</span>, Facet,</span>
|
|||
|
<span id="cb34-14"><a href="#cb34-14" tabindex="-1"></a> <span class="co"># Almost as before but we want different y-scales for each panel</span></span>
|
|||
|
<span id="cb34-15"><a href="#cb34-15" tabindex="-1"></a> <span class="at">compute_layout =</span> <span class="cf">function</span>(data, params) {</span>
|
|||
|
<span id="cb34-16"><a href="#cb34-16" tabindex="-1"></a> <span class="fu">data.frame</span>(<span class="at">PANEL =</span> <span class="fu">c</span>(1L, 2L), <span class="at">SCALE_X =</span> 1L, <span class="at">SCALE_Y =</span> <span class="fu">c</span>(1L, 2L))</span>
|
|||
|
<span id="cb34-17"><a href="#cb34-17" tabindex="-1"></a> },</span>
|
|||
|
<span id="cb34-18"><a href="#cb34-18" tabindex="-1"></a> <span class="co"># Same as before</span></span>
|
|||
|
<span id="cb34-19"><a href="#cb34-19" tabindex="-1"></a> <span class="at">map_data =</span> <span class="cf">function</span>(data, layout, params) {</span>
|
|||
|
<span id="cb34-20"><a href="#cb34-20" tabindex="-1"></a> <span class="cf">if</span> (<span class="fu">is.null</span>(data) <span class="sc">||</span> <span class="fu">nrow</span>(data) <span class="sc">==</span> <span class="dv">0</span>) {</span>
|
|||
|
<span id="cb34-21"><a href="#cb34-21" tabindex="-1"></a> <span class="fu">return</span>(<span class="fu">cbind</span>(data, <span class="at">PANEL =</span> <span class="fu">integer</span>(<span class="dv">0</span>)))</span>
|
|||
|
<span id="cb34-22"><a href="#cb34-22" tabindex="-1"></a> }</span>
|
|||
|
<span id="cb34-23"><a href="#cb34-23" tabindex="-1"></a> <span class="fu">rbind</span>(</span>
|
|||
|
<span id="cb34-24"><a href="#cb34-24" tabindex="-1"></a> <span class="fu">cbind</span>(data, <span class="at">PANEL =</span> 1L),</span>
|
|||
|
<span id="cb34-25"><a href="#cb34-25" tabindex="-1"></a> <span class="fu">cbind</span>(data, <span class="at">PANEL =</span> 2L)</span>
|
|||
|
<span id="cb34-26"><a href="#cb34-26" tabindex="-1"></a> )</span>
|
|||
|
<span id="cb34-27"><a href="#cb34-27" tabindex="-1"></a> },</span>
|
|||
|
<span id="cb34-28"><a href="#cb34-28" tabindex="-1"></a> <span class="co"># This is new. We create a new scale with the defined transformation</span></span>
|
|||
|
<span id="cb34-29"><a href="#cb34-29" tabindex="-1"></a> <span class="at">init_scales =</span> <span class="cf">function</span>(layout, <span class="at">x_scale =</span> <span class="cn">NULL</span>, <span class="at">y_scale =</span> <span class="cn">NULL</span>, params) {</span>
|
|||
|
<span id="cb34-30"><a href="#cb34-30" tabindex="-1"></a> scales <span class="ot"><-</span> <span class="fu">list</span>()</span>
|
|||
|
<span id="cb34-31"><a href="#cb34-31" tabindex="-1"></a> <span class="cf">if</span> (<span class="sc">!</span><span class="fu">is.null</span>(x_scale)) {</span>
|
|||
|
<span id="cb34-32"><a href="#cb34-32" tabindex="-1"></a> scales<span class="sc">$</span>x <span class="ot"><-</span> <span class="fu">lapply</span>(<span class="fu">seq_len</span>(<span class="fu">max</span>(layout<span class="sc">$</span>SCALE_X)), <span class="cf">function</span>(i) x_scale<span class="sc">$</span><span class="fu">clone</span>())</span>
|
|||
|
<span id="cb34-33"><a href="#cb34-33" tabindex="-1"></a> }</span>
|
|||
|
<span id="cb34-34"><a href="#cb34-34" tabindex="-1"></a> <span class="cf">if</span> (<span class="sc">!</span><span class="fu">is.null</span>(y_scale)) {</span>
|
|||
|
<span id="cb34-35"><a href="#cb34-35" tabindex="-1"></a> y_scale_orig <span class="ot"><-</span> y_scale<span class="sc">$</span><span class="fu">clone</span>()</span>
|
|||
|
<span id="cb34-36"><a href="#cb34-36" tabindex="-1"></a> y_scale_new <span class="ot"><-</span> y_scale<span class="sc">$</span><span class="fu">clone</span>()</span>
|
|||
|
<span id="cb34-37"><a href="#cb34-37" tabindex="-1"></a> y_scale_new<span class="sc">$</span>trans <span class="ot"><-</span> params<span class="sc">$</span>trans</span>
|
|||
|
<span id="cb34-38"><a href="#cb34-38" tabindex="-1"></a> <span class="co"># Make sure that oob values are kept</span></span>
|
|||
|
<span id="cb34-39"><a href="#cb34-39" tabindex="-1"></a> y_scale_new<span class="sc">$</span>oob <span class="ot"><-</span> <span class="cf">function</span>(x, ...) x</span>
|
|||
|
<span id="cb34-40"><a href="#cb34-40" tabindex="-1"></a> scales<span class="sc">$</span>y <span class="ot"><-</span> <span class="fu">list</span>(y_scale_orig, y_scale_new)</span>
|
|||
|
<span id="cb34-41"><a href="#cb34-41" tabindex="-1"></a> }</span>
|
|||
|
<span id="cb34-42"><a href="#cb34-42" tabindex="-1"></a> scales</span>
|
|||
|
<span id="cb34-43"><a href="#cb34-43" tabindex="-1"></a> },</span>
|
|||
|
<span id="cb34-44"><a href="#cb34-44" tabindex="-1"></a> <span class="co"># We must make sure that the second scale is trained on transformed data</span></span>
|
|||
|
<span id="cb34-45"><a href="#cb34-45" tabindex="-1"></a> <span class="at">train_scales =</span> <span class="cf">function</span>(x_scales, y_scales, layout, data, params) {</span>
|
|||
|
<span id="cb34-46"><a href="#cb34-46" tabindex="-1"></a> <span class="co"># Transform data for second panel prior to scale training</span></span>
|
|||
|
<span id="cb34-47"><a href="#cb34-47" tabindex="-1"></a> <span class="cf">if</span> (<span class="sc">!</span><span class="fu">is.null</span>(y_scales)) {</span>
|
|||
|
<span id="cb34-48"><a href="#cb34-48" tabindex="-1"></a> data <span class="ot"><-</span> <span class="fu">lapply</span>(data, <span class="cf">function</span>(layer_data) {</span>
|
|||
|
<span id="cb34-49"><a href="#cb34-49" tabindex="-1"></a> match_id <span class="ot"><-</span> <span class="fu">match</span>(layer_data<span class="sc">$</span>PANEL, layout<span class="sc">$</span>PANEL)</span>
|
|||
|
<span id="cb34-50"><a href="#cb34-50" tabindex="-1"></a> y_vars <span class="ot"><-</span> <span class="fu">intersect</span>(y_scales[[<span class="dv">1</span>]]<span class="sc">$</span>aesthetics, <span class="fu">names</span>(layer_data))</span>
|
|||
|
<span id="cb34-51"><a href="#cb34-51" tabindex="-1"></a> trans_scale <span class="ot"><-</span> layer_data<span class="sc">$</span>PANEL <span class="sc">==</span> 2L</span>
|
|||
|
<span id="cb34-52"><a href="#cb34-52" tabindex="-1"></a> <span class="cf">for</span> (i <span class="cf">in</span> y_vars) {</span>
|
|||
|
<span id="cb34-53"><a href="#cb34-53" tabindex="-1"></a> layer_data[trans_scale, i] <span class="ot"><-</span> y_scales[[<span class="dv">2</span>]]<span class="sc">$</span><span class="fu">transform</span>(layer_data[trans_scale, i])</span>
|
|||
|
<span id="cb34-54"><a href="#cb34-54" tabindex="-1"></a> }</span>
|
|||
|
<span id="cb34-55"><a href="#cb34-55" tabindex="-1"></a> layer_data</span>
|
|||
|
<span id="cb34-56"><a href="#cb34-56" tabindex="-1"></a> })</span>
|
|||
|
<span id="cb34-57"><a href="#cb34-57" tabindex="-1"></a> }</span>
|
|||
|
<span id="cb34-58"><a href="#cb34-58" tabindex="-1"></a> Facet<span class="sc">$</span><span class="fu">train_scales</span>(x_scales, y_scales, layout, data, params)</span>
|
|||
|
<span id="cb34-59"><a href="#cb34-59" tabindex="-1"></a> },</span>
|
|||
|
<span id="cb34-60"><a href="#cb34-60" tabindex="-1"></a> <span class="co"># this is where we actually modify the data. It cannot be done in $map_data as that function</span></span>
|
|||
|
<span id="cb34-61"><a href="#cb34-61" tabindex="-1"></a> <span class="co"># doesn't have access to the scales</span></span>
|
|||
|
<span id="cb34-62"><a href="#cb34-62" tabindex="-1"></a> <span class="at">finish_data =</span> <span class="cf">function</span>(data, layout, x_scales, y_scales, params) {</span>
|
|||
|
<span id="cb34-63"><a href="#cb34-63" tabindex="-1"></a> match_id <span class="ot"><-</span> <span class="fu">match</span>(data<span class="sc">$</span>PANEL, layout<span class="sc">$</span>PANEL)</span>
|
|||
|
<span id="cb34-64"><a href="#cb34-64" tabindex="-1"></a> y_vars <span class="ot"><-</span> <span class="fu">intersect</span>(y_scales[[<span class="dv">1</span>]]<span class="sc">$</span>aesthetics, <span class="fu">names</span>(data))</span>
|
|||
|
<span id="cb34-65"><a href="#cb34-65" tabindex="-1"></a> trans_scale <span class="ot"><-</span> data<span class="sc">$</span>PANEL <span class="sc">==</span> 2L</span>
|
|||
|
<span id="cb34-66"><a href="#cb34-66" tabindex="-1"></a> <span class="cf">for</span> (i <span class="cf">in</span> y_vars) {</span>
|
|||
|
<span id="cb34-67"><a href="#cb34-67" tabindex="-1"></a> data[trans_scale, i] <span class="ot"><-</span> y_scales[[<span class="dv">2</span>]]<span class="sc">$</span><span class="fu">transform</span>(data[trans_scale, i])</span>
|
|||
|
<span id="cb34-68"><a href="#cb34-68" tabindex="-1"></a> }</span>
|
|||
|
<span id="cb34-69"><a href="#cb34-69" tabindex="-1"></a> data</span>
|
|||
|
<span id="cb34-70"><a href="#cb34-70" tabindex="-1"></a> },</span>
|
|||
|
<span id="cb34-71"><a href="#cb34-71" tabindex="-1"></a> <span class="co"># A few changes from before to accommodate that axes are now not duplicate of each other</span></span>
|
|||
|
<span id="cb34-72"><a href="#cb34-72" tabindex="-1"></a> <span class="co"># We also add a panel strip to annotate the different panels</span></span>
|
|||
|
<span id="cb34-73"><a href="#cb34-73" tabindex="-1"></a> <span class="at">draw_panels =</span> <span class="cf">function</span>(panels, layout, x_scales, y_scales, ranges, coord,</span>
|
|||
|
<span id="cb34-74"><a href="#cb34-74" tabindex="-1"></a> data, theme, params) {</span>
|
|||
|
<span id="cb34-75"><a href="#cb34-75" tabindex="-1"></a> <span class="co"># Place panels according to settings</span></span>
|
|||
|
<span id="cb34-76"><a href="#cb34-76" tabindex="-1"></a> <span class="cf">if</span> (params<span class="sc">$</span>horizontal) {</span>
|
|||
|
<span id="cb34-77"><a href="#cb34-77" tabindex="-1"></a> <span class="co"># Put panels in matrix and convert to a gtable</span></span>
|
|||
|
<span id="cb34-78"><a href="#cb34-78" tabindex="-1"></a> panels <span class="ot"><-</span> <span class="fu">matrix</span>(panels, <span class="at">ncol =</span> <span class="dv">2</span>)</span>
|
|||
|
<span id="cb34-79"><a href="#cb34-79" tabindex="-1"></a> panel_table <span class="ot"><-</span> gtable<span class="sc">::</span><span class="fu">gtable_matrix</span>(<span class="st">"layout"</span>, panels, </span>
|
|||
|
<span id="cb34-80"><a href="#cb34-80" tabindex="-1"></a> <span class="at">widths =</span> <span class="fu">unit</span>(<span class="fu">c</span>(<span class="dv">1</span>, <span class="dv">1</span>), <span class="st">"null"</span>), <span class="at">heights =</span> <span class="fu">unit</span>(<span class="dv">1</span>, <span class="st">"null"</span>), <span class="at">clip =</span> <span class="st">"on"</span>)</span>
|
|||
|
<span id="cb34-81"><a href="#cb34-81" tabindex="-1"></a> <span class="co"># Add spacing according to theme</span></span>
|
|||
|
<span id="cb34-82"><a href="#cb34-82" tabindex="-1"></a> panel_spacing <span class="ot"><-</span> <span class="cf">if</span> (<span class="fu">is.null</span>(theme<span class="sc">$</span>panel.spacing.x)) {</span>
|
|||
|
<span id="cb34-83"><a href="#cb34-83" tabindex="-1"></a> theme<span class="sc">$</span>panel.spacing</span>
|
|||
|
<span id="cb34-84"><a href="#cb34-84" tabindex="-1"></a> } <span class="cf">else</span> {</span>
|
|||
|
<span id="cb34-85"><a href="#cb34-85" tabindex="-1"></a> theme<span class="sc">$</span>panel.spacing.x</span>
|
|||
|
<span id="cb34-86"><a href="#cb34-86" tabindex="-1"></a> }</span>
|
|||
|
<span id="cb34-87"><a href="#cb34-87" tabindex="-1"></a> panel_table <span class="ot"><-</span> gtable<span class="sc">::</span><span class="fu">gtable_add_col_space</span>(panel_table, panel_spacing)</span>
|
|||
|
<span id="cb34-88"><a href="#cb34-88" tabindex="-1"></a> } <span class="cf">else</span> {</span>
|
|||
|
<span id="cb34-89"><a href="#cb34-89" tabindex="-1"></a> panels <span class="ot"><-</span> <span class="fu">matrix</span>(panels, <span class="at">ncol =</span> <span class="dv">1</span>)</span>
|
|||
|
<span id="cb34-90"><a href="#cb34-90" tabindex="-1"></a> panel_table <span class="ot"><-</span> gtable<span class="sc">::</span><span class="fu">gtable_matrix</span>(<span class="st">"layout"</span>, panels, </span>
|
|||
|
<span id="cb34-91"><a href="#cb34-91" tabindex="-1"></a> <span class="at">widths =</span> <span class="fu">unit</span>(<span class="dv">1</span>, <span class="st">"null"</span>), <span class="at">heights =</span> <span class="fu">unit</span>(<span class="fu">c</span>(<span class="dv">1</span>, <span class="dv">1</span>), <span class="st">"null"</span>), <span class="at">clip =</span> <span class="st">"on"</span>)</span>
|
|||
|
<span id="cb34-92"><a href="#cb34-92" tabindex="-1"></a> panel_spacing <span class="ot"><-</span> <span class="cf">if</span> (<span class="fu">is.null</span>(theme<span class="sc">$</span>panel.spacing.y)) {</span>
|
|||
|
<span id="cb34-93"><a href="#cb34-93" tabindex="-1"></a> theme<span class="sc">$</span>panel.spacing</span>
|
|||
|
<span id="cb34-94"><a href="#cb34-94" tabindex="-1"></a> } <span class="cf">else</span> {</span>
|
|||
|
<span id="cb34-95"><a href="#cb34-95" tabindex="-1"></a> theme<span class="sc">$</span>panel.spacing.y</span>
|
|||
|
<span id="cb34-96"><a href="#cb34-96" tabindex="-1"></a> }</span>
|
|||
|
<span id="cb34-97"><a href="#cb34-97" tabindex="-1"></a> panel_table <span class="ot"><-</span> gtable<span class="sc">::</span><span class="fu">gtable_add_row_space</span>(panel_table, panel_spacing)</span>
|
|||
|
<span id="cb34-98"><a href="#cb34-98" tabindex="-1"></a> }</span>
|
|||
|
<span id="cb34-99"><a href="#cb34-99" tabindex="-1"></a> <span class="co"># Name panel grobs so they can be found later</span></span>
|
|||
|
<span id="cb34-100"><a href="#cb34-100" tabindex="-1"></a> panel_table<span class="sc">$</span>layout<span class="sc">$</span>name <span class="ot"><-</span> <span class="fu">paste0</span>(<span class="st">"panel-"</span>, <span class="fu">c</span>(<span class="dv">1</span>, <span class="dv">2</span>))</span>
|
|||
|
<span id="cb34-101"><a href="#cb34-101" tabindex="-1"></a> </span>
|
|||
|
<span id="cb34-102"><a href="#cb34-102" tabindex="-1"></a> <span class="co"># Construct the axes</span></span>
|
|||
|
<span id="cb34-103"><a href="#cb34-103" tabindex="-1"></a> axes <span class="ot"><-</span> <span class="fu">render_axes</span>(ranges[<span class="dv">1</span>], ranges, coord, theme, </span>
|
|||
|
<span id="cb34-104"><a href="#cb34-104" tabindex="-1"></a> <span class="at">transpose =</span> <span class="cn">TRUE</span>)</span>
|
|||
|
<span id="cb34-105"><a href="#cb34-105" tabindex="-1"></a> </span>
|
|||
|
<span id="cb34-106"><a href="#cb34-106" tabindex="-1"></a> <span class="co"># Add axes around each panel</span></span>
|
|||
|
<span id="cb34-107"><a href="#cb34-107" tabindex="-1"></a> grobWidths <span class="ot"><-</span> <span class="cf">function</span>(x) {</span>
|
|||
|
<span id="cb34-108"><a href="#cb34-108" tabindex="-1"></a> <span class="fu">unit</span>(<span class="fu">vapply</span>(x, <span class="cf">function</span>(x) {</span>
|
|||
|
<span id="cb34-109"><a href="#cb34-109" tabindex="-1"></a> grid<span class="sc">::</span><span class="fu">convertWidth</span>(</span>
|
|||
|
<span id="cb34-110"><a href="#cb34-110" tabindex="-1"></a> grid<span class="sc">::</span><span class="fu">grobWidth</span>(x), <span class="st">"cm"</span>, <span class="cn">TRUE</span>)</span>
|
|||
|
<span id="cb34-111"><a href="#cb34-111" tabindex="-1"></a> }, <span class="fu">numeric</span>(<span class="dv">1</span>)), <span class="st">"cm"</span>)</span>
|
|||
|
<span id="cb34-112"><a href="#cb34-112" tabindex="-1"></a> }</span>
|
|||
|
<span id="cb34-113"><a href="#cb34-113" tabindex="-1"></a> panel_pos_h <span class="ot"><-</span> <span class="fu">panel_cols</span>(panel_table)<span class="sc">$</span>l</span>
|
|||
|
<span id="cb34-114"><a href="#cb34-114" tabindex="-1"></a> panel_pos_v <span class="ot"><-</span> <span class="fu">panel_rows</span>(panel_table)<span class="sc">$</span>t</span>
|
|||
|
<span id="cb34-115"><a href="#cb34-115" tabindex="-1"></a> axis_width_l <span class="ot"><-</span> <span class="fu">grobWidths</span>(axes<span class="sc">$</span>y<span class="sc">$</span>left)</span>
|
|||
|
<span id="cb34-116"><a href="#cb34-116" tabindex="-1"></a> axis_width_r <span class="ot"><-</span> <span class="fu">grobWidths</span>(axes<span class="sc">$</span>y<span class="sc">$</span>right)</span>
|
|||
|
<span id="cb34-117"><a href="#cb34-117" tabindex="-1"></a> <span class="do">## We do it reverse so we don't change the position of panels when we add axes</span></span>
|
|||
|
<span id="cb34-118"><a href="#cb34-118" tabindex="-1"></a> <span class="cf">if</span> (params<span class="sc">$</span>horizontal) {</span>
|
|||
|
<span id="cb34-119"><a href="#cb34-119" tabindex="-1"></a> <span class="cf">for</span> (i <span class="cf">in</span> <span class="fu">rev</span>(<span class="fu">seq_along</span>(panel_pos_h))) {</span>
|
|||
|
<span id="cb34-120"><a href="#cb34-120" tabindex="-1"></a> panel_table <span class="ot"><-</span> gtable<span class="sc">::</span><span class="fu">gtable_add_cols</span>(panel_table, axis_width_r[i], panel_pos_h[i])</span>
|
|||
|
<span id="cb34-121"><a href="#cb34-121" tabindex="-1"></a> panel_table <span class="ot"><-</span> gtable<span class="sc">::</span><span class="fu">gtable_add_grob</span>(panel_table,</span>
|
|||
|
<span id="cb34-122"><a href="#cb34-122" tabindex="-1"></a> axes<span class="sc">$</span>y<span class="sc">$</span>right[i], <span class="at">t =</span> panel_pos_v, <span class="at">l =</span> panel_pos_h[i] <span class="sc">+</span> <span class="dv">1</span>,</span>
|
|||
|
<span id="cb34-123"><a href="#cb34-123" tabindex="-1"></a> <span class="at">clip =</span> <span class="st">"off"</span>)</span>
|
|||
|
<span id="cb34-124"><a href="#cb34-124" tabindex="-1"></a></span>
|
|||
|
<span id="cb34-125"><a href="#cb34-125" tabindex="-1"></a> panel_table <span class="ot"><-</span> gtable<span class="sc">::</span><span class="fu">gtable_add_cols</span>(panel_table, axis_width_l[i], panel_pos_h[i] <span class="sc">-</span> <span class="dv">1</span>)</span>
|
|||
|
<span id="cb34-126"><a href="#cb34-126" tabindex="-1"></a> panel_table <span class="ot"><-</span> gtable<span class="sc">::</span><span class="fu">gtable_add_grob</span>(panel_table,</span>
|
|||
|
<span id="cb34-127"><a href="#cb34-127" tabindex="-1"></a> axes<span class="sc">$</span>y<span class="sc">$</span>left[i], <span class="at">t =</span> panel_pos_v, <span class="at">l =</span> panel_pos_h[i],</span>
|
|||
|
<span id="cb34-128"><a href="#cb34-128" tabindex="-1"></a> <span class="at">clip =</span> <span class="st">"off"</span>)</span>
|
|||
|
<span id="cb34-129"><a href="#cb34-129" tabindex="-1"></a> }</span>
|
|||
|
<span id="cb34-130"><a href="#cb34-130" tabindex="-1"></a> } <span class="cf">else</span> {</span>
|
|||
|
<span id="cb34-131"><a href="#cb34-131" tabindex="-1"></a> panel_table <span class="ot"><-</span> gtable<span class="sc">::</span><span class="fu">gtable_add_cols</span>(panel_table, axis_width_r[<span class="dv">1</span>], panel_pos_h)</span>
|
|||
|
<span id="cb34-132"><a href="#cb34-132" tabindex="-1"></a> panel_table <span class="ot"><-</span> gtable<span class="sc">::</span><span class="fu">gtable_add_grob</span>(panel_table,</span>
|
|||
|
<span id="cb34-133"><a href="#cb34-133" tabindex="-1"></a> axes<span class="sc">$</span>y<span class="sc">$</span>right, <span class="at">t =</span> panel_pos_v, <span class="at">l =</span> panel_pos_h <span class="sc">+</span> <span class="dv">1</span>,</span>
|
|||
|
<span id="cb34-134"><a href="#cb34-134" tabindex="-1"></a> <span class="at">clip =</span> <span class="st">"off"</span>)</span>
|
|||
|
<span id="cb34-135"><a href="#cb34-135" tabindex="-1"></a> panel_table <span class="ot"><-</span> gtable<span class="sc">::</span><span class="fu">gtable_add_cols</span>(panel_table, axis_width_l[<span class="dv">1</span>], panel_pos_h <span class="sc">-</span> <span class="dv">1</span>)</span>
|
|||
|
<span id="cb34-136"><a href="#cb34-136" tabindex="-1"></a> panel_table <span class="ot"><-</span> gtable<span class="sc">::</span><span class="fu">gtable_add_grob</span>(panel_table,</span>
|
|||
|
<span id="cb34-137"><a href="#cb34-137" tabindex="-1"></a> axes<span class="sc">$</span>y<span class="sc">$</span>left, <span class="at">t =</span> panel_pos_v, <span class="at">l =</span> panel_pos_h,</span>
|
|||
|
<span id="cb34-138"><a href="#cb34-138" tabindex="-1"></a> <span class="at">clip =</span> <span class="st">"off"</span>)</span>
|
|||
|
<span id="cb34-139"><a href="#cb34-139" tabindex="-1"></a> }</span>
|
|||
|
<span id="cb34-140"><a href="#cb34-140" tabindex="-1"></a></span>
|
|||
|
<span id="cb34-141"><a href="#cb34-141" tabindex="-1"></a> <span class="do">## Recalculate as gtable has changed</span></span>
|
|||
|
<span id="cb34-142"><a href="#cb34-142" tabindex="-1"></a> panel_pos_h <span class="ot"><-</span> <span class="fu">panel_cols</span>(panel_table)<span class="sc">$</span>l</span>
|
|||
|
<span id="cb34-143"><a href="#cb34-143" tabindex="-1"></a> panel_pos_v <span class="ot"><-</span> <span class="fu">panel_rows</span>(panel_table)<span class="sc">$</span>t</span>
|
|||
|
<span id="cb34-144"><a href="#cb34-144" tabindex="-1"></a> axis_height_t <span class="ot"><-</span> <span class="fu">unit</span>(grid<span class="sc">::</span><span class="fu">convertHeight</span>(</span>
|
|||
|
<span id="cb34-145"><a href="#cb34-145" tabindex="-1"></a> grid<span class="sc">::</span><span class="fu">grobHeight</span>(axes<span class="sc">$</span>x<span class="sc">$</span>top[[<span class="dv">1</span>]]), <span class="st">"cm"</span>, <span class="cn">TRUE</span>), <span class="st">"cm"</span>)</span>
|
|||
|
<span id="cb34-146"><a href="#cb34-146" tabindex="-1"></a> axis_height_b <span class="ot"><-</span> <span class="fu">unit</span>(grid<span class="sc">::</span><span class="fu">convertHeight</span>(</span>
|
|||
|
<span id="cb34-147"><a href="#cb34-147" tabindex="-1"></a> grid<span class="sc">::</span><span class="fu">grobHeight</span>(axes<span class="sc">$</span>x<span class="sc">$</span>bottom[[<span class="dv">1</span>]]), <span class="st">"cm"</span>, <span class="cn">TRUE</span>), <span class="st">"cm"</span>)</span>
|
|||
|
<span id="cb34-148"><a href="#cb34-148" tabindex="-1"></a> <span class="cf">for</span> (i <span class="cf">in</span> <span class="fu">rev</span>(panel_pos_v)) {</span>
|
|||
|
<span id="cb34-149"><a href="#cb34-149" tabindex="-1"></a> panel_table <span class="ot"><-</span> gtable<span class="sc">::</span><span class="fu">gtable_add_rows</span>(panel_table, axis_height_b, i)</span>
|
|||
|
<span id="cb34-150"><a href="#cb34-150" tabindex="-1"></a> panel_table <span class="ot"><-</span> gtable<span class="sc">::</span><span class="fu">gtable_add_grob</span>(panel_table, </span>
|
|||
|
<span id="cb34-151"><a href="#cb34-151" tabindex="-1"></a> <span class="fu">rep</span>(axes<span class="sc">$</span>x<span class="sc">$</span>bottom, <span class="fu">length</span>(panel_pos_h)), <span class="at">t =</span> i <span class="sc">+</span> <span class="dv">1</span>, <span class="at">l =</span> panel_pos_h, </span>
|
|||
|
<span id="cb34-152"><a href="#cb34-152" tabindex="-1"></a> <span class="at">clip =</span> <span class="st">"off"</span>)</span>
|
|||
|
<span id="cb34-153"><a href="#cb34-153" tabindex="-1"></a> panel_table <span class="ot"><-</span> gtable<span class="sc">::</span><span class="fu">gtable_add_rows</span>(panel_table, axis_height_t, i <span class="sc">-</span> <span class="dv">1</span>)</span>
|
|||
|
<span id="cb34-154"><a href="#cb34-154" tabindex="-1"></a> panel_table <span class="ot"><-</span> gtable<span class="sc">::</span><span class="fu">gtable_add_grob</span>(panel_table, </span>
|
|||
|
<span id="cb34-155"><a href="#cb34-155" tabindex="-1"></a> <span class="fu">rep</span>(axes<span class="sc">$</span>x<span class="sc">$</span>top, <span class="fu">length</span>(panel_pos_h)), <span class="at">t =</span> i, <span class="at">l =</span> panel_pos_h, </span>
|
|||
|
<span id="cb34-156"><a href="#cb34-156" tabindex="-1"></a> <span class="at">clip =</span> <span class="st">"off"</span>)</span>
|
|||
|
<span id="cb34-157"><a href="#cb34-157" tabindex="-1"></a> }</span>
|
|||
|
<span id="cb34-158"><a href="#cb34-158" tabindex="-1"></a> </span>
|
|||
|
<span id="cb34-159"><a href="#cb34-159" tabindex="-1"></a> <span class="co"># Add strips</span></span>
|
|||
|
<span id="cb34-160"><a href="#cb34-160" tabindex="-1"></a> strips <span class="ot"><-</span> <span class="fu">render_strips</span>(</span>
|
|||
|
<span id="cb34-161"><a href="#cb34-161" tabindex="-1"></a> <span class="at">x =</span> <span class="fu">data.frame</span>(<span class="at">name =</span> <span class="fu">c</span>(<span class="st">"Original"</span>, <span class="fu">paste0</span>(<span class="st">"Transformed ("</span>, params<span class="sc">$</span>trans<span class="sc">$</span>name, <span class="st">")"</span>))),</span>
|
|||
|
<span id="cb34-162"><a href="#cb34-162" tabindex="-1"></a> <span class="at">labeller =</span> label_value, <span class="at">theme =</span> theme)</span>
|
|||
|
<span id="cb34-163"><a href="#cb34-163" tabindex="-1"></a> </span>
|
|||
|
<span id="cb34-164"><a href="#cb34-164" tabindex="-1"></a> panel_pos_h <span class="ot"><-</span> <span class="fu">panel_cols</span>(panel_table)<span class="sc">$</span>l</span>
|
|||
|
<span id="cb34-165"><a href="#cb34-165" tabindex="-1"></a> panel_pos_v <span class="ot"><-</span> <span class="fu">panel_rows</span>(panel_table)<span class="sc">$</span>t</span>
|
|||
|
<span id="cb34-166"><a href="#cb34-166" tabindex="-1"></a> strip_height <span class="ot"><-</span> <span class="fu">unit</span>(grid<span class="sc">::</span><span class="fu">convertHeight</span>(</span>
|
|||
|
<span id="cb34-167"><a href="#cb34-167" tabindex="-1"></a> grid<span class="sc">::</span><span class="fu">grobHeight</span>(strips<span class="sc">$</span>x<span class="sc">$</span>top[[<span class="dv">1</span>]]), <span class="st">"cm"</span>, <span class="cn">TRUE</span>), <span class="st">"cm"</span>)</span>
|
|||
|
<span id="cb34-168"><a href="#cb34-168" tabindex="-1"></a> <span class="cf">for</span> (i <span class="cf">in</span> <span class="fu">rev</span>(<span class="fu">seq_along</span>(panel_pos_v))) {</span>
|
|||
|
<span id="cb34-169"><a href="#cb34-169" tabindex="-1"></a> panel_table <span class="ot"><-</span> gtable<span class="sc">::</span><span class="fu">gtable_add_rows</span>(panel_table, strip_height, panel_pos_v[i] <span class="sc">-</span> <span class="dv">1</span>)</span>
|
|||
|
<span id="cb34-170"><a href="#cb34-170" tabindex="-1"></a> <span class="cf">if</span> (params<span class="sc">$</span>horizontal) {</span>
|
|||
|
<span id="cb34-171"><a href="#cb34-171" tabindex="-1"></a> panel_table <span class="ot"><-</span> gtable<span class="sc">::</span><span class="fu">gtable_add_grob</span>(panel_table, strips<span class="sc">$</span>x<span class="sc">$</span>top, </span>
|
|||
|
<span id="cb34-172"><a href="#cb34-172" tabindex="-1"></a> <span class="at">t =</span> panel_pos_v[i], <span class="at">l =</span> panel_pos_h, <span class="at">clip =</span> <span class="st">"off"</span>)</span>
|
|||
|
<span id="cb34-173"><a href="#cb34-173" tabindex="-1"></a> } <span class="cf">else</span> {</span>
|
|||
|
<span id="cb34-174"><a href="#cb34-174" tabindex="-1"></a> panel_table <span class="ot"><-</span> gtable<span class="sc">::</span><span class="fu">gtable_add_grob</span>(panel_table, strips<span class="sc">$</span>x<span class="sc">$</span>top[i], </span>
|
|||
|
<span id="cb34-175"><a href="#cb34-175" tabindex="-1"></a> <span class="at">t =</span> panel_pos_v[i], <span class="at">l =</span> panel_pos_h, <span class="at">clip =</span> <span class="st">"off"</span>)</span>
|
|||
|
<span id="cb34-176"><a href="#cb34-176" tabindex="-1"></a> }</span>
|
|||
|
<span id="cb34-177"><a href="#cb34-177" tabindex="-1"></a> }</span>
|
|||
|
<span id="cb34-178"><a href="#cb34-178" tabindex="-1"></a> </span>
|
|||
|
<span id="cb34-179"><a href="#cb34-179" tabindex="-1"></a> </span>
|
|||
|
<span id="cb34-180"><a href="#cb34-180" tabindex="-1"></a> panel_table</span>
|
|||
|
<span id="cb34-181"><a href="#cb34-181" tabindex="-1"></a> }</span>
|
|||
|
<span id="cb34-182"><a href="#cb34-182" tabindex="-1"></a>)</span></code></pre></div>
|
|||
|
<p>As is very apparent, the <code>draw_panel</code> method can become
|
|||
|
very unwieldy once it begins to take multiple possibilities into
|
|||
|
account. The fact that we want to support both horizontal and vertical
|
|||
|
layout leads to a lot of if/else blocks in the above code. In general,
|
|||
|
this is the big challenge when writing facet extensions so be prepared
|
|||
|
to be very meticulous when writing these methods.</p>
|
|||
|
<p>Enough talk - lets see if our new and powerful faceting extension
|
|||
|
works:</p>
|
|||
|
<div class="sourceCode" id="cb35"><pre class="sourceCode r"><code class="sourceCode r"><span id="cb35-1"><a href="#cb35-1" tabindex="-1"></a><span class="fu">ggplot</span>(mtcars, <span class="fu">aes</span>(<span class="at">x =</span> hp, <span class="at">y =</span> mpg)) <span class="sc">+</span> <span class="fu">geom_point</span>() <span class="sc">+</span> <span class="fu">facet_trans</span>(<span class="st">'sqrt'</span>)</span></code></pre></div>
|
|||
|
<p><img src="
|
|||
|
</div>
|
|||
|
</div>
|
|||
|
<div id="extending-existing-facet-function" class="section level2">
|
|||
|
<h2>Extending existing facet function</h2>
|
|||
|
<p>As the rendering part of a facet class is often the difficult
|
|||
|
development step, it is possible to piggyback on the existing faceting
|
|||
|
classes to achieve a range of new facetings. Below we will subclass
|
|||
|
<code>facet_wrap()</code> to make a <code>facet_bootstrap()</code> class
|
|||
|
that splits the input data into a number of panels at random.</p>
|
|||
|
<div class="sourceCode" id="cb36"><pre class="sourceCode r"><code class="sourceCode r"><span id="cb36-1"><a href="#cb36-1" tabindex="-1"></a>facet_bootstrap <span class="ot"><-</span> <span class="cf">function</span>(<span class="at">n =</span> <span class="dv">9</span>, <span class="at">prop =</span> <span class="fl">0.2</span>, <span class="at">nrow =</span> <span class="cn">NULL</span>, <span class="at">ncol =</span> <span class="cn">NULL</span>, </span>
|
|||
|
<span id="cb36-2"><a href="#cb36-2" tabindex="-1"></a> <span class="at">scales =</span> <span class="st">"fixed"</span>, <span class="at">shrink =</span> <span class="cn">TRUE</span>, <span class="at">strip.position =</span> <span class="st">"top"</span>) {</span>
|
|||
|
<span id="cb36-3"><a href="#cb36-3" tabindex="-1"></a> </span>
|
|||
|
<span id="cb36-4"><a href="#cb36-4" tabindex="-1"></a> facet <span class="ot"><-</span> <span class="fu">facet_wrap</span>(<span class="sc">~</span>.bootstrap, <span class="at">nrow =</span> nrow, <span class="at">ncol =</span> ncol, <span class="at">scales =</span> scales, </span>
|
|||
|
<span id="cb36-5"><a href="#cb36-5" tabindex="-1"></a> <span class="at">shrink =</span> shrink, <span class="at">strip.position =</span> strip.position)</span>
|
|||
|
<span id="cb36-6"><a href="#cb36-6" tabindex="-1"></a> facet<span class="sc">$</span>params<span class="sc">$</span>n <span class="ot"><-</span> n</span>
|
|||
|
<span id="cb36-7"><a href="#cb36-7" tabindex="-1"></a> facet<span class="sc">$</span>params<span class="sc">$</span>prop <span class="ot"><-</span> prop</span>
|
|||
|
<span id="cb36-8"><a href="#cb36-8" tabindex="-1"></a> <span class="fu">ggproto</span>(<span class="cn">NULL</span>, FacetBootstrap,</span>
|
|||
|
<span id="cb36-9"><a href="#cb36-9" tabindex="-1"></a> <span class="at">shrink =</span> shrink,</span>
|
|||
|
<span id="cb36-10"><a href="#cb36-10" tabindex="-1"></a> <span class="at">params =</span> facet<span class="sc">$</span>params</span>
|
|||
|
<span id="cb36-11"><a href="#cb36-11" tabindex="-1"></a> )</span>
|
|||
|
<span id="cb36-12"><a href="#cb36-12" tabindex="-1"></a>}</span>
|
|||
|
<span id="cb36-13"><a href="#cb36-13" tabindex="-1"></a></span>
|
|||
|
<span id="cb36-14"><a href="#cb36-14" tabindex="-1"></a>FacetBootstrap <span class="ot"><-</span> <span class="fu">ggproto</span>(<span class="st">"FacetBootstrap"</span>, FacetWrap,</span>
|
|||
|
<span id="cb36-15"><a href="#cb36-15" tabindex="-1"></a> <span class="at">compute_layout =</span> <span class="cf">function</span>(data, params) {</span>
|
|||
|
<span id="cb36-16"><a href="#cb36-16" tabindex="-1"></a> id <span class="ot"><-</span> <span class="fu">seq_len</span>(params<span class="sc">$</span>n)</span>
|
|||
|
<span id="cb36-17"><a href="#cb36-17" tabindex="-1"></a></span>
|
|||
|
<span id="cb36-18"><a href="#cb36-18" tabindex="-1"></a> dims <span class="ot"><-</span> <span class="fu">wrap_dims</span>(params<span class="sc">$</span>n, params<span class="sc">$</span>nrow, params<span class="sc">$</span>ncol)</span>
|
|||
|
<span id="cb36-19"><a href="#cb36-19" tabindex="-1"></a> layout <span class="ot"><-</span> <span class="fu">data.frame</span>(<span class="at">PANEL =</span> <span class="fu">factor</span>(id))</span>
|
|||
|
<span id="cb36-20"><a href="#cb36-20" tabindex="-1"></a></span>
|
|||
|
<span id="cb36-21"><a href="#cb36-21" tabindex="-1"></a> <span class="cf">if</span> (params<span class="sc">$</span>as.table) {</span>
|
|||
|
<span id="cb36-22"><a href="#cb36-22" tabindex="-1"></a> layout<span class="sc">$</span>ROW <span class="ot"><-</span> <span class="fu">as.integer</span>((id <span class="sc">-</span> 1L) <span class="sc">%/%</span> dims[<span class="dv">2</span>] <span class="sc">+</span> 1L)</span>
|
|||
|
<span id="cb36-23"><a href="#cb36-23" tabindex="-1"></a> } <span class="cf">else</span> {</span>
|
|||
|
<span id="cb36-24"><a href="#cb36-24" tabindex="-1"></a> layout<span class="sc">$</span>ROW <span class="ot"><-</span> <span class="fu">as.integer</span>(dims[<span class="dv">1</span>] <span class="sc">-</span> (id <span class="sc">-</span> 1L) <span class="sc">%/%</span> dims[<span class="dv">2</span>])</span>
|
|||
|
<span id="cb36-25"><a href="#cb36-25" tabindex="-1"></a> }</span>
|
|||
|
<span id="cb36-26"><a href="#cb36-26" tabindex="-1"></a> layout<span class="sc">$</span>COL <span class="ot"><-</span> <span class="fu">as.integer</span>((id <span class="sc">-</span> 1L) <span class="sc">%%</span> dims[<span class="dv">2</span>] <span class="sc">+</span> 1L)</span>
|
|||
|
<span id="cb36-27"><a href="#cb36-27" tabindex="-1"></a></span>
|
|||
|
<span id="cb36-28"><a href="#cb36-28" tabindex="-1"></a> layout <span class="ot"><-</span> layout[<span class="fu">order</span>(layout<span class="sc">$</span>PANEL), , drop <span class="ot">=</span> <span class="cn">FALSE</span>]</span>
|
|||
|
<span id="cb36-29"><a href="#cb36-29" tabindex="-1"></a> <span class="fu">rownames</span>(layout) <span class="ot"><-</span> <span class="cn">NULL</span></span>
|
|||
|
<span id="cb36-30"><a href="#cb36-30" tabindex="-1"></a></span>
|
|||
|
<span id="cb36-31"><a href="#cb36-31" tabindex="-1"></a> <span class="co"># Add scale identification</span></span>
|
|||
|
<span id="cb36-32"><a href="#cb36-32" tabindex="-1"></a> layout<span class="sc">$</span>SCALE_X <span class="ot"><-</span> <span class="cf">if</span> (params<span class="sc">$</span>free<span class="sc">$</span>x) id <span class="cf">else</span> 1L</span>
|
|||
|
<span id="cb36-33"><a href="#cb36-33" tabindex="-1"></a> layout<span class="sc">$</span>SCALE_Y <span class="ot"><-</span> <span class="cf">if</span> (params<span class="sc">$</span>free<span class="sc">$</span>y) id <span class="cf">else</span> 1L</span>
|
|||
|
<span id="cb36-34"><a href="#cb36-34" tabindex="-1"></a></span>
|
|||
|
<span id="cb36-35"><a href="#cb36-35" tabindex="-1"></a> <span class="fu">cbind</span>(layout, <span class="at">.bootstrap =</span> id)</span>
|
|||
|
<span id="cb36-36"><a href="#cb36-36" tabindex="-1"></a> },</span>
|
|||
|
<span id="cb36-37"><a href="#cb36-37" tabindex="-1"></a> <span class="at">map_data =</span> <span class="cf">function</span>(data, layout, params) {</span>
|
|||
|
<span id="cb36-38"><a href="#cb36-38" tabindex="-1"></a> <span class="cf">if</span> (<span class="fu">is.null</span>(data) <span class="sc">||</span> <span class="fu">nrow</span>(data) <span class="sc">==</span> <span class="dv">0</span>) {</span>
|
|||
|
<span id="cb36-39"><a href="#cb36-39" tabindex="-1"></a> <span class="fu">return</span>(<span class="fu">cbind</span>(data, <span class="at">PANEL =</span> <span class="fu">integer</span>(<span class="dv">0</span>)))</span>
|
|||
|
<span id="cb36-40"><a href="#cb36-40" tabindex="-1"></a> }</span>
|
|||
|
<span id="cb36-41"><a href="#cb36-41" tabindex="-1"></a> n_samples <span class="ot"><-</span> <span class="fu">round</span>(<span class="fu">nrow</span>(data) <span class="sc">*</span> params<span class="sc">$</span>prop)</span>
|
|||
|
<span id="cb36-42"><a href="#cb36-42" tabindex="-1"></a> new_data <span class="ot"><-</span> <span class="fu">lapply</span>(<span class="fu">seq_len</span>(params<span class="sc">$</span>n), <span class="cf">function</span>(i) {</span>
|
|||
|
<span id="cb36-43"><a href="#cb36-43" tabindex="-1"></a> <span class="fu">cbind</span>(data[<span class="fu">sample</span>(<span class="fu">nrow</span>(data), n_samples), , <span class="at">drop =</span> <span class="cn">FALSE</span>], <span class="at">PANEL =</span> i)</span>
|
|||
|
<span id="cb36-44"><a href="#cb36-44" tabindex="-1"></a> })</span>
|
|||
|
<span id="cb36-45"><a href="#cb36-45" tabindex="-1"></a> <span class="fu">do.call</span>(rbind, new_data)</span>
|
|||
|
<span id="cb36-46"><a href="#cb36-46" tabindex="-1"></a> }</span>
|
|||
|
<span id="cb36-47"><a href="#cb36-47" tabindex="-1"></a>)</span>
|
|||
|
<span id="cb36-48"><a href="#cb36-48" tabindex="-1"></a></span>
|
|||
|
<span id="cb36-49"><a href="#cb36-49" tabindex="-1"></a><span class="fu">ggplot</span>(diamonds, <span class="fu">aes</span>(carat, price)) <span class="sc">+</span> </span>
|
|||
|
<span id="cb36-50"><a href="#cb36-50" tabindex="-1"></a> <span class="fu">geom_point</span>(<span class="at">alpha =</span> <span class="fl">0.1</span>) <span class="sc">+</span> </span>
|
|||
|
<span id="cb36-51"><a href="#cb36-51" tabindex="-1"></a> <span class="fu">facet_bootstrap</span>(<span class="at">n =</span> <span class="dv">9</span>, <span class="at">prop =</span> <span class="fl">0.05</span>)</span></code></pre></div>
|
|||
|
<p><img src="
|
|||
|
<p>What we are doing above is to intercept the
|
|||
|
<code>compute_layout</code> and <code>map_data</code> methods and
|
|||
|
instead of dividing the data by a variable we randomly assigns rows to a
|
|||
|
panel based on the sampling parameters (<code>n</code> determines the
|
|||
|
number of panels, <code>prop</code> determines the proportion of data in
|
|||
|
each panel). It is important here that the layout returned by
|
|||
|
<code>compute_layout</code> is a valid layout for <code>FacetWrap</code>
|
|||
|
as we are counting on the <code>draw_panel</code> method from
|
|||
|
<code>FacetWrap</code> to do all the work for us. Thus if you want to
|
|||
|
subclass FacetWrap or FacetGrid, make sure you understand the nature of
|
|||
|
their layout specification.</p>
|
|||
|
<div id="exercises-2" class="section level3">
|
|||
|
<h3>Exercises</h3>
|
|||
|
<ol style="list-style-type: decimal">
|
|||
|
<li>Rewrite FacetTrans to take a vector of transformations and create an
|
|||
|
additional panel for each transformation.</li>
|
|||
|
<li>Based on the FacetWrap implementation rewrite FacetTrans to take the
|
|||
|
strip.placement theme setting into account.</li>
|
|||
|
<li>Think about which caveats there are in FacetBootstrap specifically
|
|||
|
related to adding multiple layers with the same data.</li>
|
|||
|
</ol>
|
|||
|
</div>
|
|||
|
</div>
|
|||
|
<div id="creating-new-guides" class="section level2">
|
|||
|
<h2>Creating new guides</h2>
|
|||
|
<p>Guides are closely related to scales and aesthetics, so an important
|
|||
|
part of guides is taking information from the scale and translating it
|
|||
|
to a graphic. This information is passed around inside guides as a
|
|||
|
<code>key</code> dataframe. For existing guides, you can glance at what
|
|||
|
a key contains by using the <code>get_guide_data()</code> function.
|
|||
|
Typical variables you may see in guides are the aesthetic mapped by the
|
|||
|
scale, such as the hexadecimal colours in the example below, what those
|
|||
|
aesthetic represent in the <code>.value</code> column and how they
|
|||
|
should be labelled in the <code>.label</code> column. Sometimes, the
|
|||
|
aesthetic is used in computations. To avoid interpreting the values and
|
|||
|
labels as aesthetics, it is customary to prefix these with
|
|||
|
<code>.</code>.</p>
|
|||
|
<div class="sourceCode" id="cb37"><pre class="sourceCode r"><code class="sourceCode r"><span id="cb37-1"><a href="#cb37-1" tabindex="-1"></a>p <span class="ot"><-</span> <span class="fu">ggplot</span>(mpg, <span class="fu">aes</span>(displ, hwy, <span class="at">colour =</span> drv)) <span class="sc">+</span></span>
|
|||
|
<span id="cb37-2"><a href="#cb37-2" tabindex="-1"></a> <span class="fu">geom_point</span>() <span class="sc">+</span></span>
|
|||
|
<span id="cb37-3"><a href="#cb37-3" tabindex="-1"></a> <span class="fu">scale_colour_discrete</span>(</span>
|
|||
|
<span id="cb37-4"><a href="#cb37-4" tabindex="-1"></a> <span class="at">labels =</span> <span class="fu">c</span>(<span class="st">"4-wheel drive"</span>, <span class="st">"front wheel drive"</span>, <span class="st">"rear wheel drive"</span>)</span>
|
|||
|
<span id="cb37-5"><a href="#cb37-5" tabindex="-1"></a> )</span>
|
|||
|
<span id="cb37-6"><a href="#cb37-6" tabindex="-1"></a></span>
|
|||
|
<span id="cb37-7"><a href="#cb37-7" tabindex="-1"></a><span class="fu">get_guide_data</span>(p, <span class="st">"colour"</span>)</span>
|
|||
|
<span id="cb37-8"><a href="#cb37-8" tabindex="-1"></a><span class="co">#> colour .value .label</span></span>
|
|||
|
<span id="cb37-9"><a href="#cb37-9" tabindex="-1"></a><span class="co">#> 1 #F8766D 4 4-wheel drive</span></span>
|
|||
|
<span id="cb37-10"><a href="#cb37-10" tabindex="-1"></a><span class="co">#> 2 #00BA38 f front wheel drive</span></span>
|
|||
|
<span id="cb37-11"><a href="#cb37-11" tabindex="-1"></a><span class="co">#> 3 #619CFF r rear wheel drive</span></span></code></pre></div>
|
|||
|
<div id="overriding-scale-extraction" class="section level3">
|
|||
|
<h3>Overriding scale extraction</h3>
|
|||
|
<p>Let’s now make a first guide extension by adjusting the guide’s key.
|
|||
|
Axes are most straightforward to extend, because they are the least
|
|||
|
complicated. We’ll build an axis that accepts custom values for the
|
|||
|
guide’s <code>key</code>. We can begin by making a custom ggproto class
|
|||
|
that inherits from the axis guide. An important extension point is the
|
|||
|
<code>extract_key()</code> method, which determines how break
|
|||
|
information is transferred from the scale to the guide. In our class, we
|
|||
|
reject the scale’s reality and substitute our own.</p>
|
|||
|
<div class="sourceCode" id="cb38"><pre class="sourceCode r"><code class="sourceCode r"><span id="cb38-1"><a href="#cb38-1" tabindex="-1"></a>GuideKey <span class="ot"><-</span> <span class="fu">ggproto</span>(</span>
|
|||
|
<span id="cb38-2"><a href="#cb38-2" tabindex="-1"></a> <span class="st">"Guide"</span>, GuideAxis,</span>
|
|||
|
<span id="cb38-3"><a href="#cb38-3" tabindex="-1"></a> </span>
|
|||
|
<span id="cb38-4"><a href="#cb38-4" tabindex="-1"></a> <span class="co"># Some parameters are required, so it is easiest to copy the base Guide's</span></span>
|
|||
|
<span id="cb38-5"><a href="#cb38-5" tabindex="-1"></a> <span class="co"># parameters into our new parameters.</span></span>
|
|||
|
<span id="cb38-6"><a href="#cb38-6" tabindex="-1"></a> <span class="co"># We add a new 'key' parameter for our own guide.</span></span>
|
|||
|
<span id="cb38-7"><a href="#cb38-7" tabindex="-1"></a> <span class="at">params =</span> <span class="fu">c</span>(GuideAxis<span class="sc">$</span>params, <span class="fu">list</span>(<span class="at">key =</span> <span class="cn">NULL</span>)),</span>
|
|||
|
<span id="cb38-8"><a href="#cb38-8" tabindex="-1"></a> </span>
|
|||
|
<span id="cb38-9"><a href="#cb38-9" tabindex="-1"></a> <span class="co"># It is important for guides to have a mapped aesthetic with the correct name</span></span>
|
|||
|
<span id="cb38-10"><a href="#cb38-10" tabindex="-1"></a> <span class="at">extract_key =</span> <span class="cf">function</span>(scale, aesthetic, key, ...) {</span>
|
|||
|
<span id="cb38-11"><a href="#cb38-11" tabindex="-1"></a> key<span class="sc">$</span>aesthetic <span class="ot"><-</span> scale<span class="sc">$</span><span class="fu">map</span>(key<span class="sc">$</span>aesthetic)</span>
|
|||
|
<span id="cb38-12"><a href="#cb38-12" tabindex="-1"></a> <span class="fu">names</span>(key)[<span class="fu">names</span>(key) <span class="sc">==</span> <span class="st">"aesthetic"</span>] <span class="ot"><-</span> aesthetic</span>
|
|||
|
<span id="cb38-13"><a href="#cb38-13" tabindex="-1"></a> key</span>
|
|||
|
<span id="cb38-14"><a href="#cb38-14" tabindex="-1"></a> }</span>
|
|||
|
<span id="cb38-15"><a href="#cb38-15" tabindex="-1"></a>)</span></code></pre></div>
|
|||
|
</div>
|
|||
|
<div id="guide-constructors" class="section level3">
|
|||
|
<h3>Guide constructors</h3>
|
|||
|
<p>Now we can make a guide constructor that creates a custom key to pass
|
|||
|
along on. The <code>new_guide()</code> function instantiates a new guide
|
|||
|
with the given parameters. This function automatically rejects any
|
|||
|
parameters that are not in the class’ <code>params</code> field, so it
|
|||
|
is important to declare these.</p>
|
|||
|
<div class="sourceCode" id="cb39"><pre class="sourceCode r"><code class="sourceCode r"><span id="cb39-1"><a href="#cb39-1" tabindex="-1"></a>guide_key <span class="ot"><-</span> <span class="cf">function</span>(</span>
|
|||
|
<span id="cb39-2"><a href="#cb39-2" tabindex="-1"></a> aesthetic, <span class="at">value =</span> aesthetic, <span class="at">label =</span> <span class="fu">as.character</span>(aesthetic),</span>
|
|||
|
<span id="cb39-3"><a href="#cb39-3" tabindex="-1"></a> ...,</span>
|
|||
|
<span id="cb39-4"><a href="#cb39-4" tabindex="-1"></a> <span class="co"># Standard guide arguments</span></span>
|
|||
|
<span id="cb39-5"><a href="#cb39-5" tabindex="-1"></a> <span class="at">theme =</span> <span class="cn">NULL</span>, <span class="at">title =</span> <span class="fu">waiver</span>(), <span class="at">order =</span> <span class="dv">0</span>, <span class="at">position =</span> <span class="fu">waiver</span>()</span>
|
|||
|
<span id="cb39-6"><a href="#cb39-6" tabindex="-1"></a>) {</span>
|
|||
|
<span id="cb39-7"><a href="#cb39-7" tabindex="-1"></a> </span>
|
|||
|
<span id="cb39-8"><a href="#cb39-8" tabindex="-1"></a> key <span class="ot"><-</span> <span class="fu">data.frame</span>(aesthetic, <span class="at">.value =</span> value, <span class="at">.label =</span> label, ...)</span>
|
|||
|
<span id="cb39-9"><a href="#cb39-9" tabindex="-1"></a> </span>
|
|||
|
<span id="cb39-10"><a href="#cb39-10" tabindex="-1"></a> <span class="fu">new_guide</span>(</span>
|
|||
|
<span id="cb39-11"><a href="#cb39-11" tabindex="-1"></a> <span class="co"># Arguments passed on to the GuideKey$params field</span></span>
|
|||
|
<span id="cb39-12"><a href="#cb39-12" tabindex="-1"></a> <span class="at">key =</span> key, <span class="at">theme =</span> theme, <span class="at">title =</span> title, <span class="at">order =</span> order, <span class="at">position =</span> position,</span>
|
|||
|
<span id="cb39-13"><a href="#cb39-13" tabindex="-1"></a> <span class="co"># Declare which aesthetics are supported</span></span>
|
|||
|
<span id="cb39-14"><a href="#cb39-14" tabindex="-1"></a> <span class="at">available_aes =</span> <span class="fu">c</span>(<span class="st">"x"</span>, <span class="st">"y"</span>),</span>
|
|||
|
<span id="cb39-15"><a href="#cb39-15" tabindex="-1"></a> <span class="co"># Set the guide class</span></span>
|
|||
|
<span id="cb39-16"><a href="#cb39-16" tabindex="-1"></a> <span class="at">super =</span> GuideKey</span>
|
|||
|
<span id="cb39-17"><a href="#cb39-17" tabindex="-1"></a> )</span>
|
|||
|
<span id="cb39-18"><a href="#cb39-18" tabindex="-1"></a>}</span></code></pre></div>
|
|||
|
<p>Our new guide can now be used inside the <code>guides()</code>
|
|||
|
function or as the <code>guide</code> argument in a position scale.</p>
|
|||
|
<div class="sourceCode" id="cb40"><pre class="sourceCode r"><code class="sourceCode r"><span id="cb40-1"><a href="#cb40-1" tabindex="-1"></a><span class="fu">ggplot</span>(mpg, <span class="fu">aes</span>(displ, hwy)) <span class="sc">+</span></span>
|
|||
|
<span id="cb40-2"><a href="#cb40-2" tabindex="-1"></a> <span class="fu">geom_point</span>() <span class="sc">+</span></span>
|
|||
|
<span id="cb40-3"><a href="#cb40-3" tabindex="-1"></a> <span class="fu">scale_x_continuous</span>(</span>
|
|||
|
<span id="cb40-4"><a href="#cb40-4" tabindex="-1"></a> <span class="at">guide =</span> <span class="fu">guide_key</span>(<span class="at">aesthetic =</span> <span class="dv">2</span><span class="sc">:</span><span class="dv">6</span> <span class="sc">+</span> <span class="fl">0.5</span>)</span>
|
|||
|
<span id="cb40-5"><a href="#cb40-5" tabindex="-1"></a> )</span></code></pre></div>
|
|||
|
<p><img src="
|
|||
|
</div>
|
|||
|
<div id="custom-drawings" class="section level3">
|
|||
|
<h3>Custom drawings</h3>
|
|||
|
<p>If we are feeling more adventurous, we can also alter they way guides
|
|||
|
are drawn. The majority of drawing code is in the
|
|||
|
<code>Guide$build_*()</code> methods, which is all orchestrated by the
|
|||
|
<code>Guide$draw()</code> method. For derived guides, such as the custom
|
|||
|
key guide we’re extending here, overriding a
|
|||
|
<code>Guide$build_*()</code> method should be sufficient. If you are
|
|||
|
writing a completely novel guide that does not resemble the structure of
|
|||
|
any existing guide, overriding the <code>Guide$draw()</code> method
|
|||
|
might be wise.</p>
|
|||
|
<p>In this example, we are changing the way the labels are drawn, so we
|
|||
|
should edit the <code>Guide$build_labels()</code> method. We’ll edit the
|
|||
|
method so that the labels are drawn with a <code>colour</code> set in
|
|||
|
the key. In addition to the <code>key</code> and <code>params</code>
|
|||
|
variable we’ve seen before, we now also have an <code>elements</code>
|
|||
|
variable, which is a list of precomputed theme elements. We can use the
|
|||
|
<code>elements$text</code> element to draw a graphical object (grob) in
|
|||
|
the style of axis text. Perhaps the most finicky thing about drawing
|
|||
|
guides is that a lot of settings depend on the guide’s
|
|||
|
<code>position</code> parameter.</p>
|
|||
|
<div class="sourceCode" id="cb41"><pre class="sourceCode r"><code class="sourceCode r"><span id="cb41-1"><a href="#cb41-1" tabindex="-1"></a><span class="co"># Same as before</span></span>
|
|||
|
<span id="cb41-2"><a href="#cb41-2" tabindex="-1"></a>GuideKey <span class="ot"><-</span> <span class="fu">ggproto</span>(</span>
|
|||
|
<span id="cb41-3"><a href="#cb41-3" tabindex="-1"></a> <span class="st">"Guide"</span>, GuideAxis,</span>
|
|||
|
<span id="cb41-4"><a href="#cb41-4" tabindex="-1"></a> <span class="at">params =</span> <span class="fu">c</span>(GuideAxis<span class="sc">$</span>params, <span class="fu">list</span>(<span class="at">key =</span> <span class="cn">NULL</span>)),</span>
|
|||
|
<span id="cb41-5"><a href="#cb41-5" tabindex="-1"></a> <span class="at">extract_key =</span> <span class="cf">function</span>(scale, aesthetic, key, ...) {</span>
|
|||
|
<span id="cb41-6"><a href="#cb41-6" tabindex="-1"></a> key<span class="sc">$</span>aesthetic <span class="ot"><-</span> scale<span class="sc">$</span><span class="fu">map</span>(key<span class="sc">$</span>aesthetic)</span>
|
|||
|
<span id="cb41-7"><a href="#cb41-7" tabindex="-1"></a> <span class="fu">names</span>(key)[<span class="fu">names</span>(key) <span class="sc">==</span> <span class="st">"aesthetic"</span>] <span class="ot"><-</span> aesthetic</span>
|
|||
|
<span id="cb41-8"><a href="#cb41-8" tabindex="-1"></a> key</span>
|
|||
|
<span id="cb41-9"><a href="#cb41-9" tabindex="-1"></a> },</span>
|
|||
|
<span id="cb41-10"><a href="#cb41-10" tabindex="-1"></a> </span>
|
|||
|
<span id="cb41-11"><a href="#cb41-11" tabindex="-1"></a> <span class="co"># New method to draw labels</span></span>
|
|||
|
<span id="cb41-12"><a href="#cb41-12" tabindex="-1"></a> <span class="at">build_labels =</span> <span class="cf">function</span>(key, elements, params) {</span>
|
|||
|
<span id="cb41-13"><a href="#cb41-13" tabindex="-1"></a> position <span class="ot"><-</span> params<span class="sc">$</span>position</span>
|
|||
|
<span id="cb41-14"><a href="#cb41-14" tabindex="-1"></a> <span class="co"># Downstream code expects a list of labels</span></span>
|
|||
|
<span id="cb41-15"><a href="#cb41-15" tabindex="-1"></a> <span class="fu">list</span>(<span class="fu">element_grob</span>(</span>
|
|||
|
<span id="cb41-16"><a href="#cb41-16" tabindex="-1"></a> elements<span class="sc">$</span>text,</span>
|
|||
|
<span id="cb41-17"><a href="#cb41-17" tabindex="-1"></a> <span class="at">label =</span> key<span class="sc">$</span>.label,</span>
|
|||
|
<span id="cb41-18"><a href="#cb41-18" tabindex="-1"></a> <span class="at">x =</span> <span class="cf">switch</span>(position, <span class="at">left =</span> <span class="dv">1</span>, <span class="at">right =</span> <span class="dv">0</span>, key<span class="sc">$</span>x),</span>
|
|||
|
<span id="cb41-19"><a href="#cb41-19" tabindex="-1"></a> <span class="at">y =</span> <span class="cf">switch</span>(position, <span class="at">top =</span> <span class="dv">0</span>, <span class="at">bottom =</span> <span class="dv">1</span>, key<span class="sc">$</span>y),</span>
|
|||
|
<span id="cb41-20"><a href="#cb41-20" tabindex="-1"></a> <span class="at">margin_x =</span> position <span class="sc">%in%</span> <span class="fu">c</span>(<span class="st">"left"</span>, <span class="st">"right"</span>),</span>
|
|||
|
<span id="cb41-21"><a href="#cb41-21" tabindex="-1"></a> <span class="at">margin_y =</span> position <span class="sc">%in%</span> <span class="fu">c</span>(<span class="st">"top"</span>, <span class="st">"bottom"</span>),</span>
|
|||
|
<span id="cb41-22"><a href="#cb41-22" tabindex="-1"></a> <span class="at">colour =</span> key<span class="sc">$</span>colour</span>
|
|||
|
<span id="cb41-23"><a href="#cb41-23" tabindex="-1"></a> ))</span>
|
|||
|
<span id="cb41-24"><a href="#cb41-24" tabindex="-1"></a> }</span>
|
|||
|
<span id="cb41-25"><a href="#cb41-25" tabindex="-1"></a>)</span></code></pre></div>
|
|||
|
<p>Because we are incorporating the <code>...</code> argument to
|
|||
|
<code>guide_key()</code> in the key, adding a <code>colour</code> column
|
|||
|
to the key is straightforward. We can check that are guide looks correct
|
|||
|
in the different positions around the panel.</p>
|
|||
|
<div class="sourceCode" id="cb42"><pre class="sourceCode r"><code class="sourceCode r"><span id="cb42-1"><a href="#cb42-1" tabindex="-1"></a><span class="fu">ggplot</span>(mpg, <span class="fu">aes</span>(displ, hwy)) <span class="sc">+</span></span>
|
|||
|
<span id="cb42-2"><a href="#cb42-2" tabindex="-1"></a> <span class="fu">geom_point</span>() <span class="sc">+</span></span>
|
|||
|
<span id="cb42-3"><a href="#cb42-3" tabindex="-1"></a> <span class="fu">guides</span>(</span>
|
|||
|
<span id="cb42-4"><a href="#cb42-4" tabindex="-1"></a> <span class="at">x =</span> <span class="fu">guide_key</span>(</span>
|
|||
|
<span id="cb42-5"><a href="#cb42-5" tabindex="-1"></a> <span class="at">aesthetic =</span> <span class="dv">2</span><span class="sc">:</span><span class="dv">6</span> <span class="sc">+</span> <span class="fl">0.5</span>,</span>
|
|||
|
<span id="cb42-6"><a href="#cb42-6" tabindex="-1"></a> <span class="at">colour =</span> <span class="fu">c</span>(<span class="st">"red"</span>, <span class="st">"grey"</span>, <span class="st">"red"</span>, <span class="st">"grey"</span>, <span class="st">"red"</span>)</span>
|
|||
|
<span id="cb42-7"><a href="#cb42-7" tabindex="-1"></a> ),</span>
|
|||
|
<span id="cb42-8"><a href="#cb42-8" tabindex="-1"></a> <span class="at">x.sec =</span> <span class="fu">guide_key</span>(</span>
|
|||
|
<span id="cb42-9"><a href="#cb42-9" tabindex="-1"></a> <span class="at">aesthetic =</span> <span class="fu">c</span>(<span class="dv">2</span>, <span class="dv">4</span>, <span class="dv">6</span>), </span>
|
|||
|
<span id="cb42-10"><a href="#cb42-10" tabindex="-1"></a> <span class="at">colour =</span> <span class="fu">c</span>(<span class="st">"tomato"</span>, <span class="st">"limegreen"</span>, <span class="st">"dodgerblue"</span>)</span>
|
|||
|
<span id="cb42-11"><a href="#cb42-11" tabindex="-1"></a> )</span>
|
|||
|
<span id="cb42-12"><a href="#cb42-12" tabindex="-1"></a> )</span></code></pre></div>
|
|||
|
<p><img src="
|
|||
|
</div>
|
|||
|
<div id="exercises-3" class="section level3">
|
|||
|
<h3>Exercises</h3>
|
|||
|
<ul>
|
|||
|
<li>Extend <code>guide_key()</code> to also pass on <code>family</code>,
|
|||
|
<code>face</code> and <code>size</code> aesthetics from the key to the
|
|||
|
labels.</li>
|
|||
|
<li>Override the <code>GuideKey$build_ticks()</code> method to also pass
|
|||
|
on <code>colour</code> and <code>linewidth</code> settings to the tick
|
|||
|
marks. Looking at <code>Guide$build_ticks()</code> is a good starting
|
|||
|
point.</li>
|
|||
|
<li>Compare <code>GuideKey$extract_key()</code> to
|
|||
|
<code>Guide$extract_key()</code>. What steps have been skimmed over in
|
|||
|
the example?</li>
|
|||
|
</ul>
|
|||
|
</div>
|
|||
|
</div>
|
|||
|
|
|||
|
|
|||
|
|
|||
|
<!-- code folding -->
|
|||
|
|
|||
|
|
|||
|
<!-- dynamically load mathjax for compatibility with self-contained -->
|
|||
|
<script>
|
|||
|
(function () {
|
|||
|
var script = document.createElement("script");
|
|||
|
script.type = "text/javascript";
|
|||
|
script.src = "https://mathjax.rstudio.com/latest/MathJax.js?config=TeX-AMS-MML_HTMLorMML";
|
|||
|
document.getElementsByTagName("head")[0].appendChild(script);
|
|||
|
})();
|
|||
|
</script>
|
|||
|
|
|||
|
</body>
|
|||
|
</html>
|