R/ConditionalPower.R

Defines functions Power ConditionalPower

Documented in ConditionalPower Power

#' (Conditional) Power of a Design
#'
#' This score evaluates \ifelse{html}{\out{P[X<sub>2</sub> > c2(design, X<sub>1</sub>) | X<sub>1</sub> = x<sub>1</sub>]}}{\eqn{\boldsymbol{P}[X_2 > c_2(design, X_1)|X_1 = x_1]}}.
#' Note that the distribution of \ifelse{html}{\out{X<sub>2</sub>}}{\eqn{X_2}} is the posterior predictive after
#' observing \ifelse{html}{\out{X<sub>1</sub> = x<sub>1</sub>}}{\eqn{X_1 = x_1}}.
#'
#' @template dist
#' @template prior
#' @template design
#' @template s
#' @template x1
#' @template optimization
#' @template dotdotdot
#'
#' @seealso \code{\link{Scores}}
#'
#' @examples
#' prior <- PointMassPrior(.4, 1)
#' cp <- ConditionalPower(Normal(), prior)
#' evaluate(
#'    cp,
#'    TwoStageDesign(50, .0, 2.0, 50, 2.0, order = 5L),
#'    x1 = 1
#' )
#' # these two are equivalent:
#' expected(cp, Normal(), prior)
#' Power(Normal(), prior)
#'
#' @aliases ConditionalPower
#' @exportClass ConditionalPower
setClass("ConditionalPower", representation(
        distribution = "DataDistribution",
        prior        = "Prior"
    ),
    contains = "ConditionalScore")


#' @template label
#'
#' @rdname ConditionalPower-class
#' @export
ConditionalPower <- function(dist, prior, label = "Pr[x2>=c2(x1)|x1]") {
    new("ConditionalPower", distribution = dist, prior = prior, label = label)
}



#' @rdname ConditionalPower-class
#' @export
Power <- function(dist, prior, label = "Pr[x2>=c2(x1)]") {
    expected(ConditionalPower(dist, prior), dist, prior, label = label)
}



#' @rdname ConditionalPower-class
#' @export
setMethod("evaluate", signature("ConditionalPower", "TwoStageDesign"),
    function(s, design, x1, optimization = FALSE, ...) {
        if (optimization) { # use design-specific implementation
            pivots <- scaled_integration_pivots(design)
            idx    <- lapply(x1, function(x) which.min(abs(x - pivots))) # more robust than ==; know x1 are pivots!
            sapply(
                1:length(x1),
                function(i) {
                    if (x1[i] < design@c1f) return(0)
                    if (x1[i] > design@c1e) return(1)
                    return(
                        expectation(
                            posterior( # candidate for memoisation
                                s@distribution,
                                s@prior,
                                pivots[idx[[i]]],
                                design@n1,
                                ...
                            ),
                            function(theta) 1 - cumulative_distribution_function( # candidate for memoisation
                                s@distribution,
                                design@c2_pivots[idx[[i]]],
                                if (is(design, 'GroupSequentialDesign')) design@n2_pivots else design@n2_pivots[idx[[i]]],
                                theta
                            )
                        )
                    )
                }
            )
        } else {
            sapply(
                x1,
                function(x1) expectation(
                    posterior(s@distribution, s@prior, x1, n1(design, round = TRUE), ...),
                    function(theta) 1 - cumulative_distribution_function(s@distribution, c2(design, x1), n2(design, x1, round = TRUE), theta)
                )
            )
        }
    })



# not user facing!
setMethod(".evaluate", signature("ConditionalPower", "TwoStageDesign"),
    function(s, design, ...) {
        pivots <- scaled_integration_pivots(design)
        n2 <- if (is(design, 'GroupSequentialDesign')) function(i) design@n2_pivots else function(i) design@n2_pivots[i]
        continuation <- sapply(
            1:length(pivots),
            function(i) {
                expectation(
                    posterior(s@distribution, s@prior, pivots[i], design@n1),
                    function(theta) 1 - cumulative_distribution_function(s@distribution, design@c2_pivots[i], n2(i), theta)
                )
            }
        )
        list(
            early_futility = 0,
            pivots         = continuation,
            early_efficacy = 1
        )
    })

setMethod(".evaluate", signature("ConditionalPower", "OneStageDesign"),
    function(s, design, ...) {
        list(
            early_futility = 0,
            pivots         = NaN,
            early_efficacy = 1
        )
    })
kkmann/adoptr documentation built on Feb. 3, 2024, 6:55 p.m.