R/getReciprocalOverlaps.R

Defines functions gr2grPercOverlap getReciprocalOverlaps

Documented in getReciprocalOverlaps gr2grPercOverlap

#' Function to calculate percentage of reciprocal overlap between two sets of ranges.
#'
#' @param query A \code{\link{GRanges-class}} object.
#' @param subject A \code{\link{GRanges-class}} object.
#' @param thresh A percentage threshold for a required overlap.
#' @param report Set to 'both' if you want to return merged query and subject ranges or select only one of them.
#' @param index An user defiend ID to disntiguish calculated overlaps.
#' @return A \code{\link{GRanges-class}} object.
#' @author David Porubsky
#' @export

getReciprocalOverlaps <- function(query, subject, thresh=50, report='both', index='') {
  ## Get overlaps between query and subject ranges
  hits <- findOverlaps(query, subject)
  ## Initialize exported variables (3new meta-columns)
  query$perc.overlap <- 0
  query$toGR <- query[,0]
  query$idx <- paste0(1:length(query), ".1")
  subject$perc.overlap <- 0
  subject$toGR <- subject[,0]
  subject$idx <- paste0(1:length(subject), ".2")
  for (i in seq_along(hits)) {
    ## Calculate % overlaps for a pair of ranges
    hit <- hits[i]
    percOverlap <- gr2grPercOverlap(gr1=query[queryHits(hit)], gr2=subject[subjectHits(hit)])
    
    ## Record only max overlap ranges
    if (query[queryHits(hit)]$perc.overlap < percOverlap) {
      query[queryHits(hit)]$perc.overlap <- percOverlap
      query[queryHits(hit)]$toGR <- subject[subjectHits(hit)][,0]
    }
    
    if (subject[subjectHits(hit)]$perc.overlap < percOverlap) {
      subject[subjectHits(hit)]$perc.overlap <- percOverlap
      subject[subjectHits(hit)]$toGR <- query[queryHits(hit)][,0]
    }  
    
    ## Assign the same index to ranges that meet reciprocal overlap threshold  
    if (percOverlap >= thresh) {
      query[queryHits(hit)]$idx <- queryHits(hit)
      subject[subjectHits(hit)]$idx <- queryHits(hit)
    }
  }
  
  ## Add index to newly added metacolumns (perc.overlap, toGR, idx)
  if (is.character(index) & nchar(index) > 0) {
    ## Add index to query ranges
    col.names.query <- names(mcols(query))
    to.modif <- (length(col.names.query)-2) : length(col.names.query)
    col.names.query[to.modif] <- paste0(col.names.query[to.modif], "_", index)
    names(mcols(query)) <- col.names.query
    ## Add index to subject ranges
    col.names.subject <- names(mcols(subject))
    to.modif <- (length(col.names.subject)-2) : length(col.names.subject)
    col.names.subject[to.modif] <- paste0(col.names.subject[to.modif], "_", index)
    names(mcols(subject)) <- col.names.subject
  }
  
  if (report == 'query') {
    return(query)
  } else if (report == 'subject') {
    return(subject)
  } else {
    ## Keep only shared meta-columns
    keep.mcols <- names(mcols(query))[names(mcols(query)) == names(mcols(subject))]
    return(c(query[,keep.mcols], subject[,keep.mcols]))
  } 
}


#' Function to calculate percentage overlap between two ranges (gr1 & gr2).
#'
#' @param gr1 A \code{\link{GRanges-class}} object.
#' @param gr2 A \code{\link{GRanges-class}} object. 
#' @return A \code{\link{GRanges-class}} object.
#' @author David Porubsky

gr2grPercOverlap <- function(gr1, gr2) {
  intersection <- IRanges::intersect(gr1, gr2)
  if (length(intersection) > 0) {
    gr1.overlap <- (width(intersection)/width(gr1))*100
    gr2.overlap <- (width(intersection)/width(gr2))*100
    recip.overlap <- min(gr1.overlap, gr2.overlap)
  } else {
    recip.overlap <- 0
  }  
  return(recip.overlap)
}
daewoooo/primatR documentation built on Oct. 22, 2024, 9:41 p.m.