R/posterior_linpred.R

#' Posterior Distribution of the Linear Predictor
#'
#' Extract posterior draws of the linear predictor for a \code{ubmsFit}
#' submodel, possibly transformed by the inverse-link function.
#'
#' @param object A fitted model of class \code{ubmsFit}
#' @param transform Should the linear predictor be transformed using the
#'  inverse link function?
#' @param submodel The name of the submodel, as a character string, for
#'  which to calculate the linear predictor
#' @param newdata Optional data frame of newdata to use when calculating the linear
#'  predictor. If not provided, the model matrix is used.
#' @param draws An integer indicating the number of draws to return. The
#'   default and maximum number of draws is the size of the posterior sample.
#' @param re.form If \code{NULL}, any estimated group-level parameters ("random
#'   effects") are included. If \code{NA}, they are ignored
#' @param ... Currently ignored
#'
#' @return A matrix of simulations from the posterior predictive distribution
#'  of the linear predictor. The dimensions are \code{draws} by number of
#'  linear predictor values (e.g., number of sites or number of observations).
#'
#' @aliases posterior_linpred
#' @method posterior_linpred ubmsFit
#' @importFrom rstantools posterior_linpred
#' @include fit.R
#' @export
setMethod("posterior_linpred", "ubmsFit",
          function(object, transform=FALSE, submodel, newdata=NULL, draws=NULL,
                   re.form=NULL, ...){

  check_type(submodel, submodel_types(object))
  samp_inds <- get_samples(object, draws)

  sim_lp(object, submodel=submodel, transform=transform, newdata=newdata,
         samples=samp_inds, re.form=re.form)
})

# Method for simulating linear predictor---------------------------------------

setGeneric("sim_lp", function(object, ...) standardGeneric("sim_lp"))


setMethod("sim_lp", "ubmsFit", function(object, submodel, transform, newdata,
                                        samples, re.form, ...){
  sm <- object[submodel]
  beta <- extract(object, beta_par(sm))[[1]]

  if(inherits(sm, "ubmsSubmodelScalar")){
    lp <- matrix(beta, nrow=1)
  } else{
    lp <- model.matrix(sm, newdata) %*% t(beta[samples,,drop=FALSE]) +
      model_offset(sm, newdata)

    if(has_random(sm) & is.null(re.form)){
      b <- extract(object, b_par(sm))[[1]]
      lp <- lp + Z_matrix(sm, newdata) %*% t(b[samples,,drop=FALSE])
    } else if(has_spatial(sm) & is.null(re.form)){
      if(!is.null(newdata)){
        stop("To use newdata with spatial model, must set re.form=NA", call.=FALSE)
      }
      newdata <- rbind(sm@data, sm@data_aug)
      lp_raw <- model.matrix(sm, newdata) %*% t(beta[samples,,drop=FALSE]) +
        model_offset(sm, newdata)
      theta <- extract(object, b_par(sm))[[1]]
      Kmat <- spatial_matrices(sm)$Kmat
      lp_raw <- lp_raw + Kmat %*% t(theta[samples,,drop=FALSE])
      lp <- matrix(NA, nrow(lp_raw), ncol(lp_raw))
      lp[!sm@sites_aug,] <- lp_raw[1:nrow(sm@data),,drop=FALSE]
      if(sum(sm@sites_aug)>0){
        lp[sm@sites_aug] <- lp_raw[(nrow(sm@data)+1):nrow(lp_raw),,drop=FALSE]
      }
    }
  }
  if(transform) lp <- do.call(sm@link, list(lp))
  t(unname(lp))
})


# Method simulating state parameter (special case of sim_lp)-------------------

setGeneric("sim_state", function(object, ...) standardGeneric("sim_state"))

setMethod("sim_state", "ubmsFit", function(object, samples, ...){
  out <- sim_lp(object, transform=TRUE, submodel="state", newdata=NULL,
                samples, re.form=NULL)
  if(has_spatial(object["state"])){
    out <- out[,!object["state"]@sites_aug,drop=FALSE]
  }
  out
})


# Method simulating detection probability (special case of sim_lp)-------------

setGeneric("sim_p", function(object, samples, ...) standardGeneric("sim_p"))

setMethod("sim_p", "ubmsFit", function(object, samples, ...){
  out <- sim_lp(object, "det", transform=TRUE, newdata=NULL,
                samples=samples, re.form=NULL)
  out[,object["det"]@missing] <- NA
  out
})

# Version of unmarked's getP method--------------------------------------------

#' @importFrom unmarked getP
setMethod("getP", "ubmsFit", function(object, draws=NULL, ...){
  samples <- get_samples(object, draws)
  resp <- object@response
  praw <- t(sim_p(object, samples))
  praw <- array(praw, c(resp@max_obs, nrow(resp@y), length(samples)))
  aperm(praw, c(2,1,3))
})
kenkellner/ubms documentation built on March 1, 2025, 7:02 a.m.