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

org.snpeff.interval.IntervalAndSubIntervals Maven / Gradle / Ivy

The newest version!
package org.snpeff.interval;

import java.util.ArrayList;
import java.util.Collection;
import java.util.Collections;
import java.util.HashMap;
import java.util.Iterator;
import java.util.List;
import java.util.Map;

import org.snpeff.serializer.MarkerSerializer;

/**
 * Interval that contains sub intervals.
 *
 * @author pcingola
 *
 */
public class IntervalAndSubIntervals extends Marker implements Iterable {

	private static final long serialVersionUID = 1636197649250882952L;
	Map subIntervals;
	List sorted;
	List sortedStrand;

	public IntervalAndSubIntervals() {
		super();
		reset();
	}

	public IntervalAndSubIntervals(Marker parent, int start, int end, boolean strandMinus, String id) {
		super(parent, start, end, strandMinus, id);
		reset();
	}

	/**
	 * Add a subinterval
	 */
	public synchronized void add(T t) {
		if (subIntervals.put(t.getId(), t) != null) {
			// Keys should be unique
			throw new RuntimeException(t.getClass().getSimpleName() //
					+ " '" + t.getId() + "' is already in " //
					+ this.getClass().getSimpleName() //
					+ " '" + id + "'" //
			);
		}

		// Sort is no longer valid
		invalidateSorted();
	}

	/**
	 * Add all intervals
	 */
	public void addAll(Iterable ts) {
		for (T t : ts)
			add(t);
		invalidateSorted();
	}

	/**
	 * Add all markers
	 */
	@SuppressWarnings("unchecked")
	public void addAll(Markers markers) {
		for (Marker m : markers)
			add((T) m);
		invalidateSorted();
	}

	/**
	 * Apply a variant.
	 */
	@Override
	@SuppressWarnings("unchecked")
	public IntervalAndSubIntervals apply(Variant variant) {
		if (!shouldApply(variant)) return this;

		// Apply to Marker
		IntervalAndSubIntervals newMarker = (IntervalAndSubIntervals) super.apply(variant);
		if (newMarker == null) return null;

		// Now apply to all sub-markers
		newMarker.reset();
		for (T m : this) {
			// Whole marker duplication
			if (variant.isDup() && variant.includes(m)) {
				// We need to add another copy of the marker: One without any modification and one with coordinates updated
				T mcopy = (T) m.cloneShallow();
				mcopy.setId(mcopy.getId() + ".dup"); // Change ID
				mcopy.setParent(newMarker);
				newMarker.add(mcopy);
			}

			// Apply variant to marker
			T mcopy = (T) m.apply(variant);

			// Do not add if interval is completely removed
			if (mcopy != null) {
				// Make sure we don't modify the original subinterval.
				if (mcopy == m) mcopy = (T) m.cloneShallow();

				mcopy.setParent(newMarker);
				newMarker.add(mcopy);
			}
		}

		return newMarker;
	}

	@SuppressWarnings("unchecked")
	@Override
	public IntervalAndSubIntervals clone() {
		IntervalAndSubIntervals copy = (IntervalAndSubIntervals) super.clone();
		copy.reset();

		for (T m : this) {
			T mcopy = (T) m.clone();
			mcopy.setParent(copy);
			copy.add(mcopy);
		}

		return copy;
	}

	@SuppressWarnings("unchecked")
	@Override
	public IntervalAndSubIntervals cloneShallow() {
		IntervalAndSubIntervals clone = (IntervalAndSubIntervals) super.cloneShallow();
		clone.reset();
		return clone;
	}

	/**
	 * Is 'id' in the subintervals?
	 */
	public boolean containsId(String id) {
		return subIntervals.containsKey(id);
	}

	/**
	 * Obtain a subinterval
	 */
	public T get(String id) {
		return subIntervals.get(id);
	}

	/**
	 * Invalidate sorted collections
	 */
	protected void invalidateSorted() {
		sorted = sortedStrand = null;
	}

	@Override
	public Iterator iterator() {
		return subIntervals().iterator();
	}

	/**
	 * A list of all markers in this transcript
	 */
	public Markers markers() {
		Markers markers = new Markers();
		markers.addAll(subIntervals());
		return markers;
	}

	public int numChilds() {
		return (subIntervals != null ? subIntervals.size() : 0);
	}

	/**
	 * Query all genomic regions that intersect 'marker'
	 */
	@Override
	public Markers query(Marker marker) {
		Markers markers = new Markers();

		for (Marker m : this) {
			if (m.intersects(marker)) {
				markers.add(m);

				Markers subMarkers = m.query(marker);
				if (subMarkers != null) markers.add(subMarkers);
			}
		}

		return markers;
	}

	/**
	 * Remove a subinterval
	 */
	public synchronized void remove(T t) {
		subIntervals.remove(t.getId());
		invalidateSorted();
	}

	/**
	 * Remove all intervals
	 */
	public synchronized void reset() {
		subIntervals = new HashMap<>();
		invalidateSorted();
	}

	/**
	 * Parse a line from a serialized file
	 */
	@SuppressWarnings("unchecked")
	@Override
	public void serializeParse(MarkerSerializer markerSerializer) {
		super.serializeParse(markerSerializer);

		Markers markers = markerSerializer.getNextFieldMarkers();
		for (Marker m : markers)
			add((T) m);
	}

	/**
	 * Create a string to serialize to a file
	 * @return
	 */
	@SuppressWarnings("unchecked")
	@Override
	public String serializeSave(MarkerSerializer markerSerializer) {
		return super.serializeSave(markerSerializer) //
				+ "\t" + markerSerializer.save((Collection) subIntervals()) //
		;
	}

	@Override
	public void setStrandMinus(boolean strandMinus) {
		this.strandMinus = strandMinus;

		// Change all subintervals
		for (T t : this)
			t.setStrandMinus(strandMinus);

		invalidateSorted(); // These are no longer correct
	}

	@Override
	public void shiftCoordinates(int shift) {
		super.shiftCoordinates(shift);

		for (T m : subIntervals())
			m.shiftCoordinates(shift);
	}

	/**
	 * Return a collection of sub intervals sorted by natural order
	 */
	public synchronized List sorted() {
		if (sorted != null) return sorted;
		ArrayList sorted = new ArrayList<>();
		sorted.addAll(subIntervals());
		Collections.sort(sorted);

		this.sorted = sorted;
		return sorted;
	}

	/**
	 * Return a collection of sub intervals sorted by start position (if strand is >= 0) or
	 * by reverse end position (if strans < 0)
	 */
	public synchronized List sortedStrand() {
		if (sortedStrand != null) return sortedStrand;

		ArrayList sortedStrand = new ArrayList<>();
		sortedStrand.addAll(subIntervals());

		if (isStrandPlus()) Collections.sort(sortedStrand, new IntervalComparatorByStart()); // Sort by start position
		else Collections.sort(sortedStrand, new IntervalComparatorByEnd(true)); // Sort by end position (reversed)

		this.sortedStrand = sortedStrand;
		return sortedStrand;
	}

	/**
	 * Return a collection of sub intervals
	 */
	public Collection subIntervals() {
		return subIntervals.values();
	}

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy