blob: 1d8f68b4886bbea8bd09099dbaf8af9ec4b96e1c [file] [log] [blame]
/*
* Copyright 2000-2013 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.util.diff;
import gnu.trove.TIntArrayList;
import org.jetbrains.annotations.TestOnly;
import java.util.Arrays;
import java.util.BitSet;
/**
* @author dyoma
*/
class Reindexer {
private final int[][] myOldIndecies = new int[2][];
private final int[] myOriginalLengths = new int[]{-1, -1};
private final int[] myDiscardedLengths = new int[]{-1, -1};
public int[][] discardUnique(int[] ints1, int[] ints2) {
int[] discarded1 = discard(ints2, ints1, 0);
return new int[][]{discarded1, discard(discarded1, ints2, 1)};
}
@TestOnly
void idInit(int length1, int length2) {
myOriginalLengths[0] = length1;
myOriginalLengths[1] = length2;
myDiscardedLengths[0] = length1;
myDiscardedLengths[1] = length2;
for (int j = 0; j < 2; j++) {
int originalLength = myOriginalLengths[j];
myOldIndecies[j] = new int[originalLength];
for (int i = 0; i < originalLength; i++) {
myOldIndecies[j][i] = i;
}
}
}
@TestOnly
int restoreIndex(int index, int array) {
return myOldIndecies[array][index];
}
private int[] discard(int[] needed, int[] toDiscard, int arrayIndex) {
myOriginalLengths[arrayIndex] = toDiscard.length;
int[] sorted1 = createSorted(needed);
TIntArrayList discarded = new TIntArrayList(toDiscard.length);
TIntArrayList oldIndecies = new TIntArrayList(toDiscard.length);
for (int i = 0; i < toDiscard.length; i++) {
int index = toDiscard[i];
if (Arrays.binarySearch(sorted1, index) >= 0) {
discarded.add(index);
oldIndecies.add(i);
}
}
myOldIndecies[arrayIndex] = oldIndecies.toNativeArray();
myDiscardedLengths[arrayIndex] = discarded.size();
return discarded.toNativeArray();
}
private int[] createSorted(int[] ints1) {
int[] sorted1 = new int[ints1.length];
System.arraycopy(ints1, 0, sorted1, 0, ints1.length);
Arrays.sort(sorted1);
return sorted1;
}
public void reindex(BitSet[] discardedChanges, LCSBuilder builder) {
BitSet changes1;
BitSet changes2;
if (myDiscardedLengths[0] == myOriginalLengths[0] && myDiscardedLengths[1] == myOriginalLengths[1]) {
changes1 = discardedChanges[0];
changes2 = discardedChanges[1];
}
else {
changes1 = new BitSet(myOriginalLengths[0]);
changes2 = new BitSet(myOriginalLengths[1]);
int x = 0;
int y = 0;
while (x < myDiscardedLengths[0] || y < myDiscardedLengths[1]) {
if ((x < myDiscardedLengths[0] && y < myDiscardedLengths[1]) && !discardedChanges[0].get(x) && !discardedChanges[1].get(y)) {
x = increment(myOldIndecies[0], x, changes1, myOriginalLengths[0]);
y = increment(myOldIndecies[1], y, changes2, myOriginalLengths[1]);
continue;
}
if (discardedChanges[0].get(x)) {
changes1.set(getOriginal(myOldIndecies[0], x));
x = increment(myOldIndecies[0], x, changes1, myOriginalLengths[0]);
continue;
}
if (discardedChanges[1].get(y)) {
changes2.set(getOriginal(myOldIndecies[1], y));
y = increment(myOldIndecies[1], y, changes2, myOriginalLengths[1]);
continue;
}
}
if (myDiscardedLengths[0] == 0) {
changes1.set(0, myOriginalLengths[0]);
}
else {
changes1.set(0, myOldIndecies[0][0]);
}
if (myDiscardedLengths[1] == 0) {
changes2.set(0, myOriginalLengths[1]);
}
else {
changes2.set(0, myOldIndecies[1][0]);
}
}
int x = 0;
int y = 0;
while (x < myOriginalLengths[0] && y < myOriginalLengths[1]) {
int startX = x;
while (x < myOriginalLengths[0] && y < myOriginalLengths[1] && !changes1.get(x) && !changes2.get(y)) {
x++;
y++;
}
if (x > startX) builder.addEqual(x - startX);
int dx = 0;
int dy = 0;
while (x < myOriginalLengths[0] && changes1.get(x)) {
dx++;
x++;
}
while (y < myOriginalLengths[1] && changes2.get(y)) {
dy++;
y++;
}
if (dx != 0 || dy != 0) builder.addChange(dx, dy);
}
if (x != myOriginalLengths[0] || y != myOriginalLengths[1]) builder.addChange(myOriginalLengths[0] - x, myOriginalLengths[1] - y);
}
private int getOriginal(int[] indexes, int i) {
return indexes[i];
}
private int increment(int[] indexes, int i, BitSet set, int length) {
if (i + 1 < indexes.length) {
set.set(indexes[i] + 1, indexes[i + 1]);
}
else {
set.set(indexes[i] + 1, length);
}
return i + 1;
}
}