TupleWithDiff.java
package sprouts.impl;
import org.jspecify.annotations.Nullable;
import sprouts.SequenceChange;
import sprouts.Tuple;
import java.util.*;
public final class TupleWithDiff<T extends @Nullable Object> implements Tuple<T>, SequenceDiffOwner {
private final TupleTree<T> _tupleTree;
private final SequenceDiff _diffToPrevious;
@SuppressWarnings("NullAway")
public static <T> Tuple<T> of(
boolean allowsNull,
Class<T> type,
List<T> items
) {
return new TupleWithDiff<>(TupleTree.of(allowsNull, type, items), null);
}
static <T> Tuple<T> of(
boolean allowsNull,
Class<T> type,
@Nullable T... items
) {
return new TupleWithDiff<>(TupleTree.of(allowsNull, type, items), null);
}
static <T> Tuple<T> ofAnyArray(
boolean allowsNull,
Class<T> type,
Object array
) {
return new TupleWithDiff<>(TupleTree.ofAnyArray(allowsNull, type, array), null);
}
@SuppressWarnings("NullAway")
public TupleWithDiff(
TupleTree<T> data, @Nullable SequenceDiff diffToPrevious
) {
_tupleTree = data;
_diffToPrevious = ( diffToPrevious == null ? SequenceDiff.initial() : diffToPrevious );
}
TupleTree<T> getData() {
return _tupleTree;
}
@Override
public Class<T> type() {
return _tupleTree.type();
}
@Override
public int size() {
return _tupleTree.size();
}
@Override
@SuppressWarnings("NullAway")
public T get(int index) {
return _tupleTree.get(index);
}
@Override
public boolean allowsNull() {
return _tupleTree.allowsNull();
}
@Override
public Tuple<T> slice(int from, int to) {
if ( from < 0 || to > this.size() )
throw new IndexOutOfBoundsException();
if ( from > to )
throw new IllegalArgumentException();
int newSize = (to - from);
if ( newSize == this.size() )
return this;
if ( newSize == 0 ) {
SequenceDiff diff = SequenceDiff.of(this, SequenceChange.RETAIN, -1, 0);
return new TupleWithDiff<>(_tupleTree.clear(), diff);
}
TupleTree<T> slice = _tupleTree.slice(from, to);
if ( Util.refEquals(slice, _tupleTree) )
return this;
SequenceDiff diff = SequenceDiff.of(this, SequenceChange.RETAIN, from, slice.size());
return new TupleWithDiff<>(slice, diff);
}
@Override
public Tuple<T> removeRange(int from, int to) {
if (from < 0 || to > this.size())
throw new IndexOutOfBoundsException("from: " + from + ", to: " + to + ", size: " + this.size());
if (from > to)
throw new IllegalArgumentException();
int numberOfItemsToRemove = to - from;
if (numberOfItemsToRemove == 0)
return this;
if (numberOfItemsToRemove == this.size()) {
SequenceDiff diff = SequenceDiff.of(this, SequenceChange.REMOVE, 0, this.size());
return new TupleWithDiff<>(_tupleTree.clear(), diff);
}
SequenceDiff diff = SequenceDiff.of(this, SequenceChange.REMOVE, from, numberOfItemsToRemove);
return new TupleWithDiff<>(_tupleTree.removeRange(from, to), diff);
}
@Override
public Tuple<T> removeAll(Tuple<T> properties) {
TupleTree<T> withoutItems = _tupleTree.removeAll(properties);
if ( Util.refEquals(withoutItems, _tupleTree) )
return this;
SequenceDiff diff = SequenceDiff.of(this, SequenceChange.REMOVE, -1, this.size() - withoutItems.size());
return new TupleWithDiff<>(withoutItems, diff);
}
@Override
public Tuple<T> addAt(int index, T item) {
TupleTree<T> withoutItems = _tupleTree.addAt(index, item);
if ( Util.refEquals(withoutItems, _tupleTree) )
return this;
SequenceDiff diff = SequenceDiff.of(this, SequenceChange.ADD, index, 1);
return new TupleWithDiff<>(withoutItems, diff);
}
@Override
public Tuple<T> setAt(int index, T item) {
TupleTree<T> newItems = _tupleTree.setAt(index, item);
if ( Util.refEquals(newItems, _tupleTree) )
return this;
SequenceDiff diff = SequenceDiff.of(this, SequenceChange.SET, index, 1);
return new TupleWithDiff<>(newItems, diff);
}
@Override
public Tuple<T> addAllAt(int index, Tuple<T> tuple) {
TupleTree<T> newItems = _tupleTree.addAllAt(index, tuple);
if ( Util.refEquals(newItems, _tupleTree) )
return this;
SequenceDiff diff = SequenceDiff.of(this, SequenceChange.ADD, index, tuple.size());
return new TupleWithDiff<>(newItems, diff);
}
@Override
public Tuple<T> setAllAt(int index, Tuple<T> tuple) {
TupleTree<T> newItems = _tupleTree.setAllAt(index, tuple);
if ( Util.refEquals(newItems, _tupleTree) )
return this;
SequenceDiff diff = SequenceDiff.of(this, SequenceChange.SET, index, tuple.size());
return new TupleWithDiff<>(newItems, diff);
}
@Override
public Tuple<T> retainAll(Tuple<T> tuple) {
if (tuple.isEmpty()) {
SequenceDiff diff = SequenceDiff.of(this, SequenceChange.RETAIN, -1, 0);
return new TupleWithDiff<>(_tupleTree.clear(), diff);
}
int[] indicesOfThingsToKeep = new int[this.size()];
int newSize = 0;
int singleSequenceIndex = size() > 0 ? -2 : -1;
int retainSequenceSize = 0;
for (int i = 0; i < this.size(); i++) {
int index = tuple.firstIndexOf(get(i));
if (index != -1) {
indicesOfThingsToKeep[newSize] = i;
newSize++;
if (singleSequenceIndex != -1) {
if (singleSequenceIndex == -2)
singleSequenceIndex = i;
else if (i > singleSequenceIndex + retainSequenceSize)
singleSequenceIndex = -1;
}
if (singleSequenceIndex >= 0)
retainSequenceSize++;
} else {
indicesOfThingsToKeep[newSize] = -1;
}
}
TupleTree<T> newItems = _tupleTree._retainAll(singleSequenceIndex, newSize, indicesOfThingsToKeep);
if ( Util.refEquals(newItems, _tupleTree) )
return this;
SequenceDiff diff = SequenceDiff.of(this, SequenceChange.RETAIN, singleSequenceIndex, newSize);
return new TupleWithDiff<>(newItems, diff);
}
@Override
public Tuple<T> clear() {
if (this.size() == 0)
return this;
SequenceDiff diff = SequenceDiff.of(this, SequenceChange.CLEAR, 0, size());
return new TupleWithDiff<>(_tupleTree.clear(), diff);
}
@Override
public Tuple<T> sort(Comparator<T> comparator) {
TupleTree<T> newItems = _tupleTree.sort(comparator);
SequenceDiff diff = SequenceDiff.of(this, SequenceChange.SORT, -1, size());
return new TupleWithDiff<>(newItems, diff);
}
@Override
public Tuple<T> makeDistinct() {
TupleTree<T> distinctItems = _tupleTree.makeDistinct();
if (Util.refEquals(distinctItems, _tupleTree))
return this;
SequenceDiff diff = SequenceDiff.of(this, SequenceChange.DISTINCT, -1, this.size() - distinctItems.size());
return new TupleWithDiff<>(distinctItems, diff);
}
@Override
public Tuple<T> reversed() {
TupleTree<T> newItems = _tupleTree.reversed();
if ( Util.refEquals(newItems, _tupleTree) )
return this;
SequenceDiff diff = SequenceDiff.of(this, SequenceChange.REVERSE, -1, this.size());
return new TupleWithDiff<>(newItems, diff);
}
@Override
public Iterator<T> iterator() {
return _tupleTree.iterator();
}
@Override
public Spliterator<T> spliterator() {
return _tupleTree.spliterator();
}
@Override
public String toString() {
return _tupleTree.toString();
}
@Override
public boolean equals(Object obj) {
return _tupleTree.equals(obj);
}
@Override
public int hashCode() {
return _tupleTree.hashCode();
}
@Override
public Optional<SequenceDiff> differenceFromPrevious() {
return Optional.ofNullable(_diffToPrevious);
}
}