
package jsat.utils;

import java.io.Serializable;
import java.util.*;
import java.util.stream.DoubleStream;
import jsat.linear.DenseVector;
import jsat.linear.Vec;

/**
 * Provides a modifiable implementation of a List using a double array. This provides considerable
 * memory efficency improvements over using an {@link ArrayList} to store doubles. <br>
 * Null is not allowed into the list. 
 * 
 * @author Edward Raff
 */
public class DoubleList extends AbstractList<Double> implements Serializable, RandomAccess
{

    private static final long serialVersionUID = 653930294509274337L;
    private double[] array;
    //Exclusive
    private int end;

    private DoubleList(double[] array, int end)
    {
        this.array = array;
        this.end = end;
    }

    @Override
    public void clear()
    {
        end = 0;
    }

    /**
     * Creates a new empty DoubleList
     */
    public DoubleList()
    {
        this(10);
    }
    
    /**
     * Creates a new empty DoubleList
     * 
     * @param capacity the starting internal capacity of the list
     */
    public DoubleList(int capacity)
    {
        this(new double[capacity], 0);
    }
    
    /**
     * Creates a new DoubleList containing the values of the given collection
     * @param c the collection of values to fill this double list with
     */
    public DoubleList(Collection<Double> c)
    {
        this(c.size());
        this.addAll(c);
    }

    @Override
    public int size()
    {
        return end;
    }

    /**
     * Performs exactly the same as {@link #add(java.lang.Double) }. 
     * @param e the value to add
     * @return true if it was added, false otherwise
     */
    public boolean add(double e)
    {
        enlageIfNeeded(1);
        array[end] = e;
        increasedSize(1);
        return true;
    }
    
    /**
     * This method treats the underlying list as a stack. 
     * Pushes an item onto the top of this "stack". 
     * @param e the item to push onto the stack
     * @return the value added to the stack
     */
    public double push(double e)
    {
        add(e);
        return e;
    }
    
    /**
     * This method treats the underlying list as a stack. Removes the item at
     * the top of this "stack" and returns that item as the value.
     *
     * @return the item at the top of this stack (the last item pushed onto it)
     */
    public double pop()
    {
        if(isEmpty())
            throw new EmptyStackException();
        return removeD(size()-1);
    }

    /**
     * This method treats the underlying list as a stack. Gets the item at the
     * top of this "stack" and returns that item as the value, but leaves it on
     * the stack.
     *
     * @return the item at the top of this stack (the last item pushed onto it)
     */
    public double peek()
    {
        if(isEmpty())
            throw new EmptyStackException();
        return get(size()-1);
    }

    /**
     * Makes the changes indicating that a number of items have been removed
     * @param removed the number of items that were removed 
     */
    private void decreaseSize(int removed)
    {
        end-=removed;
    }

    /**
     * Marks the increase of size of this list, and reflects the change in the parent 
     */
    private void increasedSize(int added)
    {
        end+=added;
    }

    private void boundsCheck(int index) throws IndexOutOfBoundsException
    {
        if(index >= size())
            throw new IndexOutOfBoundsException("List is of size " + size() + ", index requested " + index);
    }

    /**
     * Enlarge the storage array if needed
     * @param i the amount of elements we will need to add
     */
    private void enlageIfNeeded(int i)
    {
        while(end+i > array.length)
            array = Arrays.copyOf(array, Math.max(array.length*2, 8));
    }
    
    @Override
    public boolean add(Double e)
    {
        if(e == null)
            return false;
        return add(e.doubleValue());
    }

    /**
     * Operates exactly as {@link #get(int) }
     * @param index the index of the value to get
     * @return the value at the given index
     */
    public double getD(int index)
    {
        boundsCheck(index);
        return array[index];
    }
    
    @Override
    public Double get(int index)
    {
        return getD(index);
    }
    
    /**
     * Operates exactly as {@link #set(int, java.lang.Double) }
     * @param index the index to set
     * @param element the value to set
     * @return the previous value at said index
     */
    public double set(int index, double element)
    {
        boundsCheck(index);
        double ret = get(index);
        array[index] = element;
        return ret;
    }

    @Override
    public Double set(int index, Double element)
    {
        return set(index, element.doubleValue());
    }

    /**
     * Operates exactly as {@link #add(int, java.lang.Double) }
     * @param index the index to add at
     * @param element the value to add
     */
    public void add(int index, double element)
    {
        if(index == size())//special case, just appending
        {
            add(element);
        }
        else
        {
            boundsCheck(index);
            enlageIfNeeded(1);
            System.arraycopy(array, index, array, index+1, size()-index);
            set(index, element);
            increasedSize(1);
        }
    }
    
    @Override
    public void add(int index, Double element)
    {
        add(index, element.doubleValue());
    }

    /**
     * Operates exactly as {@link #remove(int) }
     * @param index the index to remove
     * @return the value removed
     */
    public double removeD(int index)
    {
        boundsCheck(index);
        double ret = array[index];
        for(int i = index; i < end-1; i++)
            array[i] = array[i+1];
        decreaseSize(1);
        return ret;
    }
 
    @Override
    public Double remove(int index)
    {
        return removeD(index);
    }
    
    /**
     * Returns the reference to the array that backs this list. 
     * Alterations to the array will be visible to the DoubelList
     * and vise versa. The array returned may not the the same
     * size as the value returned by {@link #size() }
     * @return the underlying array used by this DoubleList
     */
    public double[] getBackingArray()
    {
        return array;
    }
    
    /**
     * Obtains a view of this double list as a dense vector with equal length. 
     * This is a soft reference, and altering the values in the matrix with 
     * alter the double list, and vise versa. 
     * <br><br>
     * While no error will be thrown if the size of the underlying list changes,
     * this view should be discarded if the size of the list changes. Once the 
     * list has changed sizes, there is no guarantee on the behavior that will
     * occur if the vector is used. 
     * 
     * @return a vector view of this list
     */
    public Vec getVecView()
    {
        return new DenseVector(array, 0, end);
    }
    
    /**
     * Creates an returns an unmodifiable view of the given double array that requires 
     * only a small object allocation. 
     * 
     * @param array the array to wrap into an unmodifiable list
     * @param length the number of values of the array to use, starting from zero
     * @return an unmodifiable list view of the array
     */
    public static List<Double> unmodifiableView(double[] array, int length)
    {
        return Collections.unmodifiableList(view(array, length));
    }
    
    /**
     * Creates and returns a view of the given double array that requires only
     * a small object allocation. Changes to the list will be reflected in the 
     * array up to a point. If the modification would require increasing the 
     * capacity of the array, a new array will be allocated - at which point 
     * operations will no longer be reflected in the original array. 
     * 
     * @param array the array to wrap by a DoubleList object
     * @param length the initial length of the list
     * @return a DoubleList backed by the given array, unless modified to the 
     * point of requiring the allocation of a new array
     */
    public static DoubleList view(double[] array, int length)
    {
        if(length > array.length || length < 0)
            throw new IllegalArgumentException("length must be non-negative and no more than the size of the array("+array.length+"), not " + length);
        return new DoubleList(array, length);
    }
    
    /**
     * 
     * @return the maximum value stored in this list. 
     */
    public double max()
    {
        if(isEmpty())
            throw new EmptyStackException();
        double max = 0;
        for(int i = 0; i < end; i++)
            max = Math.max(max, array[i]);
        return max;
    }
}
