R/c_weightedStudySpecification.R

#' @include WeightedStudySpecification.R
NULL

##' @title Concatenate weights
##'
##' @description Given several variations of weights generated from a single
##'   \code{StudySpecification}, combine into a single weight.
##'
##' @details Concatenating \code{WeightedStudySpecification} objects with [c()]
##'   requires both individual \code{WeightedStudySpecification} objects to come
##'   from the same \code{StudySpecification} and have the same target (e.g all
##'   created with [ate()] or all created with [ett()], no mixing-and-matching).
##'   All arguments to [c()] must be \code{WeightedStudySpecification}.
##'
##'   \code{WeightedStudySpecification} objects may be concatenated together
##'   even without having the same \code{@dichotomy} slot. This procedure only
##'   prompts a warning for differing dichotomies if the argument
##'   \code{warn_dichotomy_not_equal} is set to \code{TRUE}.
##' @param x, .. a \code{WeightedStudySpecification} object, typically created
##'   from [ate()] or [ett()]
##' @param ... any number of additional \code{WeightedStudySpecification}
##'   objects with equivalent \code{StudySpecification} to \code{x} and
##'   eachother
##' @param warn_dichotomy_not_equal if \code{FALSE} (default),
##'   \code{WeightedStudySpecification}s are considered equivalent even if their
##'   \code{dichotomy} differs. If \code{TRUE}, a warning is produced.
##' @return A numeric \code{vector} with the weights concatenated in the input
##'   order.
##' @export
##' @importFrom methods slot
##' @importFrom stats formula
##' @examples
##' data(simdata)
##' spec <- rct_spec(z ~ unit_of_assignment(uoa1, uoa2), data = simdata)
##' w1 <- ate(spec, data = simdata[1:30,])
##' w2 <- ate(spec, data = simdata[31:40,])
##' w3 <- ate(spec, data = simdata[41:50,])
##' c_w <- c(w1, w2, w3)
##' c(length(w1), length(w2), length(w3), length(c_w))
##'
##' spec <- rct_spec(dose ~ unit_of_assignment(uoa1, uoa2), data = simdata)
##' w1 <- ate(spec, data = simdata[1:10, ], dichotomy = dose >= 300 ~ .)
##' w2 <- ate(spec, data = simdata[11:30, ], dichotomy = dose >= 200 ~ .)
##' w3 <- ate(spec, data = simdata[31:50, ], dichotomy = dose >= 100 ~ .)
##' c_w <- c(w1, w2, w3)
setMethod("c", signature(x = "WeightedStudySpecification"),
          function(x, ..., warn_dichotomy_not_equal = FALSE) {
  dots <- list(...)
  # x must be a WeightedStudySpecification to get here; ensure all other elements
  # are as well
  if (any_numeric <- any(vapply(dots, function(k) !inherits(k, "WeightedStudySpecification"), TRUE))) {
    message(paste("Concatenating a WeightedStudySpecification with a non-WeightedStudySpecification vector.",
                  "If non-WeightedStudySpecification vector has been formed from previous",
                  "concatenation of WeightedStudySpecification objects, equality of `StudySpecification`",
                  "slots cannot be confirmed"))
  }

  if (!any_numeric) {
    # Ensure all WeightedStudySpecifications have the same target
    targets <- c(x@target, lapply(dots, methods::slot, "target"))
    if (length(unique(targets)) > 1) {
      stop(paste("WeightedStudySpecifications can only be concatenated with",
                 "the same target"))
    }

    specifications <- c(x@StudySpecification, lapply(dots, methods::slot, "StudySpecification"))
    # temporarily remove calls from the StudySpecifications to avoid weird discrepancies here
    spectmp <- lapply(specifications, function(x) {
      x@call <- call("c") # placeholder for unique check
      x
    })

    if (length(unique(spectmp)) == 1) {
      if (warn_dichotomy_not_equal) {
        dichotomies <- Reduce(c, lapply(dots, methods::slot, "dichotomy"), init = x@dichotomy)
        if (length(unique(dichotomies)) > 1) {
          warning("Concatenating WeightedStudySpecifications with different `dichotomy` slots")
        }
      }
    } else {
      stop("Cannot combine WeightedStudySpecifications from differing StudySpecifications")
    }
  }

  return(Reduce(c, lapply(dots, methods::slot, ".Data"), init = x@.Data))
})

Try the propertee package in your browser

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

propertee documentation built on Aug. 22, 2025, 1:09 a.m.