R/VERBS-transmute_sd.R

Defines functions transmute_sd.data.table transmute_sd.EagerExprBuilder transmute_sd.ExprBuilder transmute_sd

Documented in transmute_sd transmute_sd.data.table transmute_sd.EagerExprBuilder transmute_sd.ExprBuilder

#' Transmute subset of data
#'
#' Like [transmute-table.express] but for a single call and maybe specifying `.SDcols`.
#'
#' @export
#'
transmute_sd <- function(.data, .SDcols = everything(), .how = identity, ...) {
    UseMethod("transmute_sd")
}

#' @rdname transmute_sd
#' @export
#' @importFrom rlang enquo
#' @importFrom rlang expr
#' @importFrom rlang is_call
#' @importFrom rlang quo_get_env
#' @importFrom rlang quo_get_expr
#' @importFrom rlang quo
#' @importFrom rlang quos
#'
#' @template data-arg
#' @param .SDcols See [data.table::data.table] and the details here.
#' @template transform-sd-args
#' @template parse-arg
#' @template chain-arg
#'
#' @details
#'
#' Like [transmute-table.express], this function never modifies the input by reference. This
#' function adds/chains a `select` expression that will be evaluated by [data.table::data.table],
#' possibly specifying the helper function `.transmute_matching`, which is assigned to the final
#' expression's evaluation environment when calling [end_expr()] (i.e., [ExprBuilder]'s `eval`
#' method).
#'
#' Said function supports two pronouns that can be used by `.how` and `.SDcols`:
#'
#' - `.COL`: the actual values of the column.
#' - `.COLNAME`: the name of the column currently being evaluated.
#'
#' Additionally, lambdas specified as formulas are also supported. In those cases, `.x` is
#' equivalent to `.COL` and `.y` to `.COLNAME`.
#'
#' Unlike a call like `DT[, (vars) := expr]`, `.SDcols` can be created dynamically with an
#' expression that evaluates to something that would be used in place of `vars` *without* eagerly
#' using the captured `data.table`. See the examples here or in [table.express-package].
#'
#' @examples
#'
#' data("mtcars")
#'
#' data.table::as.data.table(mtcars) %>%
#'     transmute_sd(~ grepl("^d", .y), ~ .x * 2)
#'
#' data.table::as.data.table(mtcars) %>%
#'     transmute_sd(~ is.numeric(.x), ~ .x * 2)
#'
transmute_sd.ExprBuilder <- function(.data, .SDcols = everything(), .how = identity, ...,
                                     .parse = getOption("table.express.parse", FALSE),
                                     .chain = getOption("table.express.chain", TRUE))
{
    which_quo <- rlang::enquo(.SDcols)
    how_quo <- rlang::enquo(.how)

    which_expr <- to_expr(rlang::quo_get_expr(which_quo), .parse = .parse)
    how_exprs <- to_expr(rlang::quo_get_expr(how_quo), .parse = .parse)

    all_sdcols <- identical(which_expr, rlang::expr(everything()))
    colon_call <- rlang::is_call(which_expr, ":")

    if (can_combine_lapply(which_quo, how_quo)) {
        hows <- standardize_lapplys(how_exprs, ..., .parse = .parse)
        ans <- .data$set_j(hows, .chain)

        if (!all_sdcols) {
            if (colon_call) {
                frame_append(ans, .SDcols = !!which_expr, .parse = FALSE)
            }
            else {
                frame_append(ans, .SDcols = c(!!which_expr), .parse = FALSE)
            }
        }
    }
    else {
        hows <- standardize_calls(how_exprs, rlang::quo_get_env(how_quo), ..., .parse = .parse)

        # just to avoid NOTE
        .transmute_matching <- EBCompanion$helper_functions$.transmute_matching

        clause <- rlang::expr(
            .transmute_matching(.SD, .which = rlang::quo(!!which_expr), .hows = rlang::quos(!!!hows))
        )

        ans <- .data$set_j(clause, .chain)
    }

    ans
}

#' @rdname transmute_sd
#' @export
#' @importFrom rlang caller_env
#'
#' @param .parent_env See [end_expr()]
#'
transmute_sd.EagerExprBuilder <- function(.data, ..., .parent_env = rlang::caller_env()) {
    end_expr.ExprBuilder(transmute_sd.ExprBuilder(.data, ...), .parent_env = .parent_env)
}

#' @rdname transmute_sd
#' @export
#' @importFrom rlang caller_env
#'
transmute_sd.data.table <- function(.data, ...) {
    eb <- ExprBuilder$new(.data)
    lazy_ans <- transmute_sd.ExprBuilder(eb, ...)
    end_expr.ExprBuilder(lazy_ans, .parent_env = rlang::caller_env())
}

Try the table.express package in your browser

Any scripts or data that you put into this service are public.

table.express documentation built on April 3, 2023, 5:43 p.m.