blob: a97b1c502aa222e1e3e4614ba1a856ea94d43cb3 [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.openapi.diff.impl.processing;
import com.intellij.openapi.diagnostic.Logger;
import com.intellij.openapi.diff.impl.string.DiffString;
import com.intellij.openapi.diff.ex.DiffFragment;
import com.intellij.openapi.diff.impl.ComparisonPolicy;
import com.intellij.openapi.diff.impl.highlighting.FragmentSide;
import com.intellij.openapi.diff.impl.highlighting.Util;
import com.intellij.openapi.util.text.StringUtil;
import com.intellij.util.diff.FilesTooBigForDiffException;
import org.jetbrains.annotations.NotNull;
import java.util.ArrayList;
public interface DiffCorrection {
DiffFragment[] correct(DiffFragment[] fragments) throws FilesTooBigForDiffException;
class TrueLineBlocks implements DiffCorrection, FragmentProcessor<FragmentsCollector> {
private static final Logger LOG = Logger.getInstance("#com.intellij.openapi.diff.impl.processing.DiffCorrection.TrueLineBlocks");
private final DiffPolicy myDiffPolicy;
@NotNull private final ComparisonPolicy myComparisonPolicy;
public TrueLineBlocks(@NotNull ComparisonPolicy comparisonPolicy) {
myDiffPolicy = new DiffPolicy.LineBlocks(comparisonPolicy);
myComparisonPolicy = comparisonPolicy;
}
@Override
public DiffFragment[] correct(DiffFragment[] fragments) throws FilesTooBigForDiffException {
FragmentsCollector collector = new FragmentsCollector();
collector.processAll(fragments, this);
return collector.toArray();
}
@Override
public void process(@NotNull DiffFragment fragment, @NotNull FragmentsCollector collector) throws FilesTooBigForDiffException {
DiffString text1 = fragment.getText1();
DiffString text2 = fragment.getText2();
if (!fragment.isEqual()) {
if (myComparisonPolicy.isEqual(fragment))
fragment = myComparisonPolicy.createFragment(text1, text2);
collector.add(fragment);
} else {
assert text1 != null;
assert text2 != null;
DiffString[] lines1 = text1.tokenize();
DiffString[] lines2 = text2.tokenize();
LOG.assertTrue(lines1.length == lines2.length);
for (int i = 0; i < lines1.length; i++)
collector.addAll(myDiffPolicy.buildFragments(lines1[i], lines2[i]));
}
}
public DiffFragment[] correctAndNormalize(DiffFragment[] fragments) throws FilesTooBigForDiffException {
return Normalize.INSTANCE.correct(correct(fragments));
}
}
class ChangedSpace implements DiffCorrection, FragmentProcessor<FragmentsCollector> {
private final DiffPolicy myDiffPolicy;
private final ComparisonPolicy myComparisonPolicy;
public ChangedSpace(ComparisonPolicy policy) {
myComparisonPolicy = policy;
myDiffPolicy = new DiffPolicy.ByChar(myComparisonPolicy);
}
@Override
public void process(@NotNull DiffFragment fragment, @NotNull FragmentsCollector collector) throws FilesTooBigForDiffException {
if (!fragment.isChange()) {
collector.add(fragment);
return;
}
DiffString text1 = fragment.getText1();
DiffString text2 = fragment.getText2();
while (StringUtil.startsWithChar(text1, '\n') || StringUtil.startsWithChar(text2, '\n')) {
DiffString newLine1 = null;
DiffString newLine2 = null;
if (StringUtil.startsWithChar(text1, '\n')) {
newLine1 = DiffString.create("\n");
text1 = text1.substring(1);
}
if (StringUtil.startsWithChar(text2, '\n')) {
newLine2 = DiffString.create("\n");
text2 = text2.substring(1);
}
collector.add(new DiffFragment(newLine1, newLine2));
}
DiffString spaces1 = text1.getLeadingSpaces();
DiffString spaces2 = text2.getLeadingSpaces();
if (spaces1.isEmpty() && spaces2.isEmpty()) {
DiffFragment trailing = myComparisonPolicy.createFragment(text1, text2);
collector.add(trailing);
return;
}
collector.addAll(myDiffPolicy.buildFragments(spaces1, spaces2));
DiffFragment textFragment = myComparisonPolicy
.createFragment(text1.substring(spaces1.length(), text1.length()), text2.substring(spaces2.length(), text2.length()));
collector.add(textFragment);
}
@Override
public DiffFragment[] correct(DiffFragment[] fragments) throws FilesTooBigForDiffException {
FragmentsCollector collector = new FragmentsCollector();
collector.processAll(fragments, this);
return collector.toArray();
}
}
interface FragmentProcessor<Collector> {
void process(@NotNull DiffFragment fragment, @NotNull Collector collector) throws FilesTooBigForDiffException;
}
class BaseFragmentRunner<ActualRunner extends BaseFragmentRunner> {
private final ArrayList<DiffFragment> myItems = new ArrayList<DiffFragment>();
private int myIndex = 0;
private DiffFragment[] myFragments;
public void add(DiffFragment fragment) {
actualAdd(fragment);
}
protected final void actualAdd(DiffFragment fragment) {
if (isEmpty(fragment)) return;
myItems.add(fragment);
}
public DiffFragment[] toArray() {
return myItems.toArray(new DiffFragment[myItems.size()]);
}
protected int getIndex() { return myIndex; }
public DiffFragment[] getFragments() { return myFragments; }
public void processAll(DiffFragment[] fragments, FragmentProcessor<ActualRunner> processor) throws FilesTooBigForDiffException {
myFragments = fragments;
for (;myIndex < myFragments.length; myIndex++) {
DiffFragment fragment = myFragments[myIndex];
processor.process(fragment, (ActualRunner)this);
}
}
public static int getTextLength(DiffString text) {
return text != null ? text.length() : 0;
}
public static boolean isEmpty(DiffFragment fragment) {
return getTextLength(fragment.getText1()) == 0 &&
getTextLength(fragment.getText2()) == 0;
}
}
class FragmentsCollector extends BaseFragmentRunner<FragmentsCollector> {
public void addAll(DiffFragment[] fragments) {
for (int i = 0; i < fragments.length; i++) {
add(fragments[i]);
}
}
}
class FragmentBuffer extends BaseFragmentRunner<FragmentBuffer> {
private int myMark = -1;
private int myMarkMode = -1;
public void markIfNone(int mode) {
if (mode == myMarkMode || myMark == -1) {
if (myMark == -1) myMark = getIndex();
} else {
flushMarked();
myMark = getIndex();
}
myMarkMode = mode;
}
@Override
public void add(DiffFragment fragment) {
flushMarked();
super.add(fragment);
}
protected void flushMarked() {
if (myMark != -1) {
actualAdd(Util.concatenate(getFragments(), myMark, getIndex()));
myMark = -1;
}
}
@Override
public void processAll(DiffFragment[] fragments, FragmentProcessor<FragmentBuffer> processor) throws FilesTooBigForDiffException {
super.processAll(fragments, processor);
flushMarked();
}
}
class ConcatenateSingleSide implements DiffCorrection, FragmentProcessor<FragmentBuffer> {
public static final DiffCorrection INSTANCE = new ConcatenateSingleSide();
private static final int DEFAULT_MODE = 1;
@Override
public DiffFragment[] correct(DiffFragment[] fragments) throws FilesTooBigForDiffException {
FragmentBuffer buffer = new FragmentBuffer();
buffer.processAll(fragments, this);
return buffer.toArray();
}
@Override
public void process(@NotNull DiffFragment fragment, @NotNull FragmentBuffer buffer) {
if (fragment.isOneSide()) buffer.markIfNone(DEFAULT_MODE);
else buffer.add(fragment);
}
}
class UnitEquals implements DiffCorrection, FragmentProcessor<FragmentBuffer> {
public static final DiffCorrection INSTANCE = new UnitEquals();
private static final int EQUAL_MODE = 1;
private static final int FORMATTING_MODE = 2;
@Override
public DiffFragment[] correct(DiffFragment[] fragments) throws FilesTooBigForDiffException {
FragmentBuffer buffer = new FragmentBuffer();
buffer.processAll(fragments, this);
return buffer.toArray();
}
@Override
public void process(@NotNull DiffFragment fragment, @NotNull FragmentBuffer buffer) {
if (fragment.isEqual()) buffer.markIfNone(EQUAL_MODE);
else if (ComparisonPolicy.TRIM_SPACE.isEqual(fragment)) buffer.markIfNone(FORMATTING_MODE);
else buffer.add(fragment);
}
}
class Normalize implements DiffCorrection {
public static final DiffCorrection INSTANCE = new Normalize();
private Normalize() {}
@Override
public DiffFragment[] correct(DiffFragment[] fragments) throws FilesTooBigForDiffException {
return UnitEquals.INSTANCE.correct(ConcatenateSingleSide.INSTANCE.correct(fragments));
}
}
class ConnectSingleSideToChange implements DiffCorrection, FragmentProcessor<FragmentBuffer> {
public static final ConnectSingleSideToChange INSTANCE = new ConnectSingleSideToChange();
private static final int CHANGE = 1;
@Override
public DiffFragment[] correct(DiffFragment[] fragments) throws FilesTooBigForDiffException {
FragmentBuffer buffer = new FragmentBuffer();
buffer.processAll(fragments, this);
return buffer.toArray();
}
@Override
public void process(@NotNull DiffFragment fragment, @NotNull FragmentBuffer buffer) {
if (fragment.isEqual()) buffer.add(fragment);
else if (fragment.isOneSide()) {
DiffString text = FragmentSide.chooseSide(fragment).getText(fragment);
if (StringUtil.endsWithChar(text, '\n'))
buffer.add(fragment);
else
buffer.markIfNone(CHANGE);
} else buffer.markIfNone(CHANGE);
}
}
}