diff --git a/NAMESPACE b/NAMESPACE index 323e34ba..80d34c6b 100644 --- a/NAMESPACE +++ b/NAMESPACE @@ -96,6 +96,7 @@ export(exp_trans) export(expand_range) export(extended_breaks) export(format_format) +export(format_log) export(fullseq) export(gradient_n_pal) export(grey_pal) diff --git a/NEWS.md b/NEWS.md index 9646d4a5..6e2d1bb7 100644 --- a/NEWS.md +++ b/NEWS.md @@ -1,5 +1,8 @@ # scales (development version) +* `label_log()` has a `signed` argument for displaying negative numbers + (@teunbrand, #421). + # scales 1.3.0 ## Better type support diff --git a/R/label-log.R b/R/label-log.R index 802d2a13..e31e2bb8 100644 --- a/R/label-log.R +++ b/R/label-log.R @@ -1,10 +1,18 @@ #' Label numbers in log format (10^3, 10^6, etc) #' -#' `label_log()` displays numbers as base^exponent, using superscript formatting. +#' `label_log()` and `format_log()` display numbers as base^exponent, using +#' superscript formatting. `label_log()` returns expressions suitable for +#' labelling in scales, whereas `format_log()` returns deparsed text. #' +#' +#' @param x A numeric vector to format #' @param base Base of logarithm to use #' @param digits Number of significant digits to show for the exponent. Argument #' is passed on to [base::format()]. +#' @param signed Should a `+` or `-` be displayed as a prefix? The +#' default, `NULL`, displays signs if there are zeroes or negative numbers +#' present. +#' @param ... Passed on to `format()`. #' @inherit label_number return #' @seealso [breaks_log()] for the related breaks algorithm. #' @export @@ -12,19 +20,43 @@ #' @examples #' demo_log10(c(1, 1e5), labels = label_log()) #' demo_log10(c(1, 1e5), breaks = breaks_log(base = 2), labels = label_log(base = 2)) -label_log <- function(base = 10, digits = 3) { +#' format_log(c(0.1, 1, 10)) +label_log <- function(base = 10, digits = 3, signed = NULL) { function(x) { - if (length(x) == 0) { - return(expression()) - } - - exponent <- format(log(x, base = base), digits = digits) - text <- paste0(base, "^", exponent) + text <- format_log(x, base = base, signed = signed, digits = digits) ret <- parse_safe(text) - # restore NAs from input vector ret[is.na(x)] <- NA - ret } } + +#' @export +#' @rdname label_log +format_log <- function(x, base = 10, signed = NULL, ...) { + + if (length(x) == 0) { + return(character()) + } + prefix <- rep("", length(x)) + finites <- x[is.finite(x)] + + signed <- signed %||% any(finites <= 0) + if (signed) { + sign <- sign(x) + prefix[sign == +1] <- "+" + prefix[sign == -1] <- "-" + x <- abs(x) + x[x == 0] <- 1 + } + + exponent <- format(zapsmall(log(x, base = base)), ...) + text <- paste0(prefix, base, "^", exponent) + + if (signed) { + text[sign == 0] <- "0" + } + text[is.na(x)] <- NA + + text +} diff --git a/man/label_log.Rd b/man/label_log.Rd index daf576fb..d0dee6ab 100644 --- a/man/label_log.Rd +++ b/man/label_log.Rd @@ -2,15 +2,26 @@ % Please edit documentation in R/label-log.R \name{label_log} \alias{label_log} +\alias{format_log} \title{Label numbers in log format (10^3, 10^6, etc)} \usage{ -label_log(base = 10, digits = 3) +label_log(base = 10, digits = 3, signed = NULL) + +format_log(x, base = 10, signed = NULL, ...) } \arguments{ \item{base}{Base of logarithm to use} \item{digits}{Number of significant digits to show for the exponent. Argument is passed on to \code{\link[base:format]{base::format()}}.} + +\item{signed}{Should a \code{+} or \code{-} be displayed as a prefix? The +default, \code{NULL}, displays signs if there are zeroes or negative numbers +present.} + +\item{x}{A numeric vector to format} + +\item{...}{Passed on to \code{format()}.} } \value{ All \code{label_()} functions return a "labelling" function, i.e. a function that @@ -23,11 +34,14 @@ they work similarly for all scales, including those that generate legends rather than axes. } \description{ -\code{label_log()} displays numbers as base^exponent, using superscript formatting. +\code{label_log()} and \code{format_log()} display numbers as base^exponent, using +superscript formatting. \code{label_log()} returns expressions suitable for +labelling in scales, whereas \code{format_log()} returns deparsed text. } \examples{ demo_log10(c(1, 1e5), labels = label_log()) demo_log10(c(1, 1e5), breaks = breaks_log(base = 2), labels = label_log(base = 2)) +format_log(c(0.1, 1, 10)) } \seealso{ \code{\link[=breaks_log]{breaks_log()}} for the related breaks algorithm. diff --git a/tests/testthat/test-label-log.R b/tests/testthat/test-label-log.R index 0a555ae0..2aee4f49 100644 --- a/tests/testthat/test-label-log.R +++ b/tests/testthat/test-label-log.R @@ -5,4 +5,5 @@ test_that("label_log() returns expression", { expect_equal(label_log()(c(0.1, 10)), expression(10^-1, 10^1)) expect_equal(label_log(base = 2)(8), expression(2^3)) expect_equal(label_log(base = 2, digits = 3)(7), expression(2^2.81)) + expect_equal(label_log(signed = TRUE)(c(-100, 100)), expression(-10^2, +10^2)) })