package com.planet_ink.coffee_mud.core.collections;
import java.io.Serializable;
import java.util.*;
import com.planet_ink.coffee_mud.core.interfaces.CMObject;
/*
Copyright 2014-2016 Bo Zimmerman
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
*/
/*
* A version of the Vector class that provides to "safe" adds
* and removes by copying the underlying vector whenever those
* operations are done. Also maintains a tree, using the CMObject
* ID() of the stored object as sort key.
*/
public class CMSTreeVector<T extends CMObject> implements Serializable, Iterable<T>, Collection<T>, List<T>, RandomAccess
{
private static final long serialVersionUID = 6687178785122561992L;
private volatile Vector<T> V;
private final TreeMap<String,T> S;
public CMSTreeVector()
{
V=new Vector<T>();
S=new TreeMap<String,T>();
}
public CMSTreeVector(int size)
{
V=new Vector<T>(size);
S=new TreeMap<String,T>();
}
public CMSTreeVector(List<T> E)
{
V=new Vector<T>();
S=new TreeMap<String,T>();
if(E!=null)
addAll(E);
}
public CMSTreeVector(T[] E)
{
V=new Vector<T>();
S=new TreeMap<String,T>();
if(E!=null)
{
for(final T o : E)
addBoth(o);
}
}
public CMSTreeVector(Enumeration<T> E)
{
V=new Vector<T>();
S=new TreeMap<String,T>();
if(E!=null)
{
for(;E.hasMoreElements();)
addBoth(E.nextElement());
}
}
public CMSTreeVector(Iterator<T> E)
{
V=new Vector<T>();
S=new TreeMap<String,T>();
for(;E.hasNext();)
addBoth(E.next());
}
public CMSTreeVector(Set<T> E)
{
V=new Vector<T>();
S=new TreeMap<String,T>();
for(final T o : E)
addBoth(o);
}
@SuppressWarnings("unchecked")
public synchronized void addAll(Enumeration<T> E)
{
if(E!=null)
{
V=(Vector<T>)V.clone();
for(;E.hasMoreElements();)
addBoth(E.nextElement());
}
}
@SuppressWarnings("unchecked")
public synchronized void addAll(T[] E)
{
if(E!=null)
{
V=(Vector<T>)V.clone();
for(final T e : E)
addBoth(e);
}
}
@SuppressWarnings("unchecked")
public synchronized void addAll(Iterator<T> E)
{
if(E!=null)
{
V=(Vector<T>)V.clone();
for(;E.hasNext();)
addBoth(E.next());
}
}
@SuppressWarnings("unchecked")
public synchronized void removeAll(Enumeration<T> E)
{
if(E!=null)
{
V=(Vector<T>)V.clone();
for(;E.hasMoreElements();)
removeBoth(E.nextElement());
}
}
@SuppressWarnings("unchecked")
public synchronized void removeAll(Iterator<T> E)
{
if(E!=null)
{
V=(Vector<T>)V.clone();
for(;E.hasNext();)
removeBoth(E.next());
}
}
@SuppressWarnings("unchecked")
public synchronized void removeAll(List<T> E)
{
if(E!=null)
{
V=(Vector<T>)V.clone();
for(final T o : E)
removeBoth(o);
}
}
public synchronized int capacity()
{
return V.capacity();
}
@SuppressWarnings("unchecked")
public synchronized Vector<T> toVector()
{
return (Vector<T>)V.clone();
}
@SuppressWarnings("unchecked")
public synchronized CMSTreeVector<T> copyOf()
{
final CMSTreeVector<T> SV=new CMSTreeVector<T>();
SV.V=(Vector<T>)V.clone();
SV.S.putAll(S);
return SV;
}
@Override
public synchronized boolean contains(Object o)
{
if(o instanceof CMObject)
return S.containsKey(((CMObject)o).ID().toUpperCase());
return V.contains(o);
}
@Override
public synchronized boolean containsAll(Collection<?> c)
{
return V.containsAll(c);
}
@SuppressWarnings("unchecked")
public synchronized void copyInto(Object[] anArray)
{
V=(Vector<T>)V.clone();
V.copyInto(anArray);
}
public synchronized T elementAt(int index)
{
return V.elementAt(index);
}
public synchronized Enumeration<T> elements()
{
return V.elements();
}
public synchronized void ensureCapacity(int minCapacity)
{
V.ensureCapacity(minCapacity);
}
@Override
public synchronized boolean equals(Object o)
{
return o==this;
}
public synchronized T firstElement()
{
return V.firstElement();
}
@Override
public synchronized T get(int index)
{
return V.get(index);
}
@Override
public synchronized int hashCode()
{
return super.hashCode();
}
public synchronized int indexOf(Object o, int index)
{
return V.indexOf(o, index);
}
@Override
public synchronized int indexOf(Object o)
{
return V.indexOf(o);
}
@Override
public synchronized boolean isEmpty()
{
return V.isEmpty();
}
public synchronized T lastElement()
{
return V.lastElement();
}
public synchronized int lastIndexOf(Object o, int index)
{
return V.lastIndexOf(o, index);
}
@Override
public synchronized int lastIndexOf(Object o)
{
return V.lastIndexOf(o);
}
@Override
public synchronized boolean retainAll(Collection<?> c)
{
final int oldSize=size();
for(final T o : V)
if(!c.contains(o))
remove(o);
return oldSize < size();
}
@SuppressWarnings("unchecked")
@Override
public synchronized T set(int index, T element)
{
if(element==null)
return null;
if(!S.containsKey(element.ID().toUpperCase()))
{
V=(Vector<T>)V.clone();
final T oldT = V.set(index,element);
if(oldT!=null)
S.remove(oldT.ID().toUpperCase());
S.put(element.ID().toUpperCase(), element);
return oldT;
}
return null;
}
public synchronized void setElementAt(T obj, int index)
{
set(index,obj);
}
@SuppressWarnings("unchecked")
public synchronized void setSize(int newSize)
{
V=(Vector<T>)V.clone();
V.setSize(newSize);
}
@Override
public synchronized int size()
{
return V.size();
}
@Override
public synchronized List<T> subList(int fromIndex, int toIndex)
{
return V.subList(fromIndex, toIndex);
}
@Override
public synchronized Object[] toArray()
{
return V.toArray();
}
@SuppressWarnings("hiding")
@Override
public synchronized <T> T[] toArray(T[] a)
{
return V.toArray(a);
}
@Override
public synchronized String toString()
{
return super.toString();
}
@SuppressWarnings("unchecked")
public synchronized void trimToSize()
{
V=(Vector<T>)V.clone();
V.trimToSize();
}
@SuppressWarnings("unchecked")
@Override
public synchronized void add(int index, T element)
{
if(element==null)
return;
if(!S.containsKey(element.ID().toUpperCase()))
{
V=(Vector<T>)V.clone();
V.add(index, element);
S.put(element.ID().toUpperCase(), element);
}
}
@SuppressWarnings("unchecked")
@Override
public synchronized boolean add(T e)
{
if(e==null)
return false;
if(!S.containsKey(e.ID().toUpperCase()))
{
V=(Vector<T>)V.clone();
if(V.add(e))
S.put(e.ID().toUpperCase(), e);
return true;
}
return false;
}
private boolean addBoth(T e)
{
if(S.containsKey(e.ID().toUpperCase()))
return false;
V.add(e);
S.put(e.ID().toUpperCase(), e);
return true;
}
@SuppressWarnings("unchecked")
@Override
public synchronized boolean addAll(Collection<? extends T> c)
{
V=(Vector<T>)V.clone();
boolean kaplah=false;
for(final Object o : c)
if(o instanceof CMObject)
kaplah = addBoth((T)o) || kaplah;
return kaplah;
}
@SuppressWarnings("unchecked")
@Override
public synchronized boolean addAll(int index, Collection<? extends T> c)
{
final int oldSize=size();
if(index>=size())
addAll(c);
else
{
for(final Object o : c)
if(o instanceof CMObject)
insertElementAt((T)o, index++);
}
return oldSize < size();
}
public synchronized void addElement(T obj)
{
add(obj);
}
@SuppressWarnings("unchecked")
@Override
public synchronized void clear()
{
V=(Vector<T>)V.clone();
V.clear();
S.clear();
}
public synchronized void insertElementAt(T obj, int index)
{
if(obj==null)
return;
if(index>=size())
add(obj);
else
if(!S.containsKey(obj.ID().toUpperCase()))
{
V.insertElementAt(obj, index);
S.put(obj.ID().toUpperCase(), obj);
}
}
@SuppressWarnings("unchecked")
@Override
public synchronized boolean remove(Object o)
{
if(!(o instanceof CMObject))
return false;
final CMObject O=(CMObject)o;
final String OID=O.ID().toUpperCase();
if(!S.containsKey(OID))
return false;
V=(Vector<T>)V.clone();
S.remove(OID);
return V.remove(o);
}
private boolean removeBoth(Object o)
{
if(!(o instanceof CMObject))
return false;
final CMObject O=(CMObject)o;
final String OID=O.ID().toUpperCase();
if(!S.containsKey(OID))
return false;
S.remove(OID);
return V.remove(o);
}
private boolean removeBoth(CMObject o)
{
final String OID=o.ID().toUpperCase();
if(!S.containsKey(OID))
return false;
S.remove(OID);
return V.remove(o);
}
@SuppressWarnings("unchecked")
@Override
public synchronized T remove(int index)
{
V=(Vector<T>)V.clone();
final T O=V.remove(index);
if(O==null)
return null;
final String OID=O.ID().toUpperCase();
if(S.containsKey(OID))
S.remove(OID);
return O;
}
@SuppressWarnings("unchecked")
@Override
public synchronized boolean removeAll(Collection<?> c)
{
V=(Vector<T>)V.clone();
boolean kaplah=false;
for(final Object o : c)
kaplah = removeBoth(o) || kaplah;
return kaplah;
}
@SuppressWarnings("unchecked")
public synchronized void removeAllElements()
{
V=(Vector<T>)V.clone();
V.removeAllElements();
S.clear();
}
public T find(final String key)
{
if(key==null)
return null;
return S.get(key.toUpperCase());
}
public synchronized boolean removeElement(Object obj)
{
return remove(obj);
}
@SuppressWarnings("unchecked")
public synchronized void removeElementAt(int index)
{
V=(Vector<T>)V.clone();
removeBoth(V.get(index));
}
@Override
public synchronized Iterator<T> iterator()
{
return new ReadOnlyIterator<T>(V.iterator());
}
@Override
public synchronized ListIterator<T> listIterator()
{
return new ReadOnlyListIterator<T>(V.listIterator());
}
@Override
public synchronized ListIterator<T> listIterator(int index)
{
return new ReadOnlyListIterator<T>(V.listIterator());
}
}