R/eval-bool.R

Defines functions stop_formula stop_bad_arith_op stop_bad_bool_op walk_operand eval_and eval_or eval_bang

eval_bang <- function(expr, data_mask, context_mask) {
  x <- walk_data_tree(expr[[2]], data_mask, context_mask)

  vars <- data_mask$.__tidyselect__.$internal$vars
  error_call <- mask_error_call(data_mask)
  sel_complement(x, vars, error_call = error_call)
}

eval_or <- function(expr, data_mask, context_mask) {
  x <- walk_operand(expr[[2]], data_mask, context_mask)
  y <- walk_operand(expr[[3]], data_mask, context_mask)

  sel_union(x, y)
}

eval_and <- function(expr, data_mask, context_mask) {
  x <- expr[[2]]
  y <- expr[[3]]

  if (is_symbol(x) && is_symbol(y)) {
    x_name <- as_string(x)
    y_name <- as_string(y)

    x <- eval_sym(x, data_mask, context_mask, strict = TRUE)
    y <- eval_sym(y, data_mask, context_mask, strict = TRUE)

    if (!is_function(x) && !is_function(y)) {
      cli::cli_abort(
        c(
          "Can't take the intersection of two columns.",
          # can't use {.code}: https://github.com/r-lib/cli/issues/422
          i = "`{x_name} & {y_name}` is always an empty selection."
        ),
        call = mask_error_call(data_mask)
      )
    }
  }

  x <- walk_operand(x, data_mask, context_mask)
  y <- walk_operand(y, data_mask, context_mask)

  sel_intersect(x, y)
}

walk_operand <- function(expr, data_mask, context_mask) {
  if (is_symbol(expr)) {
    expr <- eval_sym(expr, data_mask, context_mask, strict = TRUE)
  }
  walk_data_tree(expr, data_mask, context_mask)
}

stop_bad_bool_op <- function(bad, ok, call) {
  cli::cli_abort(
    c(
      "Can't use scalar {.code {bad}} in selections.",
      i = "Do you need {.arg {ok}} instead?"
    ),
    call = call
  )
}

stop_bad_arith_op <- function(op, call) {
  cli::cli_abort(
    "Can't use arithmetic operator `{op}` in selection context.",
    call = call
  )
}

stop_formula <- function(expr, call) {
  f <- as_label(expr)
  cli::cli_abort(
    c(
      "Formula shorthand must be wrapped in `where()`.",
      "",
      " " = "  # Bad",
      " " = "  data %>% select({f})",
      "",
      " " = "  # Good",
      " " = "  data %>% select(where({f}))"
    ),
    call = call
  )
}
lionel-/selectr documentation built on March 14, 2024, 10:04 p.m.