All Downloads are FREE. Search and download functionalities are using the official Maven repository.

htsjdk.samtools.util.Interval Maven / Gradle / Ivy

There is a newer version: 4.3.0
Show newest version
/*
 * The MIT License
 *
 * Copyright (c) 2010 The Broad Institute
 *
 * Permission is hereby granted, free of charge, to any person obtaining a copy
 * of this software and associated documentation files (the "Software"), to deal
 * in the Software without restriction, including without limitation the rights
 * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
 * copies of the Software, and to permit persons to whom the Software is
 * furnished to do so, subject to the following conditions:
 *
 * The above copyright notice and this permission notice shall be included in
 * all copies or substantial portions of the Software.
 *
 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
 * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
 * THE SOFTWARE.
 */
package htsjdk.samtools.util;

import htsjdk.samtools.SAMException;
import htsjdk.tribble.NamedFeature;
import htsjdk.tribble.annotation.Strand;

import java.util.Collection;

/**
 * Represents a simple interval on a sequence.  Coordinates are 1-based closed ended.
 *
 * @author Tim Fennell
 */
public class Interval implements Comparable, Cloneable, NamedFeature {
    private final boolean negativeStrand;
    private final String name;
    private final String contig;
    private final int start;
    private final int end;

    /**
     * Constructs an interval with the supplied sequence and start and end. If the end
     * position is less than the start position an exception is thrown.
     *
     * @param sequence the name of the sequence
     * @param start    the start position of the interval on the sequence
     * @param end      the end position of the interval on the sequence
     */
    public Interval(final String sequence, final int start, final int end) {
        this(sequence, start, end, false, null);
    }

    /**
     * Constructs an interval with the supplied {@link Locatable}.
     */
    public Interval(final Locatable locatable) {
        this(locatable.getContig(), locatable.getStart(), locatable.getEnd());
    }

    /**
     * Constructs an interval with the supplied sequence and start, end, strand and name.
     * If the end position is less than the start position an exception is thrown.
     *
     * @param sequence the name of the sequence
     * @param start    the start position of the interval on the sequence
     * @param end      the end position of the interval on the sequence
     * @param negative true to indicate negative strand, false otherwise
     * @param name     the name (possibly null) of the interval
     */
    public Interval(final String sequence, final int start, final int end, final boolean negative, final String name) {
        this.contig = sequence;
        this.start = start;
        this.end = end;
        this.negativeStrand = negative;
        this.name = name;
    }

    /**
     * Gets the name of the sequence on which the interval resides.
     * This is a simple alias of getContig()
     *
     * @deprecated use getContig() instead
     */
    @Deprecated
    public String getSequence() {
        return getContig();
    }

    /**
     * Returns true if the interval is on the negative strand, otherwise false.
     */
    public boolean isNegativeStrand() {
        return this.negativeStrand;
    }

    /**
     * Returns true if the interval is on the positive strand, otherwise false.
     */
    public boolean isPositiveStrand() {
        return !this.negativeStrand;
    }

    /**
     * Return the {@link Strand} this interval is on.
     */
    public Strand getStrand(){
        return isNegativeStrand() ? Strand.NEGATIVE : Strand.FORWARD;
    }

    /**
     * Returns the name of the interval, possibly null.
     */
    public String getName() {
        return this.name;
    }

    /**
     * Returns true if this interval overlaps the other interval, otherwise false.
     */
    public boolean intersects(final Interval other) {
        return (this.getContig().equals(other.getContig()) &&
                CoordMath.overlaps(this.getStart(), this.getEnd(), other.getStart(), other.getEnd()));
    }

    public int getIntersectionLength(final Interval other) {
        if (this.intersects(other)) {
            return (int) CoordMath.getOverlap(this.getStart(), this.getEnd(), other.getStart(), other.getEnd());
        }
        return 0;
    }

    /**
     * Returns a new Interval that represents the intersection between the two intervals.
     */
    public Interval intersect(final Interval that) {
        if (!intersects(that)) throw new IllegalArgumentException(that + " does not intersect " + this);
        return new Interval(this.getContig(),
                Math.max(this.getStart(), that.getStart()),
                Math.min(this.getEnd(), that.getEnd()),
                this.negativeStrand,
                this.name + " intersection " + that.name);
    }

    /**
     * Returns true if this interval overlaps the other interval, otherwise false.
     */
    public boolean abuts(final Interval other) {
        return this.getContig().equals(other.getContig()) &&
                (this.getStart() == other.getEnd() + 1 || other.getStart() == this.getEnd() + 1);
    }

    /**
     * Gets the length of this interval.
     */
    public int length() {
        return this.getEnd() - this.getStart() + 1;
    }

    /**
     * Returns a new interval that is padded by the amount of bases specified on either side.
     */
    public Interval pad(final int left, final int right) {
        return new Interval(this.getContig(), this.getStart() - left, this.getEnd() + right, this.negativeStrand, this.name);
    }

    /**
     * Counts the total number of bases a collection of intervals.
     */
    public static long countBases(final Collection intervals) {
        return intervals.stream()
                .mapToLong(Interval::length)
                .sum();
    }

    /**
     * Sort based on sequence.compareTo, then start pos, then end pos
     * with null objects coming lexically last
     */
    @Override
    public int compareTo(final Interval that) {
        if (that == null) return -1; // nulls last

        int result = this.getContig().compareTo(that.getContig());
        if (result == 0) {
            if (this.getStart() == that.getStart()) {
                result = this.getEnd() - that.getEnd();
            } else {
                result = this.getStart() - that.getStart();
            }
        }

        return result;
    }

    /**
     * Equals method that agrees with {@link #compareTo(Interval)}.
     */
    public boolean equals(final Object other) {
        if (!(other instanceof Interval)) return false;
        else if (this == other) return true;
        else {
            Interval that = (Interval) other;
            return (this.compareTo(that) == 0);
        }
    }

    /**
     * Equals method that also checks strand and name
     */
    public boolean equalsWithStrandAndName(final Object other) {
        if (!this.equals(other)) {
            return false;
        }

        final Interval that = (Interval) other;
        if (this.negativeStrand != that.negativeStrand) {
            return false;
        }

        return this.name.equals(that.name);
    }

    @Override
    public int hashCode() {
        int result = getContig().hashCode();
        result = 31 * result + getStart();
        result = 31 * result + getEnd();
        return result;
    }

    public String toString() {
        return getContig() + ":" + getStart() + "-" + getEnd() + "\t" + getStrand().encode() + "\t" + ((null == name) ? '.' : name);
    }

    @Override
    public Interval clone() {
        try {
            return (Interval) super.clone();
        } catch (CloneNotSupportedException cnse) {
            throw new SAMException("That's unpossible", cnse);
        }
    }

    @Override
    public String getContig() {
        return contig;
    }

    @Override
    public int getStart() {
        return start;
    }

    @Override
    public int getEnd() {
        return end;
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy