R/findDispatchMethodsS3.R

###########################################################################/**
# @RdocDefault findDispatchMethodsS3
#
# @title "Finds the S3 methods that a generic function would call"
#
# \description{
#  @get "title", ordered according to an S3 @see "base::class" @vector.
# }
#
# @synopsis
#
# \arguments{
#   \item{methodName}{A @character string specifying the name of a
#     generic function.}
#   \item{classNames}{A @character @vector of @see "base::class" names.}
#   \item{firstOnly}{If @TRUE, only the first method is returned.}
#   \item{...}{Not used.}
# }
#
# \value{
#   Returns a names @list structure.
# }
#
# \seealso{
#   @see "getDispatchMethodS3".
# }
#
# @author
#
# @keyword programming
# @keyword methods
# @keyword internal
#*/###########################################################################
setMethodS3("findDispatchMethodsS3", "default", function(methodName, classNames, firstOnly=FALSE, ...) {
  # Argument 'methodName':
  methodName <- as.character(methodName)
  if (length(methodName) == 0) {
    stop("Argument 'methodName' is empty.")
  }
  if (length(methodName) > 1) {
    stop("Argument 'methodName' must only contain one element: ", paste(head(methodName), collapse=", "))
  }

  # Argument 'classNames':
  classNames <- as.character(classNames)
  if (length(classNames) == 0) {
    stop("Argument 'classNames' is empty.")
  }

  # Argument 'firstOnly':
  firstOnly <- as.logical(firstOnly)


  res <- list()
  for (kk in seq_along(classNames)) {
    className <- classNames[kk]
    fcnName <- paste(methodName, className, sep=".")
    obj <- do.call(getAnywhere, list(fcnName))
    if (length(obj$objs) == 0) {
      # No matching objects
      next
    }

    # WORKAROUND: In R (< 3.1.?) there is a bug in getAnywhere()
    # causing it to return garbage in parts of the 'objs' list.
    hasBug <- (length(obj$objs) > length(obj$where))
    if (hasBug) {
      ## Rebuild 'objs' manually
      n <- length(obj$where)
      obj$objs <- vector("list", length=n)
      for (ii in seq_len(n)) {
        where <- obj$where[[ii]]
        tryCatch({
          if (grepl("^namespace:", where)) {
            env <- asNamespace(gsub("^namespace:", "", where))
          } else {
            env <- as.environment(where)
          }
          if (exists(fcnName, envir=env)) {
            obj$objs[[ii]] <- get(fcnName, envir=env)
          }
        }, error = function(ex) {})
      } # for (ii ...)
    }

    # Keep only functions
    keep <- which(sapply(obj$objs, FUN=is.function))
    if (length(keep) == 0) {
      # No functions
      next
    }

    # Keep the first function
    first <- keep[1]
    fcn <- obj$objs[[first]]
    where <- obj$where[first]

    resKK <- list()
    resKK$class <- className
    resKK$name <- methodName
    resKK$fullname <- fcnName
    resKK$fcn <- fcn
    resKK$where <- obj$where

    res[[className]] <- resKK

    # Return only the first match?
    if (firstOnly) {
      break
    }
  } # for (kk ...)

  res
}, private=TRUE) # findDispatchMethodsS3()

Try the R.methodsS3 package in your browser

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

R.methodsS3 documentation built on June 14, 2022, 1:06 a.m.