blob: d8c3bdf0818fc5d58b9d05763bc0b5fe3dada1aa [file] [log] [blame]
/*=============================================================================#
# Copyright (c) 2009, 2021 Stephan Wahlbrink and others.
#
# This program and the accompanying materials are made available under the
# terms of the Eclipse Public License 2.0 which is available at
# https://www.eclipse.org/legal/epl-2.0, or the Apache License, Version 2.0
# which is available at https://www.apache.org/licenses/LICENSE-2.0.
#
# SPDX-License-Identifier: EPL-2.0 OR Apache-2.0
#
# Contributors:
# Stephan Wahlbrink <sw@wahlbrink.eu> - initial API and implementation
#=============================================================================*/
package org.eclipse.statet.internal.jcommons.collections;
import static org.eclipse.statet.jcommons.lang.NullDefaultLocation.FIELD;
import static org.eclipse.statet.jcommons.lang.NullDefaultLocation.PARAMETER;
import static org.eclipse.statet.jcommons.lang.NullDefaultLocation.RETURN_TYPE;
import static org.eclipse.statet.jcommons.lang.NullDefaultLocation.TYPE_ARGUMENT;
import static org.eclipse.statet.jcommons.lang.NullDefaultLocation.TYPE_BOUND;
import java.util.Arrays;
import java.util.Collection;
import java.util.Iterator;
import java.util.NoSuchElementException;
import java.util.RandomAccess;
import java.util.Set;
import java.util.Spliterator;
import java.util.Spliterators;
import org.eclipse.statet.jcommons.collections.ImIdentityList;
import org.eclipse.statet.jcommons.collections.ImList;
import org.eclipse.statet.jcommons.collections.ImSet;
import org.eclipse.statet.jcommons.lang.NonNullByDefault;
import org.eclipse.statet.jcommons.lang.Nullable;
/**
* Constant set implementation based on an array.
* <p>
* Comparable to <code>Collections.unmodifiableList(Array.asList(...))</code>.</p>
*
* @since de.walware.ecommons.coremisc 1.5
*/
@NonNullByDefault({ PARAMETER, RETURN_TYPE, FIELD, TYPE_BOUND, TYPE_ARGUMENT })
public final class ImArraySet<E> extends AbstractImList<E> implements ImSet<E>,
RandomAccess {
private class Iter extends AbstractImListIter<E> {
private int cursor;
Iter(final int index) {
this.cursor= index;
}
@Override
public boolean hasNext() {
return (this.cursor < ImArraySet.this.array.length);
}
@Override
public int nextIndex() {
return this.cursor;
}
@Override
public E next() {
if (this.cursor >= ImArraySet.this.array.length) {
throw new NoSuchElementException();
}
return ImArraySet.this.array[this.cursor++];
}
@Override
public boolean hasPrevious() {
return (this.cursor > 0);
}
@Override
public int previousIndex() {
return this.cursor - 1;
}
@Override
public E previous() {
if (this.cursor <= 0 || ImArraySet.this.array.length <= 0) {
throw new NoSuchElementException();
}
return ImArraySet.this.array[--this.cursor];
}
}
private final E[] array;
/**
* Create a new constant list backed by the given array (directly used!).
*
* If the list is published through an API and should be constant, the
* the elements of the given array must not any longer be changed.
*
* @param a the array by which the list will be backed.
*/
public ImArraySet(final E[] a) {
this.array= a;
}
@Override
public int size() {
return this.array.length;
}
@Override
public boolean isEmpty() {
return false;
}
@Override
public boolean contains(final @Nullable Object o) {
return (indexOf(o) >= 0);
}
@Override
public boolean containsAll(final Collection<?> c) {
final Iterator<?> e= c.iterator();
while (e.hasNext()) {
if (indexOf(e.next()) < 0) {
return false;
}
}
return true;
}
@Override
public E get(final int index) {
return this.array[index];
}
@Override
public int indexOf(final @Nullable Object o) {
if (o == null) {
for (int i= 0; i < this.array.length; i++) {
if (null == this.array[i]) {
return i;
}
}
return -1;
}
else {
for (int i= 0; i < this.array.length; i++) {
if (o.equals(this.array[i])) {
return i;
}
}
return -1;
}
}
@Override
public int lastIndexOf(final @Nullable Object o) {
if (o == null) {
for (int i= this.array.length - 1; i >= 0; i--) {
if (null == this.array[i]) {
return i;
}
}
return -1;
}
else {
for (int i= this.array.length - 1; i >= 0; i--) {
if (o.equals(this.array[i])) {
return i;
}
}
return -1;
}
}
@Override
public Iterator<E> iterator() {
return new Iter(0);
}
@Override
public Spliterator<E> spliterator() {
return Spliterators.spliterator(this.array, 0, this.array.length,
Spliterator.IMMUTABLE | Spliterator.DISTINCT );
}
@Override
public Object[] toArray() {
final Object[] dest= new Object[this.array.length];
System.arraycopy(this.array, 0, dest, 0, this.array.length);
return dest;
}
@Override
@SuppressWarnings("unchecked")
public <T> T[] toArray(final T[] dest) {
final int n= this.array.length;
if (dest.length < n) {
return Arrays.copyOf(this.array, n, (Class<? extends T[]>)dest.getClass());
}
System.arraycopy(this.array, 0, dest, 0, n);
if (dest.length > n) {
dest[n]= null;
}
return dest;
}
@Override
public void copyTo(final Object[] dest, final int destPos) {
System.arraycopy(this.array, 0, dest, destPos, this.array.length);
}
@Override
public void copyTo(final int srcPos, final Object[] dest, final int destPos, final int length) {
System.arraycopy(this.array, srcPos, dest, destPos, length);
}
@Override
public ImList<E> toImList() {
return new ImArrayList<>(this.array);
}
@Override
public ImIdentityList<E> toImIdentityList() {
return new ImArrayIdentityList<>(this.array);
}
@Override
public int hashCode() {
int hashCode= 0;
for (int i= 0; i < this.array.length; i++) {
if (this.array[i] != null) {
hashCode+= this.array[i].hashCode();
}
}
return hashCode;
}
@Override
public boolean equals(final @Nullable Object obj) {
if (obj == this) {
return true;
}
if (obj instanceof Set) {
final Set<?> other= (Set<?>) obj;
return (this.array.length == other.size()
&& containsAll(other) );
}
return false;
}
@Override
public String toString() {
return Arrays.toString(this.array);
}
}