blob: 3ffe26f2ad518fa5c6f0965a93982890bbdd5509 [file] [log] [blame]
/*
* Copyright 2000-2010 JetBrains s.r.o.
*
* 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 com.intellij.openapi.vcs;
import com.intellij.openapi.util.text.StringUtil;
import com.intellij.util.Function;
import java.util.Collections;
import java.util.Iterator;
import java.util.LinkedList;
import java.util.List;
/**
* @author irengrig
*/
public abstract class Ring<T extends Comparable<T>> {
protected final LinkedList<T> myFreeNumbers;
protected final T myFirst;
protected T myNextAvailable;
public Ring(final T first) {
myFreeNumbers = new LinkedList<T>();
myNextAvailable = first;
myFirst = first;
}
public Ring(final T first, final List<T> used) {
this(first);
for (T t : used) {
minus(t);
}
}
public void reset() {
myFreeNumbers.clear();
myNextAvailable = myFirst;
}
public void back(final T number) {
final int idx = Collections.binarySearch(myFreeNumbers, number);
// todo remove
if (idx >= 0) {
System.out.println("* " + number);
}
assert idx < 0 || idx >= myFreeNumbers.size();
myFreeNumbers.add(- idx - 1, number);
}
// todo debug
public boolean haveInFree(final T t) {
return myFreeNumbers.contains(t);
}
public boolean minus(final T t) {
while (myNextAvailable.compareTo(t) <= 0) {
myFreeNumbers.add(myNextAvailable);
myNextAvailable = getNext(myNextAvailable);
}
return myFreeNumbers.remove(t);
}
public List<T> getUsed() {
final List<T> result = new LinkedList<T>();
T current = myFirst;
final Iterator<T> iterator = myFreeNumbers.iterator();
T currentUsed = iterator.hasNext() ? iterator.next() : null;
while (current.compareTo(myNextAvailable) < 0) {
final int compResult = (currentUsed == null) ? -1 : current.compareTo(currentUsed);
if (compResult < 0) {
result.add(current);
} else if (compResult == 0) {
if (iterator.hasNext()) {
currentUsed = iterator.next();
} else {
currentUsed = null;
}
}
current = getNext(current);
}
return result;
}
public T getMaxNumber() {
return myNextAvailable;
}
protected abstract T getNext(final T t);
public T getFree() {
if (myFreeNumbers.isEmpty()) {
final T tmp = myNextAvailable;
myNextAvailable = getNext(myNextAvailable);
return tmp;
}
return myFreeNumbers.removeFirst();
}
public static class IntegerRing extends Ring<Integer> {
public IntegerRing() {
super(0);
}
public IntegerRing(final List<Integer> used) {
super(0, used);
}
@Override
protected Integer getNext(Integer integer) {
return integer + 1;
}
public int size() {
return myNextAvailable - myFreeNumbers.size();
}
public boolean isNumUsed(final int num) {
return (num < myNextAvailable) && (! myFreeNumbers.contains(num));
}
}
@Override
public String toString() {
return "Ring{" +
"myFreeNumbers=" + StringUtil.join(myFreeNumbers, new Function<T, String>() {
@Override
public String fun(T t) {
return t.toString();
}
}, ",") +
", myFirst=" + myFirst +
", myNextAvailable=" + myNextAvailable +
'}';
}
}