/*
 *                    BioJava development code
 *
 * This code may be freely distributed and modified under the
 * terms of the GNU Lesser General Public Licence.  This should
 * be distributed with the code.  If you do not have a copy,
 * see:
 *
 *      http://www.gnu.org/copyleft/lesser.html
 *
 * Copyright for this code is held jointly by the individual
 * authors.  These should be listed in @author doc comments.
 *
 * For more information on the BioJava project and its aims,
 * or to join the biojava-l mailing list, visit the home page
 * at:
 *
 *      http://www.biojava.org/
 *
 */

package org.biojava.bio.seq.impl;

import java.util.Iterator;

import org.biojava.bio.Annotation;
import org.biojava.bio.BioException;
import org.biojava.bio.seq.Feature;
import org.biojava.bio.seq.FeatureFilter;
import org.biojava.bio.seq.FeatureHolder;
import org.biojava.bio.seq.GappedSequence;
import org.biojava.bio.seq.MergeFeatureHolder;
import org.biojava.bio.seq.Sequence;
import org.biojava.bio.seq.SimpleFeatureHolder;
import org.biojava.bio.seq.projection.ProjectedFeatureHolder;
import org.biojava.bio.seq.projection.ReparentContext;
import org.biojava.bio.symbol.Location;
import org.biojava.bio.symbol.SimpleGappedSymbolList;
import org.biojava.utils.AssertionFailure;
import org.biojava.utils.ChangeVetoException;

/**
 * Simple implementation of GappedSequence. Please note that this is a view onto 
 * another Sequence. Gaps created and removed are only in the view not the underlying original. This
 * means that any gaps present in the original cannot be manipulated in this
 * view. To manipulate the original you would need to use Edit objects.
 *
 * @author Thomas Down
 * @author Matthew Pocock
 * @since 1.3
 */

public class SimpleGappedSequence
extends SimpleGappedSymbolList
implements GappedSequence {
  private Sequence sequence;

  private MergeFeatureHolder features;
  private SimpleFeatureHolder localFeatures;
  private FeatureHolder projectedFeatures;

  private boolean createOnUnderlying;

  public SimpleGappedSequence(Sequence seq) {
    super(seq);
    this.sequence = seq;
    createOnUnderlying = false;
  }

  public SimpleGappedSequence(GappedSequence seq) {
    super(seq);
    this.sequence = seq;
    createOnUnderlying = false;
  }

  public boolean getCreateOnUnderlyingSequence() {
    return createOnUnderlying;
  }

  public void setCreateOnUnderlyingSequence(boolean underlying) {
    this.createOnUnderlying = underlying;
  }

  public Annotation getAnnotation() {
    return sequence.getAnnotation();
  }

  public String getName() {
    return sequence.getName();
  }

  public String getURN() {
    return sequence.getURN();
  }

  private FeatureHolder getFeatures() {
    if (features == null) {
      features = makeFeatures();
    }
    return features;
  }

  private MergeFeatureHolder makeFeatures() {
    projectedFeatures = new ProjectedFeatureHolder(
            new GappedContext());

    localFeatures = new SimpleFeatureHolder();

    features = new MergeFeatureHolder();

    try {
      features.addFeatureHolder(projectedFeatures);
      features.addFeatureHolder(localFeatures);
    } catch (ChangeVetoException cve) {
      throw new AssertionFailure("Assertion Failure: Should be able to do this", cve);
    }

    return features;
  }

  public Iterator features() {
    return getFeatures().features();
  }

  public FeatureHolder filter(FeatureFilter ff) {
    return getFeatures().filter(ff);
  }

  public FeatureHolder filter(FeatureFilter ff, boolean recurse) {
    return getFeatures().filter(ff, recurse);
  }

  public int countFeatures() {
    return getFeatures().countFeatures();
  }

  public boolean containsFeature(Feature f) {
    return getFeatures().containsFeature(f);
  }

  public FeatureFilter getSchema() {
    return getFeatures().getSchema();
  }

  public void removeFeature(Feature f)
      throws ChangeVetoException, BioException
  {
      getFeatures();
      if (localFeatures.containsFeature(f)) {
          localFeatures.removeFeature(f);
      } else {
          projectedFeatures.removeFeature(f);
      }
  }

  public Feature createFeature(Feature.Template templ)
  throws ChangeVetoException, BioException
  {
    getFeatures();
    if(createOnUnderlying) {
      return projectedFeatures.createFeature(templ);
    } else {
      Feature f = FeatureImpl.DEFAULT.realizeFeature(this, this, templ);
      localFeatures.addFeature(f);
      return f;
    }
  }

  public class GappedContext extends ReparentContext {
    public GappedContext() {
      super(SimpleGappedSequence.this, sequence);
    }

    public Location projectLocation(Location loc) {
      return loc.newInstance(locationToGapped(loc));
    }

    public Location mapLocation(Location loc) {
      return loc.newInstance(locationToGapped(loc));
    }

    public Location revertLocation(Location oldLoc) {
      return gappedToLocation(oldLoc);
    }
  }
}
