Title: | Create Ternary and Holdridge Plots |
---|---|
Description: | Plots ternary diagrams (simplex plots / Gibbs triangles) and Holdridge life zone plots <doi:10.1126/science.105.2727.367> using the standard graphics functions. Allows custom annotation, interpolating, contouring and scaling of plotting region. Includes a 'Shiny' user interface for point-and-click ternary plotting. An alternative to 'ggtern', which uses the 'ggplot2' family of plotting functions. |
Authors: | Martin R. Smith [aut, cre, cph] , Lilian Sanselme [ctb] |
Maintainer: | Martin R. Smith <[email protected]> |
License: | GPL (>= 2) |
Version: | 2.3.3.9000 |
Built: | 2025-01-19 05:32:08 UTC |
Source: | https://github.com/ms609/ternary |
Plot shapes onto a ternary diagram created with TernaryPlot()
,
or a Holdridge plot created with HoldridgePlot()
.
AddToHoldridge(PlottingFunction, pet, prec, ...) HoldridgeArrows(fromCoordinates, toCoordinates = fromCoordinates, ...) HoldridgeLines(pet, prec, ...) HoldridgePoints(pet, prec, ...) HoldridgePolygon(pet, prec, ...) HoldridgeText(pet, prec, ...) AddToTernary(PlottingFunction, coordinates, ...) TernarySegments(fromCoordinates, toCoordinates = fromCoordinates, ...) TernaryArrows(fromCoordinates, toCoordinates = fromCoordinates, ...) TernaryLines(coordinates, ...) TernaryPoints(coordinates, ...) TernaryPolygon(coordinates, ...) TernaryText(coordinates, ...) JoinTheDots(coordinates, ...)
AddToHoldridge(PlottingFunction, pet, prec, ...) HoldridgeArrows(fromCoordinates, toCoordinates = fromCoordinates, ...) HoldridgeLines(pet, prec, ...) HoldridgePoints(pet, prec, ...) HoldridgePolygon(pet, prec, ...) HoldridgeText(pet, prec, ...) AddToTernary(PlottingFunction, coordinates, ...) TernarySegments(fromCoordinates, toCoordinates = fromCoordinates, ...) TernaryArrows(fromCoordinates, toCoordinates = fromCoordinates, ...) TernaryLines(coordinates, ...) TernaryPoints(coordinates, ...) TernaryPolygon(coordinates, ...) TernaryText(coordinates, ...) JoinTheDots(coordinates, ...)
PlottingFunction |
Function to add data to a plot; perhaps one of
|
pet , prec
|
Numeric vectors giving potential evapotranspiration ratio and annual precipitation (in mm). |
... |
Additional parameters to pass to |
fromCoordinates , toCoordinates
|
For |
coordinates |
A list, matrix, data.frame or vector in which each element (or row) specifies the three coordinates of a point in ternary space. |
HoldridgeArrows()
: Add arrows to Holdridge plot
HoldridgeLines()
: Add lines to Holdridge plot
HoldridgePoints()
: Add points to Holdridge plot
HoldridgePolygon()
: Add polygons to Holdridge
plot
HoldridgeText()
: Add text to Holdridge plot
TernarySegments()
: Add segments
TernaryArrows()
: Add arrows
TernaryLines()
: Add lines
TernaryPoints()
: Add points
TernaryPolygon()
: Add polygons
TernaryText()
: Add text
JoinTheDots()
: Add points, joined by lines
Martin R. Smith ([email protected])
Other Holdridge plotting functions:
HoldridgeHypsometricCol()
,
HoldridgePlot()
,
holdridge
,
holdridgeClasses
# Data to plot coords <- list( A = c(1, 0, 2), B = c(1, 1, 1), C = c(1.5, 1.5, 0), D = c(0.5, 1.5, 1) ) # Set up plot oPar <- par(mar = rep(0, 4), xpd = NA) # reduce margins and write in them TernaryPlot() # Add elements to ternary diagram AddToTernary(lines, coords, col = "darkgreen", lty = "dotted", lwd = 3) TernaryLines(coords, col = "darkgreen") TernaryArrows(coords[1], coords[2:4], col = "orange", length = 0.2, lwd = 1) TernaryText(coords, cex = 0.8, col = "red", font = 2) seeThruBlue <- rgb(0, 0.2, 1, alpha = 0.8) TernaryPoints(coords, pch = 1, cex = 2, col = seeThruBlue) AddToTernary(graphics::points, coords, pch = 1, cex = 3) # An equivalent syntax applies to Holdridge plots: HoldridgePlot() pet <- c(0.8, 2, 0.42) prec <- c(250, 400, 1337) HoldridgeText(pet, prec, c("A", "B", "C")) AddToHoldridge(graphics::points, pet, prec, cex = 3) # Restore original plotting parameters par(oPar)
# Data to plot coords <- list( A = c(1, 0, 2), B = c(1, 1, 1), C = c(1.5, 1.5, 0), D = c(0.5, 1.5, 1) ) # Set up plot oPar <- par(mar = rep(0, 4), xpd = NA) # reduce margins and write in them TernaryPlot() # Add elements to ternary diagram AddToTernary(lines, coords, col = "darkgreen", lty = "dotted", lwd = 3) TernaryLines(coords, col = "darkgreen") TernaryArrows(coords[1], coords[2:4], col = "orange", length = 0.2, lwd = 1) TernaryText(coords, cex = 0.8, col = "red", font = 2) seeThruBlue <- rgb(0, 0.2, 1, alpha = 0.8) TernaryPoints(coords, pch = 1, cex = 2, col = seeThruBlue) AddToTernary(graphics::points, coords, pch = 1, cex = 3) # An equivalent syntax applies to Holdridge plots: HoldridgePlot() pet <- c(0.8, 2, 0.42) prec <- c(250, 400, 1337) HoldridgeText(pet, prec, c("A", "B", "C")) AddToHoldridge(graphics::points, pet, prec, cex = 3) # Restore original plotting parameters par(oPar)
Annotate()
identifies and label individual points on a ternary diagram
in the plot margins.
Annotate( coordinates, labels, side, outset = 0.16, line.col = col, lty = par("lty"), lwd = par("lwd"), col = par("col"), font = par("font"), offset = 0.5, ... )
Annotate( coordinates, labels, side, outset = 0.16, line.col = col, lty = par("lty"), lwd = par("lwd"), col = par("col"), font = par("font"), offset = 0.5, ... )
coordinates |
A list, matrix, data.frame or vector in which each element (or row) specifies the three coordinates of a point in ternary space. |
labels |
Character vector specifying text with which to annotate
each entry in |
side |
Optional vector specifying which side of the ternary
plot each point should be labelled on, using the notation |
outset |
Numeric specifying distance from plot margins to labels. |
line.col , lty , lwd
|
parameters to |
col , font , offset
|
parameters to |
... |
Further parameters to |
Martin R. Smith ([email protected])
Annotation vignette gives further suggestions for manual annotation.
# Load some data data("Seatbelts") seats <- c("drivers", "front", "rear") seat <- Seatbelts[month.abb %in% "Oct", seats] law <- Seatbelts[month.abb %in% "Oct", "law"] # Set up plot oPar <- par(mar = c(2, 0, 0, 0)) TernaryPlot(alab = seats[1], blab = seats[2], clab = seats[3]) TernaryPoints(seat, cex = 0.8, col = 2 + law) # Annotate points by year Annotate(seat, labels = 1969:1984, col = 2 + law) # Restore original graphical parameters par(oPar)
# Load some data data("Seatbelts") seats <- c("drivers", "front", "rear") seat <- Seatbelts[month.abb %in% "Oct", seats] law <- Seatbelts[month.abb %in% "Oct", "law"] # Set up plot oPar <- par(mar = c(2, 0, 0, 0)) TernaryPlot(alab = seats[1], blab = seats[2], clab = seats[3]) TernaryPoints(seat, cex = 0.8, col = 2 + law) # Annotate points by year Annotate(seat, labels = 1969:1984, col = 2 + law) # Restore original graphical parameters par(oPar)
Colour palettes recommended for use with colour blind audiences.
cbPalette8 cbPalette13 cbPalette15
cbPalette8 cbPalette13 cbPalette15
Character vectors of lengths 8, 13 and 15.
An object of class character
of length 8.
An object of class character
of length 13.
An object of class character
of length 15.
Since R 4.0, cbPalette8
is available in base R as palette.colors(8)
.
cbPalette15
is a Brewer palette.
Because colours 4 and 7 are difficult to distinguish from colours 13 and 3,
respectively, in individuals with tritanopia, cbPalette13
omits these
colours (i.e. cbPalette13 <- cbPalette15[-c(4, 7)]
).
cbPalette8
: Wong B. 2011. Color blindness. Nat. Methods. 8:441.
doi:10.1038/nmeth.1618
cbPalette15
: http://mkweb.bcgsc.ca/biovis2012/color-blindness-palette.png
data("cbPalette8") plot.new() plot.window(xlim = c(1, 16), ylim = c(0, 3)) text(1:8 * 2, 3, 1:8, col = cbPalette8) points(1:8 * 2, rep(2, 8), col = cbPalette8, pch = 15) data("cbPalette15") text(1:15, 1, col = cbPalette15) text(c(4, 7), 1, "[ ]") points(1:15, rep(0, 15), col = cbPalette15, pch = 15)
data("cbPalette8") plot.new() plot.window(xlim = c(1, 16), ylim = c(0, 3)) text(1:8 * 2, 3, 1:8, col = cbPalette8) points(1:8 * 2, rep(2, 8), col = cbPalette8, pch = 15) data("cbPalette15") text(1:15, 1, col = cbPalette15) text(c(4, 7), 1, "[ ]") points(1:15, rep(0, 15), col = cbPalette15, pch = 15)
Colour a ternary plot according to the output of a function
ColourTernary( values, spectrum = hcl.colors(256L, palette = "viridis", alpha = 0.6), resolution = sqrt(ncol(values)), direction = getOption("ternDirection", 1L), legend, ... ) ColorTernary( values, spectrum = hcl.colors(256L, palette = "viridis", alpha = 0.6), resolution = sqrt(ncol(values)), direction = getOption("ternDirection", 1L), legend, ... )
ColourTernary( values, spectrum = hcl.colors(256L, palette = "viridis", alpha = 0.6), resolution = sqrt(ncol(values)), direction = getOption("ternDirection", 1L), legend, ... ) ColorTernary( values, spectrum = hcl.colors(256L, palette = "viridis", alpha = 0.6), resolution = sqrt(ncol(values)), direction = getOption("ternDirection", 1L), legend, ... )
values |
Numeric matrix, possibly created using
|
spectrum |
Vector of colours to use as a spectrum, or |
resolution |
The number of triangles whose base should lie on the longest axis of the triangle. Higher numbers will result in smaller subdivisions and smoother colour gradients, but at a computational cost. |
direction |
(optional) Integer specifying the direction that the current ternary plot should point: 1, up; 2, right; 3, down; 4, left. |
legend |
Character vector specifying annotations for colour scale.
If not provided, no colour legend is displayed.
Specify |
... |
Further arguments to
|
Martin R. Smith ([email protected])
Fine control over continuous legends:
PlotTools::SpectrumLegend()
Other contour plotting functions:
TernaryContour()
,
TernaryDensityContour()
,
TernaryPointValues()
Other functions for colouring and shading:
TernaryTiles()
TernaryPlot(alab = "a", blab = "b", clab = "c") FunctionToContour <- function (a, b, c) { a - c + (4 * a * b) + (27 * a * b * c) } values <- TernaryPointValues(FunctionToContour, resolution = 24L) ColourTernary( values, x = "topleft", bty = "n", # No box legend = signif(seq(max(values), min(values), length.out = 4), 3) ) TernaryContour(FunctionToContour, resolution = 36L) TernaryPlot() values <- TernaryPointValues(rgb, resolution = 20) ColourTernary(values, spectrum = NULL) # Create a helper function to place white centrally: rgbWhite <- function (r, g, b) { highest <- apply(rbind(r, g, b), 2L, max) rgb(r/highest, g/highest, b/highest) } TernaryPlot() values <- TernaryPointValues(rgbWhite, resolution = 20) ColourTernary(values, spectrum = NULL)
TernaryPlot(alab = "a", blab = "b", clab = "c") FunctionToContour <- function (a, b, c) { a - c + (4 * a * b) + (27 * a * b * c) } values <- TernaryPointValues(FunctionToContour, resolution = 24L) ColourTernary( values, x = "topleft", bty = "n", # No box legend = signif(seq(max(values), min(values), length.out = 4), 3) ) TernaryContour(FunctionToContour, resolution = 36L) TernaryPlot() values <- TernaryPointValues(rgb, resolution = 20) ColourTernary(values, spectrum = NULL) # Create a helper function to place white centrally: rgbWhite <- function (r, g, b) { highest <- apply(rbind(r, g, b), 2L, max) rgb(r/highest, g/highest, b/highest) } TernaryPlot() values <- TernaryPointValues(rgbWhite, resolution = 20) ColourTernary(values, spectrum = NULL)
A stratified random sampling (average of 100 points) using a global mapping of Holdridge’s scheme.
holdridge
holdridge
An object of class data.frame
with 39 rows and 4 columns.
James Lee Tsakalos
Other Holdridge plotting functions:
AddToHoldridge()
,
HoldridgeHypsometricCol()
,
HoldridgePlot()
,
holdridgeClasses
data("holdridge", package = "Ternary") head(holdridge)
data("holdridge", package = "Ternary") head(holdridge)
holdridgeClasses
is a character vector naming, from left to right,
top to bottom, the 38 classes defined by the International Institute for
Applied Systems Analysis (IIASA).
holdridgeClasses holdridgeLifeZones holdridgeLifeZonesUp holdridgeClassesUp
holdridgeClasses holdridgeLifeZones holdridgeLifeZonesUp holdridgeClassesUp
An object of class character
of length 38.
An object of class character
of length 33.
An object of class character
of length 33.
An object of class character
of length 38.
holdridgeLifeZones
is a character vector naming, from left to right,
top to bottom, the 38 cells of the Holdridge classification plot.
holdridgeClassesUp
and holdridgeLifeZonesUp
replace spaces with new
lines, for more legible plotting with HoldridgeHexagons()
.
Martin R. Smith ([email protected])
Holdridge (1947), "Determination of world plant formations from simple climatic data", Science 105:367–368. doi:10.1126/science.105.2727.367
Holdridge (1967), Life zone ecology. Tropical Science Center, San José.
Leemans, R. (1990), "Possible change in natural vegetation patterns due to a global warming", International Institute for Applied Systems Analysis Working paper WP-90-08. https://pure.iiasa.ac.at/id/eprint/3443/1/WP-90-008.pdf
Other Holdridge plotting functions:
AddToHoldridge()
,
HoldridgeHypsometricCol()
,
HoldridgePlot()
,
holdridge
Used to colour HoldridgeHexagons()
, and may also be used to aid the
interpretation of PET + precipitation data in any graphical context.
HoldridgeHypsometricCol(pet, prec, opacity = NA)
HoldridgeHypsometricCol(pet, prec, opacity = NA)
pet , prec
|
Numeric vectors giving potential evapotranspiration ratio and annual precipitation (in mm). |
opacity |
Opacity level to be converted to the final two characters
of an RGBA hexadecimal colour definition, e.g. |
Character vector listing RGB or (if opacity != NA
)
RGBA values corresponding to each PET-precipitation value pair.
Martin R. Smith ([email protected])
Palette derived from the hypsometric colour scheme presented at Shaded Relief.
Other Holdridge plotting functions:
AddToHoldridge()
,
HoldridgePlot()
,
holdridge
,
holdridgeClasses
HoldridgePlot(hex.col = HoldridgeHypsometricCol) VeryTransparent <- function(...) HoldridgeHypsometricCol(..., opacity = 0.3) HoldridgePlot(hex.col = VeryTransparent) pet <- holdridge$PET prec <- holdridge$Precipitation ptCol <- HoldridgeHypsometricCol(pet, prec) HoldridgePoints(pet, prec, pch = 21, bg = ptCol)
HoldridgePlot(hex.col = HoldridgeHypsometricCol) VeryTransparent <- function(...) HoldridgeHypsometricCol(..., opacity = 0.3) HoldridgePlot(hex.col = VeryTransparent) pet <- holdridge$PET prec <- holdridge$Precipitation ptCol <- HoldridgeHypsometricCol(pet, prec) HoldridgePoints(pet, prec, pch = 21, bg = ptCol)
HoldridgePlot()
creates a blank triangular plot, as proposed by
Holdridge (1947, 1967), onto which potential evapotranspiration
(PET) ratio and annual precipitation data can be plotted
(using the AddToHoldridge()
family of functions) in order to interpret
climatic life zones.
HoldridgePlot( atip = NULL, btip = NULL, ctip = NULL, alab = "Potential evapotranspiration ratio", blab = "Annual precipitation / mm", clab = "Humidity province", lab.offset = 0.22, lab.col = c("#D81B60", "#1E88E5", "#111111"), xlim = NULL, ylim = NULL, region = NULL, lab.cex = 1, lab.font = 0, tip.cex = lab.cex, tip.font = 2, tip.col = "black", isometric = TRUE, atip.rotate = NULL, btip.rotate = NULL, ctip.rotate = NULL, atip.pos = NULL, btip.pos = NULL, ctip.pos = NULL, padding = 0.16, col = NA, panel.first = NULL, panel.last = NULL, grid.lines = 8, grid.col = c(NA, "#1E88E5", "#D81B60"), grid.lty = "solid", grid.lwd = par("lwd"), grid.minor.lines = 0, grid.minor.col = "lightgrey", grid.minor.lty = "solid", grid.minor.lwd = par("lwd"), hex.border = "#888888", hex.col = HoldridgeHypsometricCol, hex.lty = "solid", hex.lwd = par("lwd"), hex.cex = 0.5, hex.labels = NULL, hex.font = NULL, hex.text.col = "black", axis.cex = 0.8, axis.col = c(grid.col[2], grid.col[3], NA), axis.font = par("font"), axis.labels = TRUE, axis.lty = "solid", axis.lwd = 1, axis.rotate = TRUE, axis.pos = NULL, axis.tick = TRUE, ticks.lwd = axis.lwd, ticks.length = 0.025, ticks.col = grid.col, ... ) HoldridgeBelts( grid.col = "#004D40", grid.lty = "dotted", grid.lwd = par("lwd") ) HoldridgeHexagons( border = "#004D40", hex.col = HoldridgeHypsometricCol, lty = "dotted", lwd = par("lwd"), labels = NULL, cex = 1, text.col = NULL, font = NULL )
HoldridgePlot( atip = NULL, btip = NULL, ctip = NULL, alab = "Potential evapotranspiration ratio", blab = "Annual precipitation / mm", clab = "Humidity province", lab.offset = 0.22, lab.col = c("#D81B60", "#1E88E5", "#111111"), xlim = NULL, ylim = NULL, region = NULL, lab.cex = 1, lab.font = 0, tip.cex = lab.cex, tip.font = 2, tip.col = "black", isometric = TRUE, atip.rotate = NULL, btip.rotate = NULL, ctip.rotate = NULL, atip.pos = NULL, btip.pos = NULL, ctip.pos = NULL, padding = 0.16, col = NA, panel.first = NULL, panel.last = NULL, grid.lines = 8, grid.col = c(NA, "#1E88E5", "#D81B60"), grid.lty = "solid", grid.lwd = par("lwd"), grid.minor.lines = 0, grid.minor.col = "lightgrey", grid.minor.lty = "solid", grid.minor.lwd = par("lwd"), hex.border = "#888888", hex.col = HoldridgeHypsometricCol, hex.lty = "solid", hex.lwd = par("lwd"), hex.cex = 0.5, hex.labels = NULL, hex.font = NULL, hex.text.col = "black", axis.cex = 0.8, axis.col = c(grid.col[2], grid.col[3], NA), axis.font = par("font"), axis.labels = TRUE, axis.lty = "solid", axis.lwd = 1, axis.rotate = TRUE, axis.pos = NULL, axis.tick = TRUE, ticks.lwd = axis.lwd, ticks.length = 0.025, ticks.col = grid.col, ... ) HoldridgeBelts( grid.col = "#004D40", grid.lty = "dotted", grid.lwd = par("lwd") ) HoldridgeHexagons( border = "#004D40", hex.col = HoldridgeHypsometricCol, lty = "dotted", lwd = par("lwd"), labels = NULL, cex = 1, text.col = NULL, font = NULL )
atip , btip , ctip
|
Character string specifying text to title corners,
proceeding clockwise from the corner specified in |
alab , blab , clab
|
Character string specifying text with which to label
the corresponding sides of the triangle.
Left or right-pointing arrows are produced by
typing |
lab.offset |
Numeric specifying distance between midpoint of axis label
and the axis. The default value is given in the 'Usage' section; a value
of |
lab.col |
Character vector specifying colours for axis labels. Use a vector of length three to specify a different colour for each label. |
xlim , ylim
|
Numeric vectors of length two specifying the minimum and
maximum x and y limits of the plotted area, to which |
region |
(optional) Named list of length two specifying the the
|
lab.cex , tip.cex
|
Numeric specifying character expansion (font size) for axis labels. Use a vector of length three to specify a different value for each direction. |
lab.font , tip.font
|
Numeric specifying font style (Roman, bold, italic, bold-italic) for axis titles. Use a vector of length three to set a different font for each direction. |
isometric |
Logical specifying whether to enforce an equilateral shape
for the ternary plot.
If only one of |
atip.rotate , btip.rotate , ctip.rotate
|
Integer specifying number of degrees to rotate label of rightmost apex. |
atip.pos , btip.pos , ctip.pos
|
Integer specifying positioning of labels,
iff the corresponding |
padding |
Numeric specifying size of internal margin of the plot; increase if axis labels are being clipped. |
col |
The colour for filling the plot; see
|
panel.first |
An expression to be evaluated after the plot axes are
set up but before any plotting takes place.
This can be useful for drawing backgrounds, e.g. with |
panel.last |
An expression to be evaluated after plotting has taken
place but before the axes and box are added. See the comments about
|
grid.lines |
Integer specifying the number of grid lines to plot.
If |
grid.col , grid.minor.col
|
Colours to draw the grid lines. Use a vector of length three to set different values for each direction. |
grid.lty , grid.minor.lty
|
Character or integer vector; line type of the grid lines. Use a vector of length three to set different values for each direction. |
grid.lwd , grid.minor.lwd
|
Non-negative numeric giving line width of the grid lines. Use a vector of length three to set different values for each direction. |
grid.minor.lines |
Integer specifying the number of minor (unlabelled) grid lines to plot between each major pair. |
hex.border , hex.lty , hex.lwd
|
Parameters to pass to
|
hex.col |
Fill colour for hexagons. Provide a vector specifying a
colour for each hexagon in turn, reading from left to right and top to
bottom, or a function that accepts two arguments, numerics |
hex.cex , hex.font , hex.text.col
|
Parameters passed to
|
hex.labels |
38-element character vector specifying label for each hexagonal class, from top left to bottom right. |
axis.cex |
Numeric specifying character expansion (font size) for axis labels. Use a vector of length three to set a different value for each direction. |
axis.col , ticks.col , tip.col
|
Colours for the axis line,
tick marks and tip labels respectively.
Use a vector of length three to set a different value for each direction.
|
axis.font |
Font for text. Defaults to |
axis.labels |
This can either be a logical value specifying whether (numerical) annotations are to be made at the tickmarks, or a character or expression vector of labels to be placed at the tick points, or a list of length three, with each entry specifying labels to be placed on each axis in turn. |
axis.lty |
Line type for both the axis line and tick marks. Use a vector of length three to set a different value for each direction. |
axis.lwd , ticks.lwd
|
Line width for the axis line and tick marks. Zero or negative values will suppress the line or ticks. Use a vector of length three to set different values for each axis. |
axis.rotate |
Logical specifying whether to rotate axis labels
to parallel grid lines, or numeric specifying custom rotation for each axis,
to be passed as |
axis.pos |
Vector of length one or three specifying position of axis
labels, to be passed as |
axis.tick |
Logical specifying whether to mark the axes with tick marks. |
ticks.length |
Numeric specifying distance that ticks should extend beyond the plot margin. Also affects position of axis labels, which are plotted at the end of each tick. Use a vector of length three to set a different length for each direction. |
... |
Additional parameters to |
border |
Colour to use for hexagon borders. |
lty , lwd , cex , font
|
Graphical parameters specifying properties of hexagons to be plotted. |
labels |
Vector specifying labels for life zone hexagons to be plotted.
Suggested values: |
text.col |
Colour of text to be printed in hexagons. |
HoldridgePoints()
, HoldridgeText()
and related functions allow data
points to be added to an existing plot; AddToHoldridge()
allows plotting
using any of the standard plotting functions.
HoldridgeBelts()
and HoldridgeHexagons()
plot interpretative lines
and hexagons allowing plotted data to be linked to interpreted climate
settings.
Please cite Tsakalos et al. (2023) when using this function.
Martin R. Smith ([email protected])
Holdridge (1947), "Determination of world plant formations from simple climatic data", Science 105:367–368. doi:10.1126/science.105.2727.367
Holdridge (1967), Life zone ecology. Tropical Science Center, San José
Tsakalos, Smith, Luebert & Mucina (2023). "climenv: Download, extract and visualise climatic and elevation data.", Journal of Vegetation Science 6:e13215. doi:10.1111/jvs.13215
Other Holdridge plotting functions:
AddToHoldridge()
,
HoldridgeHypsometricCol()
,
holdridge
,
holdridgeClasses
data(holdridgeLifeZonesUp, package = "Ternary") HoldridgePlot(hex.labels = holdridgeLifeZonesUp) HoldridgeBelts()
data(holdridgeLifeZonesUp, package = "Ternary") HoldridgePlot(hex.labels = holdridgeLifeZonesUp) HoldridgeBelts()
Evaluate whether a given set of coordinates lie outwith the boundaries of a plotted ternary diagram.
OutsidePlot(x, y, tolerance = 0)
OutsidePlot(x, y, tolerance = 0)
x , y
|
Vectors of x and y coordinates of points. |
tolerance |
Consider points this close to the edge of the plot to be inside. Set to negative values to count points that are just outside the plot as inside, and to positive values to count points that are just inside the margins as outside. Maximum positive value: 1/3. |
OutsidePlot()
returns a logical vector specifying whether each
pair of x and y coordinates corresponds to a point outside the plotted
ternary diagram.
Martin R. Smith ([email protected])
Other plot limits:
TernaryXRange()
TernaryPlot() points(0.5, 0.5, col = "darkgreen") OutsidePlot(0.5, 0.5) points(0.1, 0.5, col = "red") OutsidePlot(0.1, 0.5) OutsidePlot(c(0.5, 0.1), 0.5)
TernaryPlot() points(0.5, 0.5, col = "darkgreen") OutsidePlot(0.5, 0.5) points(0.1, 0.5, col = "red") OutsidePlot(0.1, 0.5) OutsidePlot(c(0.5, 0.1), 0.5)
Geometry functions for irregular polygons.
PolygonArea(x, y = NULL, positive = TRUE) PolygonCentre(x, y = NULL) PolygonCenter(x, y = NULL) GrowPolygon(x, y = NULL, buffer = 0)
PolygonArea(x, y = NULL, positive = TRUE) PolygonCentre(x, y = NULL) PolygonCenter(x, y = NULL) GrowPolygon(x, y = NULL, buffer = 0)
x , y
|
Vectors containing the coordinates of the vertices of the polygon. |
positive |
If vertices are specified in an anticlockwise direction,
the polygon will be treated as a hole, with a negative area, unless
|
buffer |
Numeric specifying distance by which to grow polygon. |
PolygonArea()
returns the area of the specified polygon.
PolygonCentre()
returns a single-row matrix containing the
x and y coordinates of the geometric centre of the polygon.
GrowPolygon()
returns coordinates of the vertices of polygon
after moving each vertex buffer
away from the polygon's centre.
PolygonArea()
: Calculate the area of an irregular polygon
PolygonCentre()
: Locate the centre of a polygon
GrowPolygon()
: Enlarge a polygon in all directions
Martin R. Smith ([email protected])
Other tiling functions:
TriangleCentres()
,
TriangleInHull()
x <- c(-3, -1, 6, 3, -4) y <- c(-2, 4, 1, 10, 9) plot(x, y, frame.plot = FALSE) polygon(x, y) PolygonArea(x, y) points(PolygonCentre(x, y), pch = 3, cex = 2) polygon(GrowPolygon(x, y, 1), border = "darkgreen", xpd = NA # Allow drawing beyond plot border ) # Negative values shrink the polygon polygon(GrowPolygon(x, y, -1), border = "red")
x <- c(-3, -1, 6, 3, -4) y <- c(-2, 4, 1, 10, 9) plot(x, y, frame.plot = FALSE) polygon(x, y) PolygonArea(x, y) points(PolygonCentre(x, y), pch = 3, cex = 2) polygon(GrowPolygon(x, y, 1), border = "darkgreen", xpd = NA # Allow drawing beyond plot border ) # Negative values shrink the polygon polygon(GrowPolygon(x, y, -1), border = "red")
To avoid edge effects, it may be desirable to add the value of a point within a ternary plot with the value of its 'reflection' across the nearest axis or corner.
ReflectedEquivalents(x, y, direction = getOption("ternDirection", 1L))
ReflectedEquivalents(x, y, direction = getOption("ternDirection", 1L))
x , y
|
Vectors of x and y coordinates of points. |
direction |
(optional) Integer specifying the direction that the current ternary plot should point: 1, up; 2, right; 3, down; 4, left. |
ReflectedEquivalents()
returns a list of the x, y coordinates
of the points produced if the given point is reflected across each of the
edges or corners.
Other coordinate translation functions:
TernaryCoords()
,
TriangleCentres()
,
XYToTernary()
TernaryPlot(axis.labels = FALSE, point = 4) xy <- cbind( TernaryCoords(0.9, 0.08, 0.02), TernaryCoords(0.15, 0.8, 0.05), TernaryCoords(0.05, 0.1, 0.85) ) x <- xy[1, ] y <- xy[2, ] points(x, y, col = "red", pch = 1:3) ref <- ReflectedEquivalents(x, y) points(ref[[1]][, 1], ref[[1]][, 2], col = "blue", pch = 1) points(ref[[2]][, 1], ref[[2]][, 2], col = "green", pch = 2) points(ref[[3]][, 1], ref[[3]][, 2], col = "orange", pch = 3)
TernaryPlot(axis.labels = FALSE, point = 4) xy <- cbind( TernaryCoords(0.9, 0.08, 0.02), TernaryCoords(0.15, 0.8, 0.05), TernaryCoords(0.05, 0.1, 0.85) ) x <- xy[1, ] y <- xy[2, ] points(x, y, col = "red", pch = 1:3) ref <- ReflectedEquivalents(x, y) points(ref[[1]][, 1], ref[[1]][, 2], col = "blue", pch = 1) points(ref[[2]][, 1], ref[[2]][, 2], col = "green", pch = 2) points(ref[[3]][, 1], ref[[3]][, 2], col = "orange", pch = 3)
TernaryApp()
launches a 'Shiny' application for the construction of
ternary plots. The 'app' allows data to be loaded and plotted, and provides
code to reproduce the plot in R should more sophisticated plotting functions
be desired.
TernaryApp()
TernaryApp()
The 'Load data' input tab allows for the upload of datasets.
Data can be read from csv files, .txt
files created with write.table()
,
or (if the 'readxl' package is installed) Excel spreadsheets.
Data should be provided as three columns, corresponding to the three axes
of the ternary plot. Colours or point styles may be specified in columns
four to six to allow different categories of point to be plotted distinctly.
Example datasets are installed at
system.file("TernaryApp", package = "Ternary")
.
Axes are automatically labelled using column names, if present; these can be edited manually on this tab.
Allows the orientation, colour and configuration of the plot and its axes to be adjusted,
Adjust the number, spacing and styling of major and minor grid lines.
Configure the colour, position and size of tip and axis labels.
Choose whether to plot points, lines, connected points, or text. Set the style of points and lines.
A plot can be saved to PDF or as a PNG bitmap at a specified size. Alternatively, R script that will generate the displayed plot can be viewed (using the 'R code' output tab) or downloaded to file.
Martin R. Smith ([email protected])
If you use figures produced with this package in a publication, please cite
Smith, Martin R. (2017). Ternary: An R Package for Creating Ternary Plots. Zenodo, doi: doi:10.5281/zenodo.1068996.
Full detail of plotting with 'Ternary', including features not (yet) implemented in the application, is provided in the accompanying vignette.
Draws contour lines to depict the value of a function in ternary space.
TernaryContour( Func, resolution = 96L, direction = getOption("ternDirection", 1L), region = getOption("ternRegion", ternRegionDefault), within = NULL, filled = FALSE, legend, legend... = list(), nlevels = 10, levels = pretty(zlim, nlevels), zlim, color.palette = function(n) hcl.colors(n, palette = "viridis", alpha = 0.6), fill.col = color.palette(length(levels) - 1), func... = list(), ... )
TernaryContour( Func, resolution = 96L, direction = getOption("ternDirection", 1L), region = getOption("ternRegion", ternRegionDefault), within = NULL, filled = FALSE, legend, legend... = list(), nlevels = 10, levels = pretty(zlim, nlevels), zlim, color.palette = function(n) hcl.colors(n, palette = "viridis", alpha = 0.6), fill.col = color.palette(length(levels) - 1), func... = list(), ... )
Func |
Function taking vectors of coordinates |
resolution |
The number of triangles whose base should lie on the longest axis of the triangle. Higher numbers will result in smaller subdivisions and smoother colour gradients, but at a computational cost. |
direction |
(optional) Integer specifying the direction that the current ternary plot should point: 1, up; 2, right; 3, down; 4, left. |
region |
(optional) Named list of length two specifying the the
|
within |
List or matrix of x, y coordinates within which contours
should be evaluated, in any format supported by
|
filled |
Logical; if |
legend |
Character vector specifying annotations for colour scale.
If not provided, no colour legend is displayed.
Specify |
legend... |
List of additional parameters to send to
|
nlevels , levels , zlim , ...
|
parameters to pass to
|
color.palette |
parameters to pass to
|
fill.col |
Sent as |
func... |
List of additional parameters to send to |
TernaryContour()
invisibly returns a list containing:
x
,y
: the Cartesian coordinates of each evaluated point;
z
: The value of Func()
at each coordinate.
Martin R. Smith ([email protected])
Other contour plotting functions:
ColourTernary()
,
TernaryDensityContour()
,
TernaryPointValues()
FunctionToContour <- function (a, b, c) { a - c + (4 * a * b) + (27 * a * b * c) } # Set up plot originalPar <- par(mar = rep(0, 4)) TernaryPlot(alab = "a", blab = "b", clab = "c") values <- TernaryPointValues(FunctionToContour, resolution = 24L) ColourTernary( values, legend = signif(seq(max(values), min(values), length.out = 4), 2), bty = "n" ) TernaryContour(FunctionToContour, resolution = 36L) # Note that FunctionToContour is sent a vector. # Instead of BadMax <- function (a, b, c) { max(a, b, c) } # Use GoodMax <- function (a, b, c) { pmax(a, b, c) } TernaryPlot(alab = "a", blab = "b", clab = "c") ColourTernary(TernaryPointValues(GoodMax)) TernaryContour(GoodMax) # Or, for a generalizable example, GeneralMax <- function (a, b, c) { apply(rbind(a, b, c), 2, max) } TernaryPlot(alab = "a", blab = "b", clab = "c") # Fill the contour areas, rather than using tiles TernaryContour(GeneralMax, filled = TRUE, legend = c("Max", "...", "Min"), legend... = list(bty = "n"), fill.col = hcl.colors(14, palette = "viridis", alpha = 0.6)) # Re-draw edges of plot triangle over fill TernaryPolygon(diag(3)) # Restore plotting parameters par(originalPar)
FunctionToContour <- function (a, b, c) { a - c + (4 * a * b) + (27 * a * b * c) } # Set up plot originalPar <- par(mar = rep(0, 4)) TernaryPlot(alab = "a", blab = "b", clab = "c") values <- TernaryPointValues(FunctionToContour, resolution = 24L) ColourTernary( values, legend = signif(seq(max(values), min(values), length.out = 4), 2), bty = "n" ) TernaryContour(FunctionToContour, resolution = 36L) # Note that FunctionToContour is sent a vector. # Instead of BadMax <- function (a, b, c) { max(a, b, c) } # Use GoodMax <- function (a, b, c) { pmax(a, b, c) } TernaryPlot(alab = "a", blab = "b", clab = "c") ColourTernary(TernaryPointValues(GoodMax)) TernaryContour(GoodMax) # Or, for a generalizable example, GeneralMax <- function (a, b, c) { apply(rbind(a, b, c), 2, max) } TernaryPlot(alab = "a", blab = "b", clab = "c") # Fill the contour areas, rather than using tiles TernaryContour(GeneralMax, filled = TRUE, legend = c("Max", "...", "Min"), legend... = list(bty = "n"), fill.col = hcl.colors(14, palette = "viridis", alpha = 0.6)) # Re-draw edges of plot triangle over fill TernaryPolygon(diag(3)) # Restore plotting parameters par(originalPar)
Convert coordinates of a point in ternary space, in the format (a, b, c), to x and y coordinates of Cartesian space, which can be sent to standard functions in the 'graphics' package.
TernaryCoords( abc, b_coord = NULL, c_coord = NULL, direction = getOption("ternDirection", 1L), region = getOption("ternRegion", ternRegionDefault) ) ## S3 method for class 'matrix' TernaryToXY( abc, b_coord = NULL, c_coord = NULL, direction = getOption("ternDirection", 1L), region = getOption("ternRegion", ternRegionDefault) ) ## S3 method for class 'numeric' TernaryToXY( abc, b_coord = NULL, c_coord = NULL, direction = getOption("ternDirection", 1L), region = getOption("ternRegion", ternRegionDefault) ) TernaryToXY( abc, b_coord = NULL, c_coord = NULL, direction = getOption("ternDirection", 1L), region = getOption("ternRegion", ternRegionDefault) )
TernaryCoords( abc, b_coord = NULL, c_coord = NULL, direction = getOption("ternDirection", 1L), region = getOption("ternRegion", ternRegionDefault) ) ## S3 method for class 'matrix' TernaryToXY( abc, b_coord = NULL, c_coord = NULL, direction = getOption("ternDirection", 1L), region = getOption("ternRegion", ternRegionDefault) ) ## S3 method for class 'numeric' TernaryToXY( abc, b_coord = NULL, c_coord = NULL, direction = getOption("ternDirection", 1L), region = getOption("ternRegion", ternRegionDefault) ) TernaryToXY( abc, b_coord = NULL, c_coord = NULL, direction = getOption("ternDirection", 1L), region = getOption("ternRegion", ternRegionDefault) )
abc |
A vector of length three giving the position on a ternary plot
that points in the direction specified by |
b_coord |
The b coordinate, if |
c_coord |
The c coordinate, if |
direction |
(optional) Integer specifying the direction that the current ternary plot should point: 1, up; 2, right; 3, down; 4, left. |
region |
(optional) Named list of length two specifying the the
|
TernaryCoords()
returns a vector of length two that converts
the coordinates given in abc
into Cartesian (x, y) coordinates
corresponding to the plot created by the last call of TernaryPlot()
.
Martin R. Smith ([email protected])
Other coordinate translation functions:
ReflectedEquivalents()
,
TriangleCentres()
,
XYToTernary()
TernaryCoords(100, 0, 0) TernaryCoords(c(0, 100, 0)) coords <- matrix(1:12, nrow = 3) TernaryToXY(coords)
TernaryCoords(100, 0, 0) TernaryCoords(c(0, 100, 0)) coords <- matrix(1:12, nrow = 3) TernaryToXY(coords)
Use two-dimensional kernel density estimation to plot contours of point density.
TernaryDensityContour( coordinates, bandwidth, resolution = 25L, tolerance = -0.2/resolution, edgeCorrection = TRUE, direction = getOption("ternDirection", 1L), filled = FALSE, nlevels = 10, levels = pretty(zlim, nlevels), zlim, color.palette = function(n) hcl.colors(n, palette = "viridis", alpha = 0.6), fill.col = color.palette(length(levels) - 1), ... )
TernaryDensityContour( coordinates, bandwidth, resolution = 25L, tolerance = -0.2/resolution, edgeCorrection = TRUE, direction = getOption("ternDirection", 1L), filled = FALSE, nlevels = 10, levels = pretty(zlim, nlevels), zlim, color.palette = function(n) hcl.colors(n, palette = "viridis", alpha = 0.6), fill.col = color.palette(length(levels) - 1), ... )
coordinates |
A list, matrix, data.frame or vector in which each element (or row) specifies the three coordinates of a point in ternary space. |
bandwidth |
Vector of bandwidths for x and y directions.
Defaults to normal reference bandwidth (see |
resolution |
The number of triangles whose base should lie on the longest axis of the triangle. Higher numbers will result in smaller subdivisions and smoother colour gradients, but at a computational cost. |
tolerance |
Numeric specifying how close to the margins the contours should be plotted, as a fraction of the size of the triangle. Negative values will cause contour lines to extend beyond the margins of the plot. |
edgeCorrection |
Logical specifying whether to correct for edge effects (see details). |
direction |
(optional) Integer specifying the direction that the current ternary plot should point: 1, up; 2, right; 3, down; 4, left. |
filled |
Logical; if |
nlevels , levels , zlim , ...
|
parameters to pass to
|
color.palette |
parameters to pass to
|
fill.col |
Sent as |
This function is modelled on MASS::kde2d()
, which uses
"an axis-aligned bivariate normal kernel, evaluated on a square grid".
This is to say, values are calculated on a square grid, and contours fitted
between these points. This produces a couple of artefacts.
Firstly, contours may not extend beyond the outermost point within the
diagram, which may fall some distance from the margin of the plot if a
low resolution
is used. Setting a negative tolerance
parameter allows
these contours to extend closer to (or beyond) the margin of the plot.
Individual points cannot fall outside the margins of the ternary diagram,
but their associated kernels can. In order to sample regions of the kernels
that have "bled" outside the ternary diagram, each point's value is
calculated by summing the point density at that point and at equivalent
points outside the ternary diagram, "reflected" across the margin of
the plot (see function ReflectedEquivalents
). This correction can be
disabled by setting the edgeCorrection
parameter to FALSE
.
A model based on a triangular grid may be more appropriate in certain situations, but is non-trivial to implement; if this distinction is important to you, please let the maintainers known by opening a Github issue.
TernaryDensityContour()
invisibly returns a list containing:
x
,y
: the Cartesian coordinates of each grid coordinate;
z
: The density at each grid coordinate.
Adapted from MASS::kde2d()
by Martin R. Smith
Other contour plotting functions:
ColourTernary()
,
TernaryContour()
,
TernaryPointValues()
# Generate some example data nPoints <- 400L coordinates <- cbind(abs(rnorm(nPoints, 2, 3)), abs(rnorm(nPoints, 1, 1.5)), abs(rnorm(nPoints, 1, 0.5))) # Set up plot oPar <- par(mar = rep(0, 4)) TernaryPlot(axis.labels = seq(0, 10, by = 1)) # Colour background by density ColourTernary(TernaryDensity(coordinates, resolution = 10L), legend = TRUE, bty = "n", title = "Density") # Plot points TernaryPoints(coordinates, col = "red", pch = ".") # Contour by density TernaryDensityContour(coordinates, resolution = 30L) # Reset plotting parameters par(oPar)
# Generate some example data nPoints <- 400L coordinates <- cbind(abs(rnorm(nPoints, 2, 3)), abs(rnorm(nPoints, 1, 1.5)), abs(rnorm(nPoints, 1, 0.5))) # Set up plot oPar <- par(mar = rep(0, 4)) TernaryPlot(axis.labels = seq(0, 10, by = 1)) # Colour background by density ColourTernary(TernaryDensity(coordinates, resolution = 10L), legend = TRUE, bty = "n", title = "Density") # Plot points TernaryPoints(coordinates, col = "red", pch = ".") # Contour by density TernaryDensityContour(coordinates, resolution = 30L) # Reset plotting parameters par(oPar)
Create and style a blank ternary plot.
TernaryPlot( atip = NULL, btip = NULL, ctip = NULL, alab = NULL, blab = NULL, clab = NULL, lab.offset = 0.16, lab.col = NULL, point = "up", clockwise = TRUE, xlim = NULL, ylim = NULL, region = ternRegionDefault, lab.cex = 1, lab.font = 0, tip.cex = lab.cex, tip.font = 2, tip.col = "black", isometric = TRUE, atip.rotate = NULL, btip.rotate = NULL, ctip.rotate = NULL, atip.pos = NULL, btip.pos = NULL, ctip.pos = NULL, padding = 0.08, col = NA, panel.first = NULL, panel.last = NULL, grid.lines = 10, grid.col = "darkgrey", grid.lty = "solid", grid.lwd = par("lwd"), grid.minor.lines = 4, grid.minor.col = "lightgrey", grid.minor.lty = "solid", grid.minor.lwd = par("lwd"), axis.lty = "solid", axis.labels = TRUE, axis.cex = 0.8, axis.font = par("font"), axis.rotate = TRUE, axis.pos = NULL, axis.tick = TRUE, axis.lwd = 1, ticks.lwd = axis.lwd, ticks.length = 0.025, axis.col = "black", ticks.col = grid.col, ... ) HorizontalGrid( grid.lines = 10, grid.col = "grey", grid.lty = "dotted", grid.lwd = par("lwd"), direction = getOption("ternDirection", 1L) )
TernaryPlot( atip = NULL, btip = NULL, ctip = NULL, alab = NULL, blab = NULL, clab = NULL, lab.offset = 0.16, lab.col = NULL, point = "up", clockwise = TRUE, xlim = NULL, ylim = NULL, region = ternRegionDefault, lab.cex = 1, lab.font = 0, tip.cex = lab.cex, tip.font = 2, tip.col = "black", isometric = TRUE, atip.rotate = NULL, btip.rotate = NULL, ctip.rotate = NULL, atip.pos = NULL, btip.pos = NULL, ctip.pos = NULL, padding = 0.08, col = NA, panel.first = NULL, panel.last = NULL, grid.lines = 10, grid.col = "darkgrey", grid.lty = "solid", grid.lwd = par("lwd"), grid.minor.lines = 4, grid.minor.col = "lightgrey", grid.minor.lty = "solid", grid.minor.lwd = par("lwd"), axis.lty = "solid", axis.labels = TRUE, axis.cex = 0.8, axis.font = par("font"), axis.rotate = TRUE, axis.pos = NULL, axis.tick = TRUE, axis.lwd = 1, ticks.lwd = axis.lwd, ticks.length = 0.025, axis.col = "black", ticks.col = grid.col, ... ) HorizontalGrid( grid.lines = 10, grid.col = "grey", grid.lty = "dotted", grid.lwd = par("lwd"), direction = getOption("ternDirection", 1L) )
atip , btip , ctip
|
Character string specifying text to title corners,
proceeding clockwise from the corner specified in |
alab , blab , clab
|
Character string specifying text with which to label
the corresponding sides of the triangle.
Left or right-pointing arrows are produced by
typing |
lab.offset |
Numeric specifying distance between midpoint of axis label
and the axis. The default value is given in the 'Usage' section; a value
of |
lab.col |
Character vector specifying colours for axis labels. Use a vector of length three to specify a different colour for each label. |
point |
Character string specifying the orientation of the ternary plot:
should the triangle point |
clockwise |
Logical specifying the direction of axes. If |
xlim , ylim
|
Numeric vectors of length two specifying the minimum and
maximum x and y limits of the plotted area, to which |
region |
(optional) Named list of length two specifying the the
|
lab.cex , tip.cex
|
Numeric specifying character expansion (font size) for axis labels. Use a vector of length three to specify a different value for each direction. |
lab.font , tip.font
|
Numeric specifying font style (Roman, bold, italic, bold-italic) for axis titles. Use a vector of length three to set a different font for each direction. |
isometric |
Logical specifying whether to enforce an equilateral shape
for the ternary plot.
If only one of |
atip.rotate , btip.rotate , ctip.rotate
|
Integer specifying number of degrees to rotate label of rightmost apex. |
atip.pos , btip.pos , ctip.pos
|
Integer specifying positioning of labels,
iff the corresponding |
padding |
Numeric specifying size of internal margin of the plot; increase if axis labels are being clipped. |
col |
The colour for filling the plot; see
|
panel.first |
An expression to be evaluated after the plot axes are
set up but before any plotting takes place.
This can be useful for drawing backgrounds, e.g. with |
panel.last |
An expression to be evaluated after plotting has taken
place but before the axes and box are added. See the comments about
|
grid.lines |
Integer specifying the number of grid lines to plot.
If |
grid.col , grid.minor.col
|
Colours to draw the grid lines. Use a vector of length three to set different values for each direction. |
grid.lty , grid.minor.lty
|
Character or integer vector; line type of the grid lines. Use a vector of length three to set different values for each direction. |
grid.lwd , grid.minor.lwd
|
Non-negative numeric giving line width of the grid lines. Use a vector of length three to set different values for each direction. |
grid.minor.lines |
Integer specifying the number of minor (unlabelled) grid lines to plot between each major pair. |
axis.lty |
Line type for both the axis line and tick marks. Use a vector of length three to set a different value for each direction. |
axis.labels |
This can either be a logical value specifying whether (numerical) annotations are to be made at the tickmarks, or a character or expression vector of labels to be placed at the tick points, or a list of length three, with each entry specifying labels to be placed on each axis in turn. |
axis.cex |
Numeric specifying character expansion (font size) for axis labels. Use a vector of length three to set a different value for each direction. |
axis.font |
Font for text. Defaults to |
axis.rotate |
Logical specifying whether to rotate axis labels
to parallel grid lines, or numeric specifying custom rotation for each axis,
to be passed as |
axis.pos |
Vector of length one or three specifying position of axis
labels, to be passed as |
axis.tick |
Logical specifying whether to mark the axes with tick marks. |
axis.lwd , ticks.lwd
|
Line width for the axis line and tick marks. Zero or negative values will suppress the line or ticks. Use a vector of length three to set different values for each axis. |
ticks.length |
Numeric specifying distance that ticks should extend beyond the plot margin. Also affects position of axis labels, which are plotted at the end of each tick. Use a vector of length three to set a different length for each direction. |
axis.col , ticks.col , tip.col
|
Colours for the axis line,
tick marks and tip labels respectively.
Use a vector of length three to set a different value for each direction.
|
... |
Additional parameters to |
direction |
(optional) Integer specifying the direction that the current ternary plot should point: 1, up; 2, right; 3, down; 4, left. |
The plot will be generated using the standard 'graphics' plot functions, on
which additional elements can be added using cartesian coordinates, perhaps
using functions such as arrows
,
legend
or text
.
HorizontalGrid()
: Add grid.lines
horizontal lines to the ternary plot
Martin R. Smith ([email protected])
AddToTernary()
: Add elements to a ternary plot
TernaryCoords()
: Convert ternary coordinates to Cartesian
(x and y) coordinates
TernaryXRange()
, TernaryYRange()
: What are the x and y limits
of the plotted region?
TernaryPlot( atip = "Top", btip = "Bottom", ctip = "Right", axis.col = "red", col = rgb(0.8, 0.8, 0.8) ) HorizontalGrid(grid.lines = 2, grid.col = "blue", grid.lty = 1) # the second line corresponds to the base of the triangle, and is not drawn
TernaryPlot( atip = "Top", btip = "Bottom", ctip = "Right", axis.col = "red", col = rgb(0.8, 0.8, 0.8) ) HorizontalGrid(grid.lines = 2, grid.col = "blue", grid.lty = 1) # the second line corresponds to the base of the triangle, and is not drawn
Intended to facilitate coloured contour plots with ColourTernary()
,
TernaryPointValue()
evaluates a function at points on a triangular grid;
TernaryDensity()
calculates the density of points in each grid cell.
TernaryPointValues( Func, resolution = 48L, direction = getOption("ternDirection", 1L), ... ) TernaryDensity( coordinates, resolution = 48L, direction = getOption("ternDirection", 1L) )
TernaryPointValues( Func, resolution = 48L, direction = getOption("ternDirection", 1L), ... ) TernaryDensity( coordinates, resolution = 48L, direction = getOption("ternDirection", 1L) )
Func |
Function taking vectors of coordinates |
resolution |
The number of triangles whose base should lie on the longest axis of the triangle. Higher numbers will result in smaller subdivisions and smoother colour gradients, but at a computational cost. |
direction |
(optional) Integer specifying the direction that the current ternary plot should point: 1, up; 2, right; 3, down; 4, left. |
... |
Additional parameters to |
coordinates |
A list, matrix, data.frame or vector in which each element (or row) specifies the three coordinates of a point in ternary space. |
TernaryPointValues()
returns a matrix whose rows correspond to:
x, y: co-ordinates of the centres of smaller triangles
z: The value of Func(a, b, c, ...)
, where a
, b
and c
are the
ternary coordinates of x
and y
.
down: 0
if the triangle concerned points upwards (or right),
1
otherwise
Martin R. Smith ([email protected])
Other contour plotting functions:
ColourTernary()
,
TernaryContour()
,
TernaryDensityContour()
TernaryPointValues(function (a, b, c) a * b * c, resolution = 2) TernaryPlot(grid.lines = 4) cols <- TernaryPointValues(rgb, resolution = 4) text(as.numeric(cols["x", ]), as.numeric(cols["y", ]), labels = ifelse(cols["down", ] == "1", "v", "^"), col = cols["z", ]) TernaryPlot(axis.labels = seq(0, 10, by = 1)) nPoints <- 4000L coordinates <- cbind(abs(rnorm(nPoints, 2, 3)), abs(rnorm(nPoints, 1, 1.5)), abs(rnorm(nPoints, 1, 0.5))) density <- TernaryDensity(coordinates, resolution = 10L) ColourTernary(density, legend = TRUE, bty = "n", title = "Density") TernaryPoints(coordinates, col = "red", pch = ".")
TernaryPointValues(function (a, b, c) a * b * c, resolution = 2) TernaryPlot(grid.lines = 4) cols <- TernaryPointValues(rgb, resolution = 4) text(as.numeric(cols["x", ]), as.numeric(cols["y", ]), labels = ifelse(cols["down", ] == "1", "v", "^"), col = cols["z", ]) TernaryPlot(axis.labels = seq(0, 10, by = 1)) nPoints <- 4000L coordinates <- cbind(abs(rnorm(nPoints, 2, 3)), abs(rnorm(nPoints, 1, 1.5)), abs(rnorm(nPoints, 1, 0.5))) density <- TernaryDensity(coordinates, resolution = 10L) ColourTernary(density, legend = TRUE, bty = "n", title = "Density") TernaryPoints(coordinates, col = "red", pch = ".")
Function to fill a ternary plot with coloured tiles. Useful in combination with
TernaryPointValues
and TernaryContour
.
TernaryTiles( x, y, down, resolution, col, direction = getOption("ternDirection", 1L) )
TernaryTiles( x, y, down, resolution, col, direction = getOption("ternDirection", 1L) )
x , y
|
Numeric vectors specifying x and y coordinates of centres of each triangle. |
down |
Logical vector specifying |
resolution |
The number of triangles whose base should lie on the longest axis of the triangle. Higher numbers will result in smaller subdivisions and smoother colour gradients, but at a computational cost. |
col |
Vector specifying the colour with which to fill each triangle. |
direction |
(optional) Integer specifying the direction that the current ternary plot should point: 1, up; 2, right; 3, down; 4, left. |
Martin R. Smith ([email protected])
Other functions for colouring and shading:
ColourTernary()
TernaryPlot() TernaryXRange() TernaryYRange() TernaryTiles(0, 0.5, TRUE, 10, "red") xy <- TernaryCoords(c(4, 3, 3)) TernaryTiles(xy[1], xy[2], FALSE, 5, "darkblue")
TernaryPlot() TernaryXRange() TernaryYRange() TernaryTiles(0, 0.5, TRUE, 10, "red") xy <- TernaryCoords(c(4, 3, 3)) TernaryTiles(xy[1], xy[2], FALSE, 5, "darkblue")
X and Y coordinates of ternary plotting area
TernaryXRange(direction = getOption("ternDirection", 1L)) TernaryYRange(direction = getOption("ternDirection", 1L))
TernaryXRange(direction = getOption("ternDirection", 1L)) TernaryYRange(direction = getOption("ternDirection", 1L))
direction |
(optional) Integer specifying the direction that the current ternary plot should point: 1, up; 2, right; 3, down; 4, left. |
TernaryXRange()
and TernaryYRange()
return the minimum and
maximum X or Y coordinate of the area in which a ternary plot is drawn,
oriented in the specified direction.
Because the plotting area is a square, the triangle of the ternary plot
will not occupy the full range in one direction.
Assumes that the defaults have not been overwritten by specifying xlim
or
ylim
.
TernaryYRange()
: Returns the minimum and maximum Y coordinate for a
ternary plot in the specified direction.
Martin R. Smith ([email protected])
Other plot limits:
OutsidePlot()
Calculate x and y coordinates of the midpoints of triangles tiled to cover a ternary plot.
TriangleCentres(resolution = 48L, direction = getOption("ternDirection", 1L))
TriangleCentres(resolution = 48L, direction = getOption("ternDirection", 1L))
resolution |
The number of triangles whose base should lie on the longest axis of the triangle. Higher numbers will result in smaller subdivisions and smoother colour gradients, but at a computational cost. |
direction |
(optional) Integer specifying the direction that the current ternary plot should point: 1, up; 2, right; 3, down; 4, left. |
TriangleCentres()
returns a matrix with three named rows:
x
x coordinates of triangle midpoints;
y
y coordinates of triangle midpoints;
triDown
0
for upwards-pointing triangles, 1
for downwards-pointing.
Martin R. Smith ([email protected])
Add triangles to a plot: TernaryTiles()
Other coordinate translation functions:
ReflectedEquivalents()
,
TernaryCoords()
,
XYToTernary()
Other tiling functions:
Polygon-Geometry
,
TriangleInHull()
TernaryPlot(grid.lines = 4) centres <- TriangleCentres(4) text(centres["x", ], centres["y", ], ifelse(centres["triDown", ], "v", "^"))
TernaryPlot(grid.lines = 4) centres <- TriangleCentres(4) text(centres["x", ], centres["y", ], ifelse(centres["triDown", ], "v", "^"))
Does triangle overlap convex hull of points?
TriangleInHull(triangles, coordinates, buffer)
TriangleInHull(triangles, coordinates, buffer)
triangles |
Three-row matrix as produced by |
coordinates |
A matrix with two or three rows specifying the coordinates of points in x, y or a, b, c format. |
buffer |
Include triangles whose centres lie within |
TriangleInHull()
returns a list with the elements:
$inside
: vector specifying whether each of a
set of triangles produced by TriangleCentres()
overlaps the convex
hull of points specified by coordinates
.
$hull
: Coordinates of convex hull of coordinates
, after expansion
to cover overlapping triangles.
Martin R. Smith ([email protected])
Other tiling functions:
Polygon-Geometry
,
TriangleCentres()
set.seed(0) nPts <- 50 a <- runif(nPts, 0.3, 0.7) b <- 0.15 + runif(nPts, 0, 0.7 - a) c <- 1 - a - b coordinates <- rbind(a, b, c) TernaryPlot(grid.lines = 5) TernaryPoints(coordinates, pch = 3, col = 4) triangles <- TriangleCentres(resolution = 5) inHull <- TriangleInHull(triangles, coordinates) polygon(inHull$hull, border = 4) values <- rbind(triangles, z = ifelse(inHull$inside, "#33cc3333", "#cc333333")) points(triangles["x", ], triangles["y", ], pch = ifelse(triangles["triDown", ], 6, 2), col = ifelse(inHull$inside, "#33cc33", "#cc3333")) ColourTernary(values)
set.seed(0) nPts <- 50 a <- runif(nPts, 0.3, 0.7) b <- 0.15 + runif(nPts, 0, 0.7 - a) c <- 1 - a - b coordinates <- rbind(a, b, c) TernaryPlot(grid.lines = 5) TernaryPoints(coordinates, pch = 3, col = 4) triangles <- TriangleCentres(resolution = 5) inHull <- TriangleInHull(triangles, coordinates) polygon(inHull$hull, border = 4) values <- rbind(triangles, z = ifelse(inHull$inside, "#33cc3333", "#cc333333")) points(triangles["x", ], triangles["y", ], pch = ifelse(triangles["triDown", ], 6, 2), col = ifelse(inHull$inside, "#33cc33", "#cc3333")) ColourTernary(values)
Convert cartesian (x, y) coordinates to a point in ternary space.
XYToTernary( x, y, direction = getOption("ternDirection", 1L), region = getOption("ternRegion", ternRegionDefault) ) XYToHoldridge(x, y) XYToPetPrec(x, y)
XYToTernary( x, y, direction = getOption("ternDirection", 1L), region = getOption("ternRegion", ternRegionDefault) ) XYToHoldridge(x, y) XYToPetPrec(x, y)
x , y
|
Numeric values giving the x and y coordinates of a point or points. |
direction |
(optional) Integer specifying the direction that the current ternary plot should point: 1, up; 2, right; 3, down; 4, left. |
region |
(optional) Named list of length two specifying the the
|
XYToTernary()
Returns the ternary point(s) corresponding to the
specified x and y coordinates, where a + b + c = 1.
Martin R. Smith ([email protected])
Other coordinate translation functions:
ReflectedEquivalents()
,
TernaryCoords()
,
TriangleCentres()
XYToTernary(c(0.1, 0.2), 0.5)
XYToTernary(c(0.1, 0.2), 0.5)