R/summary.R

Defines functions summary.synthdid_estimate synthdid_controls

Documented in summary.synthdid_estimate synthdid_controls

#' Outputs a table of important synthetic controls and their corresponding weights, sorted by weight.
#' The table is truncated to exclude synthetic controls that do not matter for any estimate ---
#' for each estimate, the truncated controls may have total weight no larger that 1-mass.
#' @param estimates, a list of estimates output by synthdid_estimate. Or a single estimate.
#' @param sort.by, the index of the estimate to sort by. Defaults to 1.
#' @param mass, which controls the length of the table. Defaults to 0.9.
#' @param weight.type, 'omega' for units, 'lambda' for time periods
#' @export synthdid_controls
synthdid_controls = function(estimates, sort.by = 1, mass = .9, weight.type = 'omega') {
  if (class(estimates) == 'synthdid_estimate') { estimates = list(estimates) }
  if (is.null(names(estimates))) { names(estimates) = sprintf('estimate %d', 1:length(estimates)) }
  if (!weight.type %in% c('omega', 'lambda')) { stop('weight.type must be "omega" or "lambda"') } 
  weights = do.call(cbind, lapply(estimates, function(est) { attr(est, 'weights')[[weight.type]] }))
  if (is.null(dim(weights))) { dim(weights) = c(length(weights), 1) }

  Y = attr(estimates[[1]], 'setup')$Y
  o = rev(order(weights[, sort.by]))
  tab = weights[o, , drop = FALSE]
  rownames(tab) = if(weight.type == 'omega') { rownames(Y)[o] } else { colnames(Y)[o] }
  colnames(tab) = names(estimates)

  # truncate table to retain a weight sum of at least mass for each unit
  tab.len = max(apply(tab, 2, function(col) { Position(function(x) { x >= mass }, cumsum(col), nomatch=nrow(tab)) }))
  tab[1:tab.len, , drop=FALSE]
}

#' Summarize a synthdid object
#' @param object The object to summarize
#' @param weight.digits The number of digits to use when displaying weights (omega, lambda)
#' @param fast Be fast but less accurate, e.g. jackknife instead of bootstrap se estimate
#' @param ... Additional arguments (currently ignored).
#' @method summary synthdid_estimate
#' @export
summary.synthdid_estimate = function(object, weight.digits=3, fast=FALSE, ...) {
  N0 = attr(object, 'setup')$N0
  T0 = attr(object, 'setup')$T0
  list(estimate = c(object),
    se = sqrt(if(fast) { vcov(object, method = 'jackknife') } else { vcov(object) }),
    controls = round(synthdid_controls(object, weight.type='omega'),  digits=weight.digits),
    periods  = round(synthdid_controls(object, weight.type='lambda'), digits=weight.digits),
    dimensions = c( N1 = nrow(Y(object))-N0, N0 = N0, N0.effective = round(1 / sum(omega(object)^2),  weight.digits),
		    T1 = ncol(Y(object))-T0, T0 = T0, T0.effective = round(1 / sum(lambda(object)^2), weight.digits)))
}
synth-inference/synthdid documentation built on Jan. 26, 2024, 7:21 a.m.