/*
* Copyright 2000-2015 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.diff.tools.util.side;
import com.intellij.diff.DiffContext;
import com.intellij.diff.actions.impl.OpenInEditorWithMouseAction;
import com.intellij.diff.actions.impl.SetEditorSettingsAction;
import com.intellij.diff.contents.DocumentContent;
import com.intellij.diff.requests.ContentDiffRequest;
import com.intellij.diff.requests.DiffRequest;
import com.intellij.diff.requests.SimpleDiffRequest;
import com.intellij.diff.tools.holders.EditorHolderFactory;
import com.intellij.diff.tools.holders.TextEditorHolder;
import com.intellij.diff.tools.util.DiffDataKeys;
import com.intellij.diff.tools.util.SyncScrollSupport;
import com.intellij.diff.tools.util.SyncScrollSupport.ThreesideSyncScrollSupport;
import com.intellij.diff.tools.util.base.InitialScrollPositionSupport;
import com.intellij.diff.tools.util.base.TextDiffSettingsHolder;
import com.intellij.diff.tools.util.base.TextDiffViewerUtil;
import com.intellij.diff.util.*;
import com.intellij.openapi.actionSystem.AnAction;
import com.intellij.openapi.editor.Editor;
import com.intellij.openapi.editor.LogicalPosition;
import com.intellij.openapi.editor.event.DocumentEvent;
import com.intellij.openapi.editor.event.VisibleAreaEvent;
import com.intellij.openapi.editor.event.VisibleAreaListener;
import com.intellij.openapi.editor.ex.EditorEx;
import com.intellij.openapi.editor.ex.EditorMarkupModel;
import com.intellij.openapi.util.Pair;
import com.intellij.pom.Navigatable;
import com.intellij.util.containers.ContainerUtil;
import org.jetbrains.annotations.NonNls;
import org.jetbrains.annotations.NotNull;
import org.jetbrains.annotations.Nullable;
import consulo.annotations.RequiredDispatchThread;
import javax.swing.*;
import java.util.List;
public abstract class ThreesideTextDiffViewer extends ThreesideDiffViewer<TextEditorHolder> {
@Nullable private List<? extends EditorEx> myEditors;
@NotNull private final List<? extends EditorEx> myEditableEditors;
@NotNull private final MyVisibleAreaListener myVisibleAreaListener1 = new MyVisibleAreaListener(Side.LEFT);
@NotNull private final MyVisibleAreaListener myVisibleAreaListener2 = new MyVisibleAreaListener(Side.RIGHT);
@Nullable protected ThreesideSyncScrollSupport mySyncScrollSupport;
@NotNull protected final SetEditorSettingsAction myEditorSettingsAction;
public ThreesideTextDiffViewer(@NotNull DiffContext context, @NotNull ContentDiffRequest request) {
super(context, request, TextEditorHolder.TextEditorHolderFactory.INSTANCE);
//new MyFocusOppositePaneAction().setupAction(myPanel, this); // TODO
myEditorSettingsAction = new SetEditorSettingsAction(getTextSettings(), getEditors());
myEditorSettingsAction.applyDefaults();
new MyOpenInEditorWithMouseAction().install(getEditors());
myEditableEditors = TextDiffViewerUtil.getEditableEditors(getEditors());
TextDiffViewerUtil.checkDifferentDocuments(myRequest);
}
@Override
@RequiredDispatchThread
protected void onInit() {
super.onInit();
installEditorListeners();
}
@Override
@RequiredDispatchThread
protected void onDispose() {
destroyEditorListeners();
super.onDispose();
}
@NotNull
@Override
protected List<TextEditorHolder> createEditorHolders(@NotNull EditorHolderFactory<TextEditorHolder> factory) {
List<TextEditorHolder> holders = super.createEditorHolders(factory);
boolean[] forceReadOnly = TextDiffViewerUtil.checkForceReadOnly(myContext, myRequest);
for (int i = 0; i < 3; i++) {
if (forceReadOnly[i]) holders.get(i).getEditor().setViewer(true);
}
ThreeSide.LEFT.select(holders).getEditor().setVerticalScrollbarOrientation(EditorEx.VERTICAL_SCROLLBAR_LEFT);
((EditorMarkupModel)ThreeSide.BASE.select(holders).getEditor().getMarkupModel()).setErrorStripeVisible(false);
for (TextEditorHolder holder : holders) {
DiffUtil.disableBlitting(holder.getEditor());
}
return holders;
}
@NotNull
@Override
protected List<JComponent> createTitles() {
return DiffUtil.createSyncHeightComponents(DiffUtil.createTextTitles(myRequest, getEditors()));
}
//
// Listeners
//
@RequiredDispatchThread
protected void installEditorListeners() {
new TextDiffViewerUtil.EditorActionsPopup(createEditorPopupActions()).install(getEditors());
new TextDiffViewerUtil.EditorFontSizeSynchronizer(getEditors()).install(this);
getEditor(ThreeSide.LEFT).getScrollingModel().addVisibleAreaListener(myVisibleAreaListener1);
getEditor(ThreeSide.BASE).getScrollingModel().addVisibleAreaListener(myVisibleAreaListener1);
getEditor(ThreeSide.BASE).getScrollingModel().addVisibleAreaListener(myVisibleAreaListener2);
getEditor(ThreeSide.RIGHT).getScrollingModel().addVisibleAreaListener(myVisibleAreaListener2);
SyncScrollSupport.SyncScrollable scrollable1 = getSyncScrollable(Side.LEFT);
SyncScrollSupport.SyncScrollable scrollable2 = getSyncScrollable(Side.RIGHT);
if (scrollable1 != null && scrollable2 != null) {
mySyncScrollSupport = new ThreesideSyncScrollSupport(getEditors(), scrollable1, scrollable2);
myEditorSettingsAction.setSyncScrollSupport(mySyncScrollSupport);
}
}
@RequiredDispatchThread
public void destroyEditorListeners() {
getEditor(ThreeSide.LEFT).getScrollingModel().removeVisibleAreaListener(myVisibleAreaListener1);
getEditor(ThreeSide.BASE).getScrollingModel().removeVisibleAreaListener(myVisibleAreaListener1);
getEditor(ThreeSide.BASE).getScrollingModel().removeVisibleAreaListener(myVisibleAreaListener2);
getEditor(ThreeSide.RIGHT).getScrollingModel().removeVisibleAreaListener(myVisibleAreaListener2);
mySyncScrollSupport = null;
}
protected void disableSyncScrollSupport(boolean disable) {
if (mySyncScrollSupport != null) {
if (disable) {
mySyncScrollSupport.enterDisableScrollSection();
}
else {
mySyncScrollSupport.exitDisableScrollSection();
}
}
}
//
// Diff
//
@NotNull
public TextDiffSettingsHolder.TextDiffSettings getTextSettings() {
return TextDiffViewerUtil.getTextSettings(myContext);
}
@NotNull
protected List<AnAction> createEditorPopupActions() {
return TextDiffViewerUtil.createEditorPopupActions();
}
@Override
protected void onDocumentChange(@NotNull DocumentEvent event) {
super.onDocumentChange(event);
myContentPanel.repaintDividers();
}
//
// Getters
//
@NotNull
public EditorEx getCurrentEditor() {
return getEditor(getCurrentSide());
}
@NotNull
public DocumentContent getCurrentContent() {
return getContent(getCurrentSide());
}
@NotNull
protected List<? extends DocumentContent> getContents() {
//noinspection unchecked
return (List)myRequest.getContents();
}
@NotNull
public List<? extends EditorEx> getEditors() {
if (myEditors == null) {
myEditors = ContainerUtil.map(getEditorHolders(), holder -> holder.getEditor());
}
return myEditors;
}
@NotNull
protected List<? extends EditorEx> getEditableEditors() {
return myEditableEditors;
}
@NotNull
public EditorEx getEditor(@NotNull ThreeSide side) {
return side.select(getEditors());
}
@NotNull
public DocumentContent getContent(@NotNull ThreeSide side) {
return side.select(getContents());
}
@Nullable
public ThreeSide getEditorSide(@Nullable Editor editor) {
if (getEditor(ThreeSide.BASE) == editor) return ThreeSide.BASE;
if (getEditor(ThreeSide.RIGHT) == editor) return ThreeSide.RIGHT;
if (getEditor(ThreeSide.LEFT) == editor) return ThreeSide.LEFT;
return null;
}
//
// Abstract
//
@RequiredDispatchThread
protected void scrollToLine(@NotNull ThreeSide side, int line) {
DiffUtil.scrollEditor(getEditor(side), line, false);
setCurrentSide(side);
}
@Nullable
protected abstract SyncScrollSupport.SyncScrollable getSyncScrollable(@NotNull Side side);
@RequiredDispatchThread
@NotNull
protected LogicalPosition transferPosition(@NotNull ThreeSide baseSide,
@NotNull ThreeSide targetSide,
@NotNull LogicalPosition position) {
if (mySyncScrollSupport == null) return position;
if (baseSide == targetSide) return position;
SyncScrollSupport.SyncScrollable scrollable12 = mySyncScrollSupport.getScrollable12();
SyncScrollSupport.SyncScrollable scrollable23 = mySyncScrollSupport.getScrollable23();
int baseLine; // line number in BASE
if (baseSide == ThreeSide.LEFT) {
baseLine = scrollable12.transfer(Side.LEFT, position.line);
}
else if (baseSide == ThreeSide.RIGHT) {
baseLine = scrollable23.transfer(Side.RIGHT, position.line);
}
else {
baseLine = position.line;
}
int targetLine;
if (targetSide == ThreeSide.LEFT) {
targetLine = scrollable12.transfer(Side.RIGHT, baseLine);
}
else if (targetSide == ThreeSide.RIGHT) {
targetLine = scrollable23.transfer(Side.LEFT, baseLine);
}
else {
targetLine = baseLine;
}
return new LogicalPosition(targetLine, position.column);
}
//
// Misc
//
@Nullable
@Override
protected Navigatable getNavigatable() {
return getCurrentContent().getNavigatable(LineCol.fromCaret(getCurrentEditor()));
}
public static boolean canShowRequest(@NotNull DiffContext context, @NotNull DiffRequest request) {
return ThreesideDiffViewer.canShowRequest(context, request, TextEditorHolder.TextEditorHolderFactory.INSTANCE);
}
//
// Actions
//
private class MyOpenInEditorWithMouseAction extends OpenInEditorWithMouseAction {
@Override
protected Navigatable getNavigatable(@NotNull Editor editor, int line) {
ThreeSide side = getEditorSide(editor);
if (side == null) return null;
return getContent(side).getNavigatable(new LineCol(line));
}
}
protected class MyToggleAutoScrollAction extends TextDiffViewerUtil.ToggleAutoScrollAction {
public MyToggleAutoScrollAction() {
super(getTextSettings());
}
}
//
// Helpers
//
@Nullable
@Override
public Object getData(@NonNls String dataId) {
if (DiffDataKeys.CURRENT_EDITOR.is(dataId)) {
return getCurrentEditor();
}
return super.getData(dataId);
}
private class MyVisibleAreaListener implements VisibleAreaListener {
@NotNull Side mySide;
public MyVisibleAreaListener(@NotNull Side side) {
mySide = side;
}
@Override
public void visibleAreaChanged(VisibleAreaEvent e) {
if (mySyncScrollSupport != null) mySyncScrollSupport.visibleAreaChanged(e);
myContentPanel.repaint();
}
}
protected abstract class MyInitialScrollPositionHelper extends InitialScrollPositionSupport.ThreesideInitialScrollHelper {
@NotNull
@Override
protected List<? extends Editor> getEditors() {
return ThreesideTextDiffViewer.this.getEditors();
}
@Override
protected void disableSyncScroll(boolean value) {
disableSyncScrollSupport(value);
}
@Override
protected boolean doScrollToLine() {
if (myScrollToLine == null) return false;
scrollToLine(myScrollToLine.first, myScrollToLine.second);
return true;
}
}
protected class TextShowPartialDiffAction extends ShowPartialDiffAction {
public TextShowPartialDiffAction(@NotNull PartialDiffMode mode) {
super(mode);
}
@NotNull
@Override
protected SimpleDiffRequest createRequest() {
SimpleDiffRequest request = super.createRequest();
ThreeSide currentSide = getCurrentSide();
LogicalPosition currentPosition = DiffUtil.getCaretPosition(getCurrentEditor());
// we won't use DiffUserDataKeysEx.EDITORS_CARET_POSITION to avoid desync scroll position (as they can point to different places)
// TODO: pass EditorsVisiblePositions in case if view was scrolled without changing caret position ?
if (currentSide == mySide1) {
request.putUserData(DiffUserDataKeys.SCROLL_TO_LINE, Pair.create(Side.LEFT, currentPosition.line));
}
else if (currentSide == mySide2) {
request.putUserData(DiffUserDataKeys.SCROLL_TO_LINE, Pair.create(Side.RIGHT, currentPosition.line));
}
else {
LogicalPosition position1 = transferPosition(currentSide, mySide1, currentPosition);
LogicalPosition position2 = transferPosition(currentSide, mySide2, currentPosition);
request.putUserData(DiffUserDataKeysEx.EDITORS_CARET_POSITION, new LogicalPosition[]{position1, position2});
}
return request;
}
}
}