R/addAnnotation.R

Defines functions flipQueryAnnotation addAnnotation

Documented in addAnnotation flipQueryAnnotation

#' Add annotation ranges to a SVbyEye plot.
#'
#' This function takes a \code{ggplot2} object generated using \code{\link{plotMiro}} function and adds extra annotation on top of query
#' or target coordinates. These ranges are specified in 'annot.gr' object and are visualized either as arrowheads or rectangles.
#'
#' @param ggplot.obj A \code{ggplot2} object generated using \code{\link{plotMiro}} function.
#' @param annot.gr A \code{\link{GRanges-class}} object with a set of ranges to be added as extra annotation.
#' @param shape A user defined shape ranges in 'annot.gr' are visualized, either 'arrowhead' or 'rectangle'.
#' @param fill.by A name of an extra field present in 'annot.gr' to be used to define color scheme.
#' @param label.by A name of an extra field present in 'annot.gr' to be used as a label over each annotation range.
#' @param color.palette A discrete color palette defined as named character vector (elements = colors, names = discrete levels).
#' @param max.colors A maximum number of discrete color levels for which legend will be reported. If more than that legend will be removed.
#' @param coordinate.space A coordinate space ranges in 'annot.gr' are reported, either 'target', 'query' or 'self'.
#' @param annotation.group A name of an extra field present in 'annot.gr' to be used to connect set of ranges of the same group
#' by a straight black line (Useful to plot exons from one or multiple genes).
#' @param annotation.level A \code{numeric} that defines a fraction of y-axis to be the y-axis position for the annotation track (Default : `0.05`).
#' @param offset.annotation Set to \code{TRUE} if subsequent annotation ranges should be offsetted below and above the midline.
#' @param annotation.label A \code{character} string to be used as a label to added annotation track.
#' @param y.label.id A user defined metadata column id within `annot.gr` that for each annotation range contains
#' corresponding y-axis label.
#' @return A \code{ggplot2} object.
#' @import ggplot2
#' @importFrom grid unit
#' @importFrom methods is
#' @importFrom scales comma
#' @importFrom wesanderson wes_palette
#' @importFrom randomcoloR randomColor
#' @importFrom gggenes geom_gene_arrow
#' @importFrom ggnewscale new_scale_fill new_scale_color
#' @importFrom IRanges IRanges ranges
#' @importFrom GenomicRanges start end sort makeGRangesFromDataFrame
#' @importFrom GenomeInfoDb seqnames
#' @author David Porubsky
#' @export
#' @examples
#' ## Get PAF to plot
#' paf.file <- system.file("extdata", "test1.paf", package = "SVbyEye")
#' ## Read in PAF
#' paf.table <- readPaf(paf.file = paf.file, include.paf.tags = TRUE, restrict.paf.tags = "cg")
#' ## Make a plot
#' plt <- plotMiro(paf.table = paf.table)
#' ## Load target annotation file
#' target.annot <- system.file("extdata", "test1_target_annot.txt", package = "SVbyEye")
#' target.annot.df <- read.table(target.annot, header = TRUE, sep = "\t", stringsAsFactors = FALSE)
#' target.annot.gr <- GenomicRanges::makeGRangesFromDataFrame(target.annot.df)
#' ## Add target annotation as arrowhead
#' plt <- addAnnotation(ggplot.obj = plt, annot.gr = target.annot.gr, coordinate.space = "target")
#' ## Load query annotation file
#' query.annot <- system.file("extdata", "test1_query_annot.txt", package = "SVbyEye")
#' query.annot.df <- read.table(query.annot, header = TRUE, sep = "\t", stringsAsFactors = FALSE)
#' query.annot.gr <- GenomicRanges::makeGRangesFromDataFrame(query.annot.df)
#' ## Add query annotation as rectangle
#' addAnnotation(
#'     ggplot.obj = plt, annot.gr = query.annot.gr, shape = "rectangle",
#'     coordinate.space = "query"
#' )
#' ## Add segmental duplication annotation
#' plt <- plotMiro(paf.table = paf.table)
#' sd.annot <- system.file("extdata", "test1.sd.annot.RData", package = "SVbyEye")
#' sd.annot.gr <- get(load(sd.annot))
#' ## Create a custom discrete levels
#' sd.categ <- findInterval(sd.annot.gr$fracMatch, vec = c(0.95, 0.98, 0.99))
#' sd.categ <- dplyr::recode(sd.categ, "0" = "<95%", "1" = "95-98%", "2" = "98-99%", "3" = ">=99%")
#' sd.categ <- factor(sd.categ, levels = c("<95%", "95-98%", "98-99%", ">=99%"))
#' sd.annot.gr$sd.categ <- sd.categ
#' ## Create a custom color palette
#' color.palette <- c(
#'     "<95%" = "gray72", "95-98%" = "gray47", "98-99%" = "#cccc00",
#'     ">=99%" = "#ff6700"
#' )
#' ## Add annotation to the plot
#' addAnnotation(
#'     ggplot.obj = plt, annot.gr = sd.annot.gr, fill.by = "sd.categ",
#'     color.palette = color.palette, coordinate.space = "target"
#' )
#' ## Offset annotation ranges
#' addAnnotation(
#'     ggplot.obj = plt, annot.gr = sd.annot.gr, fill.by = "sd.categ",
#'     color.palette = color.palette, coordinate.space = "target", offset.annotation = TRUE
#' )
#' ## Add label to the added annotation
#' addAnnotation(
#'     ggplot.obj = plt, annot.gr = sd.annot.gr, fill.by = "sd.categ",
#'     color.palette = color.palette, coordinate.space = "target", annotation.label = "SD"
#' )
#'
#' ## Add gene-like annotation
#' test.gr <- GenomicRanges::GRanges(
#'     seqnames = "target.region",
#'     ranges = IRanges::IRanges(
#'         start = c(19000000, 19030000, 19070000),
#'         end = c(19010000, 19050000, 19090000)
#'     ),
#'     ID = "gene1"
#' )
#' addAnnotation(
#'     ggplot.obj = plt, annot.gr = test.gr, coordinate.space = "target",
#'     shape = "rectangle", annotation.group = "ID", offset.annotation = TRUE,
#'     fill.by = "ID"
#' )
#' ## Add gene names
#' addAnnotation(
#'     ggplot.obj = plt, annot.gr = test.gr, coordinate.space = "target",
#'     shape = "rectangle", annotation.group = "ID", offset.annotation = TRUE,
#'     fill.by = "ID", label.by = "ID"
#' )
#'
addAnnotation <- function(ggplot.obj = NULL, annot.gr = NULL, shape = "arrowhead", fill.by = NULL, label.by = NULL, color.palette = NULL, max.colors = 20, coordinate.space = "target", annotation.group = NULL, annotation.level = 0.05, offset.annotation = FALSE, annotation.label = NULL, y.label.id = NULL) {
    ## Check user input ##
    stopifnot(methods::is(annot.gr, "GRanges"), length(annot.gr) > 0)

    ## Get plotted data
    gg.data <- ggplot.obj$data
    if (coordinate.space != "self") {
        target.id <- unique(gg.data$seq.name[gg.data$seq.id == "target"])
        query.id <- unique(gg.data$seq.name[gg.data$seq.id == "query"])
    }

    ## Get x-axis limits (expected to be always continuous)
    ## For x-axis range also consider user defined cartesian coordinates for the target region
    if (coordinate.space == "target") {
        # xlim <- range(c(gg.data$seq.pos[gg.data$seq.id == "target"], ggplot.obj$coordinates$limits$x))
        xlim <- range(c(
            gg.data$seq.pos[gg.data$seq.id == "target"],
            ggplot2::layer_scales(ggplot.obj)$x$limits
        ))
    } else {
        # xlim <- ggplot.obj$coordinates$limits$x
        # xlim <- ggplot2::layer_scales(ggplot.obj)$x$range$range
        xlim <- range(
            ggplot2::layer_scales(ggplot.obj)$x$range$range,
            ggplot2::layer_scales(ggplot.obj)$x$limits
        )
    }

    ## Get y-axis limits
    if ("ScaleContinuous" %in% class(ggplot2::layer_scales(ggplot.obj)$y)) { ## To finish!!!
        ylim <- ggplot2::layer_scales(ggplot.obj)$y$range$range
        ylabels <- ggplot2::layer_scales(ggplot.obj)$y$labels
        ybreaks <- ggplot2::layer_scales(ggplot.obj)$y$breaks
        if (length(ylabels) == 0) {
            ylabels <- ""
        }
        if (length(ybreaks) == 0) {
            ybreaks <- 0
        }
        # ylabels.ord <- ggplot2::layer_scales(ggplot.obj)$y$breaks
        ylabels <- ylabels[order(ybreaks)]
        ybreaks <- sort(ybreaks) ## [Testing]
        names(ybreaks) <- ylabels
    } else {
        stop("'addAnnotation' function works only for ggplot2 objects with continuous scale for both x and y-axis !!!")
    }

    ## Define the offset value to be the user defined fraction of the y-axis range [default: 0.05]
    offset <- diff(ylim) * annotation.level

    ## Subset annotation ranges to plot specific genomic positions
    if ("pos.genomic" %in% colnames(gg.data)) {
        limits <- gg.data %>%
            dplyr::group_by(.data$seq.id, .data$seq.name) %>%
            dplyr::reframe(gen.range = range(.data$pos.genomic))
        limits.l <- split(limits, limits$seq.name)
        annot.l <- list()
        for (i in seq_along(limits.l)) {
            lims <- limits.l[[i]]
            lims.gr <- GenomicRanges::GRanges(
                seqnames = unique(lims$seq.name),
                ranges = IRanges::IRanges(start = lims$gen.range[1], end = lims$gen.range[2])
            )
            seq.id <- unique(lims$seq.name)
            if (seq.id %in% as.character(GenomeInfoDb::seqnames(annot.gr))) {
                gr <- annot.gr[GenomeInfoDb::seqnames(annot.gr) == seq.id]
                # gr <- gr[start(gr) > min(lims$gen.range) & end(gr) < max(lims$gen.range)]
                # gr <- gr[start(gr) >= min(lims$gen.range) & end(gr) <= max(lims$gen.range)]
                gr <- subsetByOverlaps(gr, lims.gr) ## More permissive subsetting of annotation ranges!!!
                start(gr)[start(gr) < min(lims$gen.range)] <- min(lims$gen.range)
                end(gr)[end(gr) > max(lims$gen.range)] <- max(lims$gen.range)
                annot.l[[length(annot.l) + 1]] <- gr
            }
        }
        annot.gr <- do.call(c, annot.l)
    }

    ## Shift genomic positions in case multiple query or target sequences have been concatenated
    if ("pos.shift" %in% colnames(gg.data)) {
        pos.shifts <- gg.data %>%
            dplyr::group_by(.data$seq.id, .data$seq.name) %>%
            dplyr::summarize(pos.shift = unique(.data$pos.shift), .groups = "drop")
        # GenomicRanges::start(annot.gr) <- GenomicRanges::start(annot.gr) + pos.shifts$pos.shift[match(as.character(GenomeInfoDb::seqnames(annot.gr)), pos.shifts$seq.name)]
        # GenomicRanges::end(annot.gr) <- GenomicRanges::end(annot.gr) + pos.shifts$pos.shift[match(as.character(GenomeInfoDb::seqnames(annot.gr)), pos.shifts$seq.name)]
        new.start <- GenomicRanges::start(annot.gr) + pos.shifts$pos.shift[match(as.character(GenomeInfoDb::seqnames(annot.gr)), pos.shifts$seq.name)]
        new.end <- GenomicRanges::end(annot.gr) + pos.shifts$pos.shift[match(as.character(GenomeInfoDb::seqnames(annot.gr)), pos.shifts$seq.name)]
        ranges(annot.gr) <- IRanges::IRanges(start = new.start, end = new.end)
    }

    ## Get query and target coordinate ranges
    if (coordinate.space == "query" | coordinate.space == "target") {
        t.range <- range(gg.data$seq.pos[gg.data$seq.id == "target"])
        q.range <- range(gg.data$seq.pos[gg.data$seq.id == "query"])
        ## Adjust target ranges given the size difference with respect to query ranges
        range.offset <- diff(q.range) - diff(t.range)
        t.range[2] <- t.range[2] + range.offset ## Make a start position as offset and change only end position
    }
    ## Translate query coordinates to target coordinates
    if (coordinate.space == "query") {
        if (length(annot.gr) > 0) {
            ## Covert query to target coordinates
            new.start <- SVbyEye::q2t(x = start(annot.gr), q.range = q.range, t.range = t.range)
            new.end <- SVbyEye::q2t(x = end(annot.gr), q.range = q.range, t.range = t.range)
            new.ranges <- IRanges::IRanges(start = new.start, end = new.end)
            suppressWarnings(IRanges::ranges(annot.gr) <- new.ranges)
        }
    }

    ## Get annotation track offset
    if (coordinate.space == "target") {
        y.offset <- max(ylim) + offset
    } else if (coordinate.space == "query") {
        y.offset <- min(ylim) + -offset
    } else if (coordinate.space == "self") {
        y.offset <- min(ylim) + -offset
    } else {
        stop("Please specify 'coordinate.space' as either 'target', 'query' or 'self' !!!")
    }

    if (!is.null(annot.gr) & methods::is(annot.gr, "GRanges")) {
        ## Restrict annotation ranges to x-limits
        annot.gr <- annot.gr[GenomicRanges::start(annot.gr) >= xlim[1] & GenomicRanges::end(annot.gr) <= xlim[2]]
        if (length(annot.gr) > 0) {
            ## Offset overlapping annotation ranges up&down based on start position ##
            if (offset.annotation) {
                # annot.gr <- GenomicRanges::sort(annot.gr, ignore.strand = TRUE)
                if (coordinate.space == "target") {
                    # y.offset <- rep(y.offset + c(0, offset), times = ceiling(length(annot.gr) / 2))[seq_along(annot.gr)]
                    y.offset <- getAnnotationLevels(annot.gr = annot.gr, offset = y.offset, annotation.group = annotation.group, direction = "positive")
                } else if (coordinate.space == "query" | coordinate.space == "self") {
                    # y.offset <- rep(y.offset - c(0, offset), times = ceiling(length(annot.gr) / 2))[seq_along(annot.gr)]
                    y.offset <- getAnnotationLevels(annot.gr = annot.gr, offset = y.offset, annotation.group = annotation.group, direction = "negative")
                }
            }

            ## Prepare data for plotting ##
            ## Convert to data frame object
            annot.df <- as.data.frame(annot.gr)

            ## Add y-axis coordinates
            ## Match y-axis labels if to user defined ID column via 'y.label.id' parameter
            if (!is.null(y.label.id)) {
                if (y.label.id %in% colnames(annot.df)) {
                    # annot.df$y.offset <- match(annot.df[, eval(y.label.id)], ylabels) + offset ## Revert to if broken
                    annot.df$y.offset <- ybreaks[match(annot.df[, eval(y.label.id)], names(ybreaks))] + offset
                } else {
                    warning("User defined 'y.label.id' is not a valid metadata column in 'annot.gr', skipping !!!")
                    annot.df$y.offset <- y.offset
                }
            } else {
                annot.df$y.offset <- y.offset
            }

            ## Define color scale ##
            if (!is.null(fill.by)) {
                if (fill.by %in% colnames(annot.df)) {
                    ## Get color scales ##
                    ## Define continuous color scale
                    if (is.numeric(annot.df[, eval(fill.by)])) {
                        pal <- wesanderson::wes_palette("Zissou1", 100, type = "continuous")
                        col.scale <- "gradient"
                        ## Define discrete color scale
                    } else {
                        discrete.levels <- unique(annot.df[, eval(fill.by)])
                        n.uniq <- length(discrete.levels)
                        ## Get user define discrete color palette
                        if (all(discrete.levels %in% names(color.palette))) {
                            pal <- color.palette
                            if (length(pal) >= max.colors) {
                                col.scale <- "discreteNoLegend"
                            } else {
                                col.scale <- "discrete"
                            }
                            ## Create default discrete color palette
                        } else {
                            if (n.uniq <= max.colors) {
                                pal <- randomcoloR::randomColor(count = n.uniq)
                                col.scale <- "discrete"
                            } else {
                                warning("More than 20 color levels, legend won't be reported!!!")
                                col.scale <- "discreteNoLegend"
                            }
                        }
                    }
                } else {
                    stop("User defined 'fill.by' value is not a valid field in 'annot.gr' !!!")
                }
            } else {
                pal <- "black"
                col.scale <- "discreteNoLegend"
            }

            ## Connect annotation ranges based on group defined in 'annotation.group' ##
            if (!is.null(annotation.group)) {
                if (annotation.group %in% colnames(annot.df)) {
                    link.df <- annot.df %>%
                        dplyr::group_by(dplyr::across(dplyr::all_of(annotation.group))) %>%
                        dplyr::summarise(
                            seqnames = unique(seqnames),
                            start = min(start),
                            end = max(end),
                            y.offset = unique(y.offset)
                        )

                    plt <- ggplot.obj +
                        ggplot2::geom_segment(data = link.df, ggplot2::aes(x = start, xend = end, y = y.offset, yend = y.offset))
                } else {
                    annotation.group <- NULL
                }
            } else {
                plt <- ggplot.obj
            }

            ## Add annotation to the ggplot object ##
            if (shape == "arrowhead") {
                ## Make sure field 'strand' is defined
                if (!"strand" %in% colnames(annot.df)) {
                    annot.df$strand <- "*"
                }
                if (!is.null(fill.by)) {
                    plt <- plt + ggnewscale::new_scale_fill() + ggnewscale::new_scale_color() +
                        geom_arrowhead(data = annot.df, ggplot2::aes(xmin = start, xmax = end, y = y.offset, color = .data[[fill.by]], fill = .data[[fill.by]], strand = strand))
                } else {
                    plt <- plt + ggnewscale::new_scale_fill() + ggnewscale::new_scale_color() +
                        geom_arrowhead(data = annot.df, ggplot2::aes(xmin = start, xmax = end, y = y.offset, color = NULL, fill = NULL, strand = strand))
                }
            } else if (shape == "rectangle") {
                if (!is.null(fill.by)) {
                    plt <- plt + ggnewscale::new_scale_fill() + ggnewscale::new_scale_color() +
                        geom_roundrect(data = annot.df, ggplot2::aes(xmin = start, xmax = end, y = y.offset, color = .data[[fill.by]], fill = .data[[fill.by]]), radius = grid::unit(0, "mm"))
                } else {
                    plt <- plt + ggnewscale::new_scale_fill() + ggnewscale::new_scale_color() +
                        geom_roundrect(data = annot.df, ggplot2::aes(xmin = start, xmax = end, y = y.offset, color = NULL, fill = NULL), radius = grid::unit(0, "mm"))
                }
            }

            ## Add annotation label above each annotation range ##
            if (!is.null(label.by)) {
                if (label.by %in% colnames(annot.df)) {
                    if (!is.null(annotation.group)) {
                        label.df <- annot.df %>%
                            dplyr::group_by(dplyr::across(dplyr::all_of(annotation.group))) %>%
                            dplyr::summarise(
                                seqnames = unique(seqnames),
                                start = min(start),
                                end = max(end),
                                y.offset = unique(y.offset),
                                label = unique(.data[[label.by]])
                            ) %>%
                            dplyr::mutate(midpoint = start + ((end - start) / 2))
                    } else {
                        label.df <- annot.df %>%
                            dplyr::mutate(midpoint = start + ((end - start) / 2), label = .data[[label.by]])
                    }
                    ## Add label to the plot
                    plt <- plt +
                        ggplot2::geom_text(data = label.df, ggplot2::aes(x = .data[["midpoint"]], y = y.offset + 0.025, label = .data[["label"]]))
                }
            }

            ## Add y-label to annotation track if defined
            if (!is.null(annotation.label)) {
                if (nchar(annotation.label) > 0) {
                    annot.yrange <- range(annot.df$y.offset)
                    annot.break <- annot.yrange[1] + (diff(annot.yrange) / 2)
                    if (coordinate.space == "target") {
                        y.breaks <- c(ybreaks, annot.break)
                        y.labels <- c(ylabels, annotation.label)
                    } else {
                        y.breaks <- c(annot.break, ybreaks)
                        y.labels <- c(annotation.label, ylabels)
                    }
                    suppressMessages(
                        plt <- plt + ggplot2::scale_y_continuous(breaks = y.breaks, labels = y.labels) +
                            ggplot2::theme(
                                axis.text.y = ggplot2::element_text(),
                                axis.ticks.y = ggplot2::element_line()
                            )
                    )
                }
            }

            ## Expand x-axis [TODO not sure if needed, if set secondary axis disapears!!!]
            # suppressMessages(
            #   plt <- plt + ggplot2::(add = 0)
            # )

            if (col.scale == "gradient") {
                plt <- plt + ggplot2::scale_fill_gradientn(colours = pal) + ggplot2::scale_color_gradientn(colours = pal)
            } else if (col.scale == "discrete") {
                plt <- plt + ggplot2::scale_fill_manual(values = pal) + ggplot2::scale_color_manual(values = pal)
            } else if (col.scale == "discreteNoLegend") {
                plt <- plt + ggplot2::scale_fill_manual(values = pal, guide = "none") + ggplot2::scale_color_manual(values = pal, guide = "none")
            }
            return(plt)
        } else {
            warning("None of the ranges reported in 'annot.gr' falls into the x-axis limits !!!")
            return(ggplot.obj)
        }
    } else {
        return(ggplot.obj)
        warning("Make sure that 'annot.gr' is 'GRanges' class object !!!")
    }
}


#' Flip query annotation ranges
#'
#' This function takes loaded PAF alignments using \code{\link{readPaf}} function and postprocessed using
#' \code{\link{flipPaf}} function. In case the PAF alignments were flipped ranges defined in 'query.annot.gr'
#' will be flipped accordingly to match query coordinates defined in 'paf.table'.
#'
#' @param query.annot.gr A \code{\link{GRanges-class}} object with a set of ranges in query coordinates. See function
#' \code{\link{liftRangesToAlignment}} if coordinates need to be lifted from target space.
#' @inheritParams breakPaf
#' @return A \code{\link{GRanges-class}} object.
#' @importFrom GenomicRanges strand makeGRangesFromDataFrame
#' @importFrom IRanges reflect ranges IRanges
#' @importFrom GenomeInfoDb seqnames
#' @importFrom dplyr recode
#' @author David Porubsky
#' @export
#' @examples
#' ## Get PAF to process
#' paf.file <- system.file("extdata", "test1.paf", package = "SVbyEye")
#' ## Read in PAF
#' paf.table <- readPaf(paf.file = paf.file, include.paf.tags = TRUE, restrict.paf.tags = "cg")
#' ## Flip PAF alignments
#' paf.table <- flipPaf(paf.table = paf.table, force = TRUE)
#' ## Load query annotation file
#' query.annot <- system.file("extdata", "test1_query_annot.txt", package = "SVbyEye")
#' query.annot.df <- read.table(query.annot, header = TRUE, sep = "\t", stringsAsFactors = FALSE)
#' query.annot.gr <- GenomicRanges::makeGRangesFromDataFrame(query.annot.df)
#' ## Synchronize orientation of query annotation file with flipped PAF alignments
#' flipQueryAnnotation(paf.table = paf.table, query.annot.gr = query.annot.gr)
#'
flipQueryAnnotation <- function(paf.table, query.annot.gr = NULL) {
    ## Check user input ##
    ## Check if submitted query annotation is GRanges object
    if (!is.null(query.annot.gr)) {
        if (!is(query.annot.gr, "GRanges")) {
            stop("Parameter 'query.annot.gr' has to be of 'GenomicRanges' object !!!")
        }
    }
    ## Make sure submitted paf.table has at least 12 mandatory fields
    if (ncol(paf.table) >= 12) {
        paf <- paf.table
        paf$direction.flip <- FALSE
    } else {
        stop("Submitted PAF alignments do not contain a minimum of 12 mandatory fields, see PAF file format definition !!!")
    }

    ## Process per query name
    paf.table.l <- split(paf.table, paf.table$q.name)
    query.annot.grl <- split(query.annot.gr, GenomeInfoDb::seqnames(query.annot.gr))
    ## Keep only non-empty list elements
    query.annot.grl <- query.annot.grl[lengths(query.annot.grl) > 0]

    flipped.annot <- list()
    for (q.name in names(query.annot.grl)) {
        annot.gr <- query.annot.grl[[q.name]]
        paf.subtable <- paf.table.l[[q.name]]
        if (nrow(paf.subtable) > 0) {
            ## Flip query annotation if query.flip is TRUE
            if (unique(paf.subtable$query.flip)) {
                # annot.gr.flipped <- mirrorRanges(gr = annot.gr, seqlength = unique(paf.subtable$q.len))
                annot.gr.flipped <- annot.gr
                suppressWarnings(
                    IRanges::ranges(annot.gr.flipped) <- IRanges::reflect(x = IRanges::ranges(annot.gr), bounds = IRanges::IRanges(start = 1L, end = unique(paf.subtable$q.len)))
                )
                ## Flip ranges strand orientation
                GenomicRanges::strand(annot.gr.flipped) <- dplyr::recode(as.character(GenomicRanges::strand(annot.gr.flipped)), "+" = "-", "-" = "+")
                flipped.annot[[length(flipped.annot) + 1]] <- annot.gr.flipped
            } else {
                flipped.annot[[length(flipped.annot) + 1]] <- annot.gr
            }
        }
    }
    flipped.annot.gr <- suppressWarnings(do.call(c, flipped.annot))
    return(flipped.annot.gr)
}
daewoooo/SVbyEye documentation built on Oct. 15, 2024, 6:12 a.m.