blob: ca6b2cdfd3afc3108d7879a616fe47f9178a1082 [file] [log] [blame]
/*
* Copyright (c) OSGi Alliance (2017). All Rights Reserved.
*
* 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.
*/
package org.osgi.util.converter;
import java.lang.reflect.Array;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collection;
import java.util.Collections;
import java.util.Iterator;
import java.util.List;
import java.util.ListIterator;
/**
* @author $Id$
*/
class ListDelegate<T> implements List<T> {
private volatile List<T> delegate;
private volatile boolean cloned;
private final ConvertingImpl convertingImpl;
private final InternalConverter converter;
@SuppressWarnings({
"unchecked", "rawtypes"
})
static <T> List<T> forArray(Object arr, ConvertingImpl converting, InternalConverter c) {
return new ListDelegate<T>(new ArrayDelegate(arr), converting, c);
}
static <T> List<T> forCollection(Collection<T> object,
ConvertingImpl converting, InternalConverter c) {
if (object instanceof List) {
return new ListDelegate<T>((List<T>) object, converting, c);
}
return new ListDelegate<T>(new CollectionDelegate<>(object),
converting, c);
}
private ListDelegate(List<T> del, ConvertingImpl conv, InternalConverter c) {
delegate = del;
convertingImpl = conv;
converter = c;
}
// Whenever a modification is made, the delegate is cloned and detached.
@SuppressWarnings("unchecked")
private void cloneDelegate() {
if (cloned) {
return;
} else {
cloned = true;
delegate = new ArrayList<T>((List<T>) Arrays.asList(toArray()));
}
}
@Override
public int size() {
return delegate.size();
}
@Override
public boolean isEmpty() {
return delegate.isEmpty();
}
@Override
public boolean contains(Object o) {
return containsAll(Collections.singletonList(o));
}
@Override
public Iterator<T> iterator() {
return listIterator();
}
@Override
public Object[] toArray() {
return toArray(new Object[size()]);
}
@SuppressWarnings("unchecked")
@Override
public <X> X[] toArray(X[] a) {
int mySize = size();
if (Array.getLength(a) < size()) {
a = (X[]) Array.newInstance(a.getClass().getComponentType(),
mySize);
}
for (int i = 0; i < a.length; i++) {
if (mySize > i) {
a[i] = (X) get(i);
} else {
a[i] = null;
}
}
return a;
}
@Override
public boolean add(T e) {
cloneDelegate();
return delegate.add(e);
}
@Override
public boolean remove(Object o) {
cloneDelegate();
return delegate.remove(o);
}
@Override
public boolean containsAll(Collection< ? > c) {
List<Object> l = Arrays.asList(toArray());
for (Object o : c) {
if (!l.contains(o))
return false;
}
return true;
}
@Override
public boolean addAll(Collection< ? extends T> c) {
cloneDelegate();
return delegate.addAll(c);
}
@Override
public boolean addAll(int index, Collection< ? extends T> c) {
cloneDelegate();
return delegate.addAll(index, c);
}
@Override
public boolean removeAll(Collection< ? > c) {
cloneDelegate();
return delegate.removeAll(c);
}
@Override
public boolean retainAll(Collection< ? > c) {
cloneDelegate();
return delegate.retainAll(c);
}
@Override
public void clear() {
cloned = true;
delegate = new ArrayList<>();
}
@SuppressWarnings("unchecked")
@Override
public T get(int index) {
return (T) convertingImpl.convertCollectionValue(delegate.get(index), converter);
}
@Override
public T set(int index, T element) {
cloneDelegate();
return delegate.set(index, element);
}
@Override
public void add(int index, T element) {
cloneDelegate();
delegate.add(index, element);
}
@Override
public T remove(int index) {
cloneDelegate();
return delegate.remove(index);
}
@Override
public int indexOf(Object o) {
return delegate.indexOf(o);
}
@Override
public int lastIndexOf(Object o) {
return delegate.lastIndexOf(o);
}
@SuppressWarnings("unchecked")
@Override
public ListIterator<T> listIterator() {
return (ListIterator<T>) Arrays.asList(toArray()).listIterator();
}
@SuppressWarnings("unchecked")
@Override
public ListIterator<T> listIterator(int index) {
return (ListIterator<T>) Arrays.asList(toArray()).listIterator(index);
}
@SuppressWarnings("unchecked")
@Override
public List<T> subList(int fromIndex, int toIndex) {
return (List<T>) Arrays.asList(toArray()).subList(fromIndex, toIndex);
}
@Override
public int hashCode() {
return delegate.hashCode();
}
@Override
public boolean equals(Object obj) {
if (obj == this)
return true;
if (!(obj instanceof List))
return false;
List< ? > l1 = new ArrayList<>(this);
List< ? > l2 = new ArrayList<>((List< ? >) obj);
return l1.equals(l2);
}
@Override
public String toString() {
return delegate.toString();
}
}