Nothing
#' Create a Glyph Ribbon plot using ggplot2
#'
#' This function creates a ribbon geometry designed to display glyphs based on
#' the combination of `x_major` and `y_major`. For each `x_minor` value,
#' `geom_glyph_ribbon()` displays a y interval defined by `ymin_minor` and `ymax_minor`.
#'
#' @inheritParams ggplot2::layer
#' @inheritParams ggplot2::GeomRibbon
#' @param x_major,y_major,x_minor,ymin_minor,ymax_minor Each combination of
#' `x_major` and `y_major` forms a unique grid cell. `ymin_minor` and `ymax_minor` define
#' the lower and upper bounds of the geom_ribbon.
#' @param width The width of each glyph. The `default` is set
#' to the smallest distance between two consecutive coordinates, converted from meters
#' to degrees of latitude using the Haversine method.
#' @param height The height of each glyph. The `default` is calculated using the ratio (1:1.618)
#' relative to the `width`, to maintain a consistent aspect ratio.
#' @param x_scale,y_scale The scaling function applied to each set of minor
#' values within a grid cell. Defaults to `identity`.
#' @param global_rescale A setting that determines whether to perform rescaling globally or on individual glyphs.
#' @param ... Additional arguments passed on to function.
#' @return A ggplot object.
#' @examples
#'
#' library(ggplot2)
#'
#' # Basic glyph map with base map and custom theme
#' aus_temp |>
#' ggplot(aes(x_major = long, y_major = lat,
#' x_minor = month, ymin_minor = tmin, ymax_minor = tmax)) +
#' geom_sf(data = ozmaps::abs_ste, fill = "grey95",
#' color = "white",inherit.aes = FALSE) +
#' geom_glyph_ribbon() +
#' ggthemes::theme_map()
#'
#'
#' # Adjust width and height of the glyph
#' aus_temp |>
#' ggplot(aes(x_major = long, y_major = lat,
#' x_minor = month, ymin_minor = tmin, ymax_minor = tmax)) +
#' geom_sf(data = ozmaps::abs_ste, fill = "grey95",
#' color = "white",inherit.aes = FALSE) +
#' geom_glyph_ribbon(width = rel(4.5), height = rel(3)) +
#' ggthemes::theme_map()
#'
#' # Extend glyph map with reference box and line
#' aus_temp |>
#' ggplot(aes(x_major = long, y_major = lat,
#' x_minor = month, ymin_minor = tmin, ymax_minor = tmax)) +
#' geom_sf(data = ozmaps::abs_ste, fill = "grey95",
#' color = "white",inherit.aes = FALSE) +
#' add_glyph_boxes() +
#' add_ref_lines() +
#' geom_glyph_ribbon() +
#' ggthemes::theme_map()
geom_glyph_ribbon <- function( mapping = NULL, data = NULL, show.legend = NA,
stat = "identity", position = "identity",
x_major = NULL, y_major = NULL,
x_minor = NULL, ymin_minor = NULL, ymax_minor = NULL,
height = "default", width = "default",
x_scale = identity, y_scale = identity,
global_rescale = TRUE, inherit.aes = TRUE, ...) {
ggplot2::layer(
geom = GeomGlyphRibbon,
mapping = mapping,
data = data,
stat = stat,
position = position,
show.legend = show.legend,
inherit.aes = inherit.aes,
params = list(
height = height,
width = width,
x_scale = list(x_scale),
y_scale = list(y_scale),
global_rescale = global_rescale,
...)
)
}
#' GeomGlyphRibbon
#' @rdname sugarglider
#' @format NULL
#' @usage NULL
#' @seealso \link[ggplot2]{GeomRibbon} from the ggplot2 package.
#' @keywords internal
#' @export
GeomGlyphRibbon <- ggplot2::ggproto(
"GeomGlyphRibbon", ggplot2::GeomRibbon,
## Aesthetic
required_aes = c("x_major", "y_major",
"x_minor", "ymin_minor", "ymax_minor"),
default_aes = ggplot2::aes(
linetype = 1, fill = "black", color = "black",
linewidth = 0.5, alpha = 0.8,
width = "default",
height = "default",
x_scale = list(identity),
y_scale = list(identity),
global_rescale = TRUE
),
setup_data = function(data, params) {
params <- update_params(data, params)
data <- glyph_setup_data(data, params)
},
# Draw polygons
draw_panel = function(data, panel_params, ...) {
ggplot2::GeomRibbon$draw_panel(data, panel_params, ...)
#ggiraph::GeomInteractiveRibbon$draw_group(data, panel_params, ...)
}
)
#' Add Glyph Boxes layer to glyph plot
#'
#'This function introduces a custom layer to a ggplot, employing 'glyph boxes'
#'to visually represent individual glyph. Users can specify various aesthetics
#'including alpha, height, width, color, line type, and fill to customize the appearance.
#'
#' @inheritParams ggplot2::layer
#' @inheritParams ggplot2::GeomRect
#' @param data The data to be displayed in this layer. If \code{NULL}, the default, the data is
#' inherited from the plot data as specified in the call to \code{ggplot()}.
#' @param x_major,y_major Aesthetics to map plot coordinates
#' for major and minor glyph components.
#' @param width The width of each glyph. The `default` is set
#' to the smallest distance between two consecutive coordinates, converted from meters
#' to degrees of latitude using the Haversine method.
#' @param height The height of each glyph. The `default` is calculated using the ratio (1:1.618)
#' relative to the `width`, to maintain a consistent aspect ratio.
#' @param fill The color used to fill the glyph box.
#' @param linewidth The thickness of the glyph box.
#' @param ... Additional arguments passed on to function.
#'
#' @return A layer object that can be added to a ggplot.
#'
#' @export
add_glyph_boxes <- function( mapping = NULL, data = NULL,
stat = "identity", position = "identity",
x_major = NULL, y_major = NULL,
height = "default", width = "default",
fill = "white", linewidth = 0.1,
inherit.aes = TRUE, show.legend = NA, ...) {
ggplot2::layer(
geom = GeomGlyphBox,
mapping = mapping,
data = data,
stat = stat,
position = position,
show.legend = show.legend,
inherit.aes = inherit.aes,
params = list(
height = height,
width = width,
fill = fill,
linewidth = linewidth,
...)
)
}
#' GeomGlyphBox
#' @rdname sugarglider
#' @format NULL
#' @usage NULL
#' @seealso \link[cubble]{geom_glyph_box} from the cubble package.
#' @keywords internal
#' @export
GeomGlyphBox <- ggplot2::ggproto(
"GeomGlyphBox", ggplot2::Geom,
## Aesthetic
required_aes = c("x_major", "y_major"),
default_aes = ggplot2::aes(
linetype = "solid", fill = "white", color = "black",
linewidth = 0.1, alpha = 0.5,
width = "default",
height = "default"
),
setup_data = function(data, params) {
params <- update_params(data, params)
data <- configure_glyph_data(data, params)
glyph_box(data, params)
},
draw_panel = function(data, panel_params, coord, alpha = alpha, linewidth = linewidth,
fill = fill, linetype = linetype, color = color, ...) {
ggplot2:::GeomRect$draw_panel(data, panel_params, coord, ...)
}
)
#' Add reference lines to glyph plot
#'
#' This function draw reference lines that include both major and minor division markers.
#'
#' @inheritParams ggplot2::layer
#' @inheritParams ggplot2::GeomPath
#' @param data The data to be displayed in this layer. If \code{NULL}, the default, the data is
#' inherited from the plot data as specified in the call to \code{ggplot()}.
#' @param x_major,y_major Aesthetics to map plot coordinates
#' for major and minor glyph components.
#' @param width The width of each glyph. The `default` is set
#' to the smallest distance between two consecutive coordinates, converted from meters
#' to degrees of latitude using the Haversine method.
#' @param height The height of each glyph. The `default` is calculated using the ratio (1:1.618)
#' relative to the `width`, to maintain a consistent aspect ratio.
#' @param linewidth The thickness of the reference line.
#' @param ... Additional arguments passed on to function.
#' @return A ggplot2 layer.
#' @export
add_ref_lines <- function( mapping = NULL, data = NULL,
stat = "identity", position = "identity",
show.legend = NA, x_major = NULL, y_major = NULL,
height = "default", width = "default",
inherit.aes = TRUE, linewidth = 0.1, ...) {
ggplot2::layer(
geom = GeomGlyphLine,
mapping = mapping,
data = data,
stat = stat,
position = position,
show.legend = show.legend,
inherit.aes = inherit.aes,
params = list(
height = height,
width = width,
linewidth = linewidth,
...)
)
}
#' GeomGlyphLine
#' @rdname sugarglider
#' @format NULL
#' @usage NULL
#' @seealso \link[cubble]{geom_glyph_line} from the cubble package.
#' @keywords internal
#' @export
GeomGlyphLine <- ggplot2::ggproto(
"GeomGlyphLine", ggplot2::Geom,
## Aesthetic
required_aes = c("x_major", "y_major"),
default_aes = ggplot2::aes(
linetype = "solid", color = "black",
linewidth = 0.01, alpha = 0.5,
width = "default",
height = "default"
),
setup_data = function(data, params) {
params <- update_params(data, params)
data <- configure_glyph_data(data, params)
ref_line(data, params)
},
draw_panel = function(data, panel_params, coord, linewidth = linewidth, ...) {
ggplot2:::GeomPath$draw_panel(data, panel_params, coord, ...)
}
)
#' Add Legend Layer to a ggplot
#'
#' This function adds a custom legend layer to a ggplot object using
#' the specified aesthetics and parameters.
#'
#' @inheritParams ggplot2::layer
#' @param x_minor Aesthetics to map plot coordinates
#' for major and minor glyph components.
#' @param x_scale,y_scale The scaling function applied to each set of minor
#' values within a grid cell. Defaults to `identity`.
#' @param global_rescale A setting that determines whether to perform rescaling globally or on individual glyphs.
#' @param fill The fill color for the geometric object.
#' @param color The color of the geometric object's border or line.
#' @param linewidth The width of the geometric object's line.
#' @param alpha The transparency level of the geometric object, ranging from 0 (fully transparent) to 1 (fully opaque).
#' @param ... Additional arguments passed on to function.
#' @return A ggplot2 layer.
#' @export
add_glyph_legend <- function( mapping = NULL, data = NULL,
stat = "identity", position = "identity", show.legend = NA,
x_minor = NULL,x_scale = identity, y_scale = identity,
fill = "black", color = "black", linewidth = 0.5, alpha = 0.8,
global_rescale = TRUE, inherit.aes = TRUE, ...) {
ggplot2::layer(
geom = GeomGlyphLegend,
mapping = mapping,
data = data,
stat = stat,
position = position,
show.legend = show.legend,
inherit.aes = inherit.aes,
params = list(
x_scale = list(x_scale),
y_scale = list(y_scale),
global_rescale = global_rescale,
fill = fill,
color = color,
linewidth = linewidth,
alpha = alpha,
...)
)
}
#' GeomGlyphLegend
#' @rdname sugarglider
#' @format NULL
#' @usage NULL
#' @keywords internal
#' @export
GeomGlyphLegend <- ggplot2::ggproto(
"GeomGlyphLegend", ggplot2::Geom,
## Aesthetic
required_aes = c("x_minor"),
default_aes = ggplot2::aes(
linetype = 1, fill = "black", color = "black",
linewidth = 0.5, alpha = 0.8,
x_scale = list(identity),
y_scale = list(identity),
global_rescale = TRUE
),
setup_data = function(data, params){
data <- configure_glyph_data(data, params, legend = TRUE)
},
# Draw polygons
draw_panel = function(data, panel_params, params, ...) {
grob <- glyph_setup_grob(data, panel_params)
legend_vp <- viewport(x = 0.13, y = 0.02,
width = 0.23, height = 0.23,
just = c("bottom"))
editGrob(grob, vp = legend_vp, name = grob$name)
}
)
# Helper functions -------------------------------------------------------------
#' Prepare data for ribbon and segment glyph maps
#' @return A processed data frame, where columns are adjusted and rescaled for
#' different geometric types, such as `geom_glyph_ribbon()` or `geom_glyph_segment()`.
#' @keywords internal
glyph_setup_data <- function(data, params,...) {
arg <- list(...)
if (!inherits(data$x_minor, c("Date", "yearmonth", "numeric", "factor",
"yearweek", "yearquarter", "yearqtr",
"POSIXct", "POSIXlt"))) {
stop("Error: Unsupported class for x_minor. Supported classes are Date,
yearmonth, numeric, factor, yearweek, yearquarter, yearqtr, POSIXct, and POSIXlt.")
}
# Ensure geom draws each glyph as a distinct path
if (dplyr::n_distinct(data$group) == 1 ||
(inherits(data$x_minor, "factor") && !any(c("colour", "color", "fill") %in% names(data)))) {
data$group <- as.integer(factor(paste(data$x_major, data$y_major)))
data <- data |> dplyr::group_by(.data$group)
}
# Convert minor axis to numeric
if (!is.numeric(data$x_minor)){
data[["x_minor"]] <- as.numeric(data[["x_minor"]])
}
if (isTRUE(arg$segment)){
data[["y_minor"]] <- as.numeric(data[["y_minor"]])
data[["yend_minor"]] <- as.numeric(data[["yend_minor"]])
} else {
data[["ymin_minor"]] <- as.numeric(data[["ymin_minor"]])
data[["ymax_minor"]] <- as.numeric(data[["ymax_minor"]])
}
# Handle missing data
if (any(is.na(data)) ) {
warning(paste("Removed rows containing missing values"))
data <- data |> na.omit()
}
if (custom_scale(params$x_scale)) {
x_scale <- get_scale(params$x_scale)
data <- data |>
dplyr::mutate(
x_minor = x_scale(.data$x_minor)
)
}
if (custom_scale(params$y_scale)) {
if (isTRUE(arg$segment)) {
y_scale <- get_scale(params$y_scale)
data <- data |>
dplyr::mutate(
y_minor = y_scale(.data$y_minor),
yend_minor = y_scale(.data$yend_minor)
)
} else {
y_scale <- get_scale(params$y_scale)
data <- data |>
dplyr::mutate(
ymin_minor = y_scale(.data$ymin_minor),
ymax_minor = y_scale(.data$ymax_minor)
)
}
}
if (isTRUE(params$global_rescale)) { data <- data |> dplyr::ungroup() }
if (isTRUE(arg$legend)){
# Skip the linear transformation and rescale for legend data
data <- data |>
dplyr::mutate(com = interaction(.data$x_major, .data$y_major)) |>
dplyr::filter(com == sample(com, 1)) |>
dplyr::select(-com)
} else {
## Scale minor axis
if (isTRUE(arg$segment)){
data <- data |>
dplyr::mutate(
y_minor = rescale11y(y_minor, yend_minor)[[1]],
yend_minor = rescale11y(y_minor, yend_minor)[[2]]
)
} else {
data <- data |>
tidyr::pivot_longer(cols = c("ymin_minor", "ymax_minor"),
names_to = "type", values_to = "value") |>
dplyr::mutate(scaled_data = rescale(value)) |>
dplyr::select(-value) |>
tidyr::pivot_wider(names_from = "type", values_from = "scaled_data")
}
## Linear transformation using scaled positional adjustment
if (isTRUE(arg$segment)){
# For geom_segment_glyph
data <- data |>
dplyr::mutate(
x = glyph_mapping(.data$x_major,
rescale(.data$x_minor),
params$width),
xend = glyph_mapping(.data$x_major,
rescale(.data$x_minor),
params$width),
y = glyph_mapping(.data$y_major,
.data$y_minor,
params$height),
yend = glyph_mapping(.data$y_major,
.data$yend_minor,
params$height)
)
} else {
# For geom_ribbon_glyph
data <- data |>
dplyr::mutate(
x = glyph_mapping(.data$x_major,
rescale(.data$x_minor),
params$width),
ymin = glyph_mapping(.data$y_major,
.data$ymin_minor,
params$height),
ymax = glyph_mapping(.data$y_major,
.data$ymax_minor,
params$height))
}
}
# Ensure linewidth is initialized
if (!("linewidth" %in% colnames(data))) {
data$linewidth <- 0.5
}
data |> dplyr::ungroup()
}
#' Create reference boxes for glyph plot
#' @return A data frame with additional columns (`xmin`, `xmax`, `ymin`, `ymax`)
#' defining the bounding box for each glyph based on the major axes and specified dimensions.
#' @keywords internal
glyph_box <- function(data, params) {
# Code from cubble:
data <- data |>
dplyr::mutate(
xmin = data$x_major - params$width/2,
xmax = data$x_major + params$width/2,
ymin = data$y_major - params$height/2,
ymax = data$y_major + params$height/2,
)
data
}
#' Calculate reference lines for glyph plot
#' @return A data frame with additional columns (`x`, `y`, `group`) that define
#' reference lines for each glyph based on the major axes and specified width.
#' @keywords internal
ref_line <- function(data, params){
# Code from cubble:
data <- data |>
tidyr::expand_grid(delta = c(-1, 1)) |>
dplyr::mutate(
group = .data$group,
x = .data$x_major + (params$width / 2) * .data$delta,
y = .data$y_major
)
data
}
#' Scaled positional adjustment
#' @return A numeric vector representing the spatio-temporal transformation of minor axes into
#' spatial coordinates.
#' @keywords internal
glyph_mapping <- function(spatial, scaled_value, length) {
spatial + scaled_value * (length / 2)
}
#' Convert ggplot2 object into grob
#' @return A grob (graphical object) converted from a ggplot2 object.
#' @keywords internal
glyph_setup_grob <- function(data, panel_params){
if (!(("ymin_minor" %in% names(data) && "ymax_minor" %in% names(data)) ||
("y_minor" %in% names(data) && "yend_minor" %in% names(data)))) {
stop("Data must include either 'ymin_minor' and 'ymax_minor' columns for
geom_glyph_ribbon() or 'y_minor' and 'yend_minor' columns for geom_glyph_segment().")
}
if ("ymin_minor" %in% names(data)) {
p_grob <- data |>
ggplot2::ggplot(
ggplot2::aes(x = x_minor,
ymin = ymin_minor,
ymax = ymax_minor
)) +
geom_ribbon(fill = data$fill,
color = data$colour,
linewidth = data$linewidth,
alpha = data$alpha) +
theme_bw() +
theme(
panel.background = element_rect(fill = NA, color = NA),
plot.background = element_rect(fill = NA, color = NA)
) +
labs(x = "") +
scale_x_discrete(expand = c(0,0))
} else {
p_grob <- data |>
ggplot2::ggplot(
ggplot2::aes(x = x_minor,
y = y_minor,
yend = yend_minor)) +
geom_segment(
color = data$colour,
linewidth = data$linewidth,
alpha = data$alpha) +
theme_bw() +
theme(
panel.background = element_rect(fill = NA, color = NA),
plot.background = element_rect(fill = NA, color = NA)
) +
labs(x = "", y = "") +
scale_x_discrete(expand = c(0,0))
}
ggplotify::as.grob(p_grob)
}
#' Setup Glyph Data Based on Geometric Plot Type
#' @return A modified data frame prepared for plotting glyphs, based on the
#' geometric plot type.
#' @keywords internal
configure_glyph_data <- function(data, params,...){
if (!(("ymin_minor" %in% names(data) && "ymax_minor" %in% names(data)) ||
("y_minor" %in% names(data) && "yend_minor" %in% names(data)))) {
stop("Data must include either 'ymin_minor' and 'ymax_minor' columns for
geom_glyph_ribbon() or 'y_minor' and 'yend_minor' columns for geom_glyph_segment().")
}
# If "y_minor" is provided
if ("y_minor" %in% names(data)) {
data <- glyph_setup_data(data, params, segment = TRUE, ...)
} else {
data <- glyph_setup_data(data, params, ...)
}
}
#' Calculate the Smallest Distance Across Unique Combinations of Major Axes
#'
#' This function calculates the smallest distance between all unique coordinate combinations of major axes
#' using the Haversine formula, which measures distances on the surface of a sphere (i.e., Earth).
#' The function returns the minimum distance in degrees of latitude and a corresponding height that
#' maintains a glyph ratio of 1:1.618 (the golden ratio).
#'
#' @param data A data frame containing columns `x_major` and `y_major`, which represent the coordinates
#' (longitude and latitude) for each point.
#'
#' @return A list containing:
#' \describe{
#' \item{width}{The smallest distance between any two unique points, converted from meters to degrees of latitude.}
#' \item{height}{The height corresponding to the width, calculated using the golden ratio (1:1.618).}
#' }
#'
#' @keywords internal
calculate_min_dist <- function(data) {
coordinates <- data |>
dplyr::mutate(unique_coord = paste(x_major, y_major, sep = ",")) |>
dplyr::group_by(unique_coord) |>
dplyr::slice_head(n = 1)
# Calculate distance using Haversine method
dist_matrix <- geosphere::distm(coordinates[, c("x_major", "y_major")],
fun = geosphere::distHaversine)
diag(dist_matrix) <- Inf
# Convert meters to degrees of latitude (1 degree latiude = 111,320 meters)
min_distance <- min(apply(dist_matrix, 1, min))/111320
list(
# Glyph ratio of 1:1.618
width = min_distance,
height = min_distance / 1.618
)
}
#' Defined Glyph dimension
#' @return A list containing updated `width` and `height` values.
#' @keywords internal
update_params <- function(data, params) {
if (dplyr::n_distinct(data$x_major) == 1 &&
dplyr::n_distinct(data$y_major) == 1) {
params$width <- rel(3)
params$height <- rel(2)
} else {
min_dist <- calculate_min_dist(data)
params$width <- ifelse(params$width == "default", min_dist$width, params$width)
params$height <- ifelse(params$height == "default", min_dist$height, params$height)
}
return(params)
}
#' Update Glyph Dimensions
#'
#' Adjusts the width and height of glyphs based on the distinct values of
#' `x_major` and `y_major` in the data.
#'
#' @return A list containing updated `width` and `height` values.
#' @keywords internal
update_params <- function(data, params) {
if (dplyr::n_distinct(data$x_major) == 1 &&
dplyr::n_distinct(data$y_major) == 1) {
params$width <- rel(3)
params$height <- rel(2)
} else {
min_dist <- calculate_min_dist(data)
params$width <- ifelse(params$width == "default", min_dist$width, params$width)
params$height <- ifelse(params$height == "default", min_dist$height, params$height)
}
return(params)
}
# Rescale Functions ----------------------------------------------------------
#' Rescale Functions
#'
#' @return A rescaled numeric vector or list of numeric vectors,
#' with values adjusted to fit within the interval [-1, 1] or [0, 1],
#' depending on the function used.
#' @name rescale
#' @keywords internal
#' @rdname rescale
rescale <- function(dx) {
stopifnot(!is.na(dx))
stopifnot(length(dx) > 0)
rng <- range(dx, na.rm = TRUE)
if (rng[1] == rng[2]) return(rep(0, length(dx))) # Avoid division by zero
as.numeric(2 * (dx - rng[1])/(rng[2] - rng[1]) - 1)
}
#' @keywords internal
#' @rdname rescale
rescale01y <- function(y, yend, ylim=NULL) {
if (is.null(ylim)) {
rngy <- range(y, na.rm = TRUE)
rngyend <- range(yend, na.rm = TRUE)
} else {
rng <- ylim
}
ymin = min(rngy[1], rngyend[1])
ymax = max(rngy[2], rngyend[2])
y = (y - ymin) / (ymax - ymin)
yend = (yend - ymin) / (ymax - ymin)
return(list(y, yend))
}
#' @keywords internal
#' @rdname rescale
rescale11y <- function(y, yend, xlim=NULL) {
newy = 2 * (rescale01y(y, yend)[[1]] - 0.5)
newyend = 2 * (rescale01y(y, yend)[[2]] - 0.5)
return(list(newy, newyend))
}
#' Retrieve function from global environment
#'
#' This function retrieves a function from a list.
#'
#' @return The function retrieved either directly from the list or from the global environment.
#'
#' @keywords internal
get_scale <- function(x) {
stopifnot(is.list(x))
fnc <- x[[1]]
if (is.character(fnc)) {
fnc <- get(unique(x)[1], envir = globalenv(), mode = "function")
}
stopifnot(is.function(fnc))
fnc
}
#' Retrieve scaling function
#'
#' This function checks whether a custom scaling function (`dx`) has been provided.
#'
#' @return A logical value:
#' `FALSE` if `dx` is `NULL` (no scaling function provided) or
#' `TRUE` if `dx` is not the `identity` function (custom scaling function provided).
#'
#' @keywords internal
custom_scale <- function(dx){
if (is.null(dx)){
return(FALSE)
}
if (!identical(dx, identity)){
return(TRUE)
}
}
# Global variables declaration -------------------------------------------------
utils::globalVariables(c(".data", "na.omit", "value", "com", "x_minor", "geom_segment",
"ymin_minor", "ymax_minor", "geom_ribbon", "theme_bw", "labs",
"theme", "margin", "element_blank", "y_minor", "yend_minor",
"scale_x_discrete", "x_major", "y_major", "unique_coord",
"gpar"))
Any scripts or data that you put into this service are public.
Add the following code to your website.
For more information on customizing the embed code, read Embedding Snippets.