2013-05-07 14:02:08 +02:00
|
|
|
/****************************************************************************
|
|
|
|
|
**
|
2015-01-14 18:07:15 +01:00
|
|
|
** Copyright (C) 2015 The Qt Company Ltd.
|
|
|
|
|
** Contact: http://www.qt.io/licensing
|
2013-05-07 14:02:08 +02:00
|
|
|
**
|
|
|
|
|
** This file is part of Qt Creator.
|
|
|
|
|
**
|
|
|
|
|
** Commercial License Usage
|
|
|
|
|
** Licensees holding valid commercial Qt licenses may use this file in
|
|
|
|
|
** accordance with the commercial license agreement provided with the
|
|
|
|
|
** Software or, alternatively, in accordance with the terms contained in
|
2015-01-14 18:07:15 +01:00
|
|
|
** a written agreement between you and The Qt Company. For licensing terms and
|
|
|
|
|
** conditions see http://www.qt.io/terms-conditions. For further information
|
2014-10-01 13:21:18 +02:00
|
|
|
** use the contact form at http://www.qt.io/contact-us.
|
2013-05-07 14:02:08 +02:00
|
|
|
**
|
|
|
|
|
** GNU Lesser General Public License Usage
|
|
|
|
|
** Alternatively, this file may be used under the terms of the GNU Lesser
|
2014-10-01 13:21:18 +02:00
|
|
|
** General Public License version 2.1 or version 3 as published by the Free
|
|
|
|
|
** Software Foundation and appearing in the file LICENSE.LGPLv21 and
|
|
|
|
|
** LICENSE.LGPLv3 included in the packaging of this file. Please review the
|
|
|
|
|
** following information to ensure the GNU Lesser General Public License
|
|
|
|
|
** requirements will be met: https://www.gnu.org/licenses/lgpl.html and
|
|
|
|
|
** http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.
|
2013-05-07 14:02:08 +02:00
|
|
|
**
|
2015-01-14 18:07:15 +01:00
|
|
|
** In addition, as a special exception, The Qt Company gives you certain additional
|
|
|
|
|
** rights. These rights are described in The Qt Company LGPL Exception
|
2013-05-07 14:02:08 +02:00
|
|
|
** version 1.1, included in the file LGPL_EXCEPTION.txt in this package.
|
|
|
|
|
**
|
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
2013-05-30 12:15:22 +02:00
|
|
|
#include "diffeditor.h"
|
2013-05-07 14:02:08 +02:00
|
|
|
#include "diffeditorconstants.h"
|
2014-01-30 14:27:50 +01:00
|
|
|
#include "diffeditordocument.h"
|
2014-02-24 11:10:17 +01:00
|
|
|
#include "diffeditorguicontroller.h"
|
2014-01-30 14:27:50 +01:00
|
|
|
#include "sidebysidediffeditorwidget.h"
|
2014-02-13 16:43:28 +01:00
|
|
|
#include "unifieddiffeditorwidget.h"
|
2013-05-07 14:02:08 +02:00
|
|
|
|
|
|
|
|
#include <coreplugin/icore.h>
|
2013-06-03 12:32:07 +02:00
|
|
|
#include <coreplugin/coreconstants.h>
|
2014-01-30 17:04:54 +01:00
|
|
|
#include <coreplugin/minisplitter.h>
|
|
|
|
|
|
2014-09-26 09:14:03 +02:00
|
|
|
#include <texteditor/texteditor.h>
|
2014-01-30 17:04:54 +01:00
|
|
|
#include <texteditor/texteditorsettings.h>
|
|
|
|
|
#include <texteditor/displaysettings.h>
|
2014-09-24 12:40:28 +02:00
|
|
|
#include <texteditor/marginsettings.h>
|
2013-06-03 12:32:07 +02:00
|
|
|
|
2015-01-10 23:40:32 +02:00
|
|
|
#include <utils/fileutils.h>
|
2015-01-30 11:35:31 +01:00
|
|
|
#include <utils/qtcassert.h>
|
2015-01-10 23:40:32 +02:00
|
|
|
|
2014-02-13 16:43:28 +01:00
|
|
|
#include <QStackedWidget>
|
2013-05-07 14:02:08 +02:00
|
|
|
#include <QToolButton>
|
|
|
|
|
#include <QSpinBox>
|
|
|
|
|
#include <QStyle>
|
|
|
|
|
#include <QLabel>
|
2013-05-21 13:51:26 +02:00
|
|
|
#include <QHBoxLayout>
|
2013-05-23 13:36:27 +02:00
|
|
|
#include <QToolBar>
|
|
|
|
|
#include <QComboBox>
|
2014-06-30 15:04:36 +02:00
|
|
|
#include <QDir>
|
2014-02-13 16:43:28 +01:00
|
|
|
#include <QTextCodec>
|
2014-06-24 10:53:03 +03:00
|
|
|
#include <QTextBlock>
|
2014-02-13 16:43:28 +01:00
|
|
|
|
|
|
|
|
static const char settingsGroupC[] = "DiffEditor";
|
|
|
|
|
static const char diffEditorTypeKeyC[] = "DiffEditorType";
|
|
|
|
|
static const char sideBySideDiffEditorValueC[] = "SideBySide";
|
|
|
|
|
static const char unifiedDiffEditorValueC[] = "Unified";
|
|
|
|
|
|
|
|
|
|
static const char legacySettingsGroupC[] = "Git";
|
|
|
|
|
static const char useDiffEditorKeyC[] = "UseDiffEditor";
|
2013-05-07 14:02:08 +02:00
|
|
|
|
2014-01-30 17:04:54 +01:00
|
|
|
using namespace TextEditor;
|
|
|
|
|
|
2013-05-07 14:02:08 +02:00
|
|
|
namespace DiffEditor {
|
2014-01-30 17:04:54 +01:00
|
|
|
namespace Internal {
|
|
|
|
|
|
2014-09-26 11:37:54 +02:00
|
|
|
class DescriptionEditorWidget : public TextEditorWidget
|
2014-01-30 17:04:54 +01:00
|
|
|
{
|
|
|
|
|
Q_OBJECT
|
|
|
|
|
public:
|
|
|
|
|
DescriptionEditorWidget(QWidget *parent = 0);
|
|
|
|
|
virtual QSize sizeHint() const;
|
|
|
|
|
|
2014-06-24 10:53:03 +03:00
|
|
|
signals:
|
2015-01-28 13:45:02 +01:00
|
|
|
void requestBranchList();
|
2014-06-24 10:53:03 +03:00
|
|
|
|
2014-01-30 17:04:54 +01:00
|
|
|
protected:
|
2014-06-24 10:53:03 +03:00
|
|
|
void mouseMoveEvent(QMouseEvent *e);
|
|
|
|
|
void mouseReleaseEvent(QMouseEvent *e);
|
|
|
|
|
|
2014-09-24 12:40:28 +02:00
|
|
|
void setDisplaySettings(const DisplaySettings &ds);
|
|
|
|
|
void setMarginSettings(const MarginSettings &ms);
|
|
|
|
|
|
2014-06-24 10:53:03 +03:00
|
|
|
bool findContentsUnderCursor(const QTextCursor &cursor);
|
|
|
|
|
void highlightCurrentContents();
|
|
|
|
|
void handleCurrentContents();
|
|
|
|
|
|
|
|
|
|
private:
|
|
|
|
|
QTextCursor m_currentCursor;
|
2014-01-30 17:04:54 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
DescriptionEditorWidget::DescriptionEditorWidget(QWidget *parent)
|
2014-09-26 11:37:54 +02:00
|
|
|
: TextEditorWidget(parent)
|
2014-01-30 17:04:54 +01:00
|
|
|
{
|
2014-08-28 18:46:39 +02:00
|
|
|
setupFallBackEditor("DiffEditor.DescriptionEditor");
|
|
|
|
|
|
2014-01-30 17:04:54 +01:00
|
|
|
DisplaySettings settings = displaySettings();
|
|
|
|
|
settings.m_textWrapping = false;
|
|
|
|
|
settings.m_displayLineNumbers = false;
|
|
|
|
|
settings.m_highlightCurrentLine = false;
|
|
|
|
|
settings.m_displayFoldingMarkers = false;
|
|
|
|
|
settings.m_markTextChanges = false;
|
|
|
|
|
settings.m_highlightBlocks = false;
|
2014-09-26 11:37:54 +02:00
|
|
|
TextEditorWidget::setDisplaySettings(settings);
|
2014-01-30 17:04:54 +01:00
|
|
|
|
|
|
|
|
setCodeFoldingSupported(true);
|
|
|
|
|
setFrameStyle(QFrame::NoFrame);
|
|
|
|
|
|
|
|
|
|
setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Preferred);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QSize DescriptionEditorWidget::sizeHint() const
|
|
|
|
|
{
|
2014-09-26 11:37:54 +02:00
|
|
|
QSize size = TextEditorWidget::sizeHint();
|
2014-01-30 17:04:54 +01:00
|
|
|
size.setHeight(size.height() / 5);
|
|
|
|
|
return size;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void DescriptionEditorWidget::setDisplaySettings(const DisplaySettings &ds)
|
|
|
|
|
{
|
|
|
|
|
DisplaySettings settings = displaySettings();
|
|
|
|
|
settings.m_visualizeWhitespace = ds.m_visualizeWhitespace;
|
2014-09-26 11:37:54 +02:00
|
|
|
TextEditorWidget::setDisplaySettings(settings);
|
2014-01-30 17:04:54 +01:00
|
|
|
}
|
|
|
|
|
|
2014-09-24 12:40:28 +02:00
|
|
|
void DescriptionEditorWidget::setMarginSettings(const MarginSettings &ms)
|
|
|
|
|
{
|
|
|
|
|
Q_UNUSED(ms);
|
2014-09-26 11:37:54 +02:00
|
|
|
TextEditorWidget::setMarginSettings(MarginSettings());
|
2014-09-24 12:40:28 +02:00
|
|
|
}
|
|
|
|
|
|
2014-06-24 10:53:03 +03:00
|
|
|
void DescriptionEditorWidget::mouseMoveEvent(QMouseEvent *e)
|
|
|
|
|
{
|
|
|
|
|
if (e->buttons()) {
|
2014-09-26 11:37:54 +02:00
|
|
|
TextEditorWidget::mouseMoveEvent(e);
|
2014-06-24 10:53:03 +03:00
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Qt::CursorShape cursorShape;
|
|
|
|
|
|
|
|
|
|
const QTextCursor cursor = cursorForPosition(e->pos());
|
|
|
|
|
if (findContentsUnderCursor(cursor)) {
|
|
|
|
|
highlightCurrentContents();
|
|
|
|
|
cursorShape = Qt::PointingHandCursor;
|
|
|
|
|
} else {
|
|
|
|
|
setExtraSelections(OtherSelection, QList<QTextEdit::ExtraSelection>());
|
|
|
|
|
cursorShape = Qt::IBeamCursor;
|
|
|
|
|
}
|
|
|
|
|
|
2014-09-26 11:37:54 +02:00
|
|
|
TextEditorWidget::mouseMoveEvent(e);
|
2014-06-24 10:53:03 +03:00
|
|
|
viewport()->setCursor(cursorShape);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void DescriptionEditorWidget::mouseReleaseEvent(QMouseEvent *e)
|
|
|
|
|
{
|
|
|
|
|
if (e->button() == Qt::LeftButton && !(e->modifiers() & Qt::ShiftModifier)) {
|
|
|
|
|
const QTextCursor cursor = cursorForPosition(e->pos());
|
|
|
|
|
if (findContentsUnderCursor(cursor)) {
|
|
|
|
|
handleCurrentContents();
|
|
|
|
|
e->accept();
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2014-09-26 11:37:54 +02:00
|
|
|
TextEditorWidget::mouseReleaseEvent(e);
|
2014-06-24 10:53:03 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool DescriptionEditorWidget::findContentsUnderCursor(const QTextCursor &cursor)
|
|
|
|
|
{
|
|
|
|
|
m_currentCursor = cursor;
|
|
|
|
|
return cursor.block().text() == QLatin1String(Constants::EXPAND_BRANCHES);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void DescriptionEditorWidget::highlightCurrentContents()
|
|
|
|
|
{
|
|
|
|
|
QTextEdit::ExtraSelection sel;
|
|
|
|
|
sel.cursor = m_currentCursor;
|
|
|
|
|
sel.cursor.select(QTextCursor::LineUnderCursor);
|
|
|
|
|
sel.format.setFontUnderline(true);
|
2014-09-26 11:37:54 +02:00
|
|
|
setExtraSelections(TextEditorWidget::OtherSelection,
|
2014-06-24 10:53:03 +03:00
|
|
|
QList<QTextEdit::ExtraSelection>() << sel);
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void DescriptionEditorWidget::handleCurrentContents()
|
|
|
|
|
{
|
|
|
|
|
m_currentCursor.select(QTextCursor::LineUnderCursor);
|
|
|
|
|
m_currentCursor.removeSelectedText();
|
|
|
|
|
m_currentCursor.insertText(QLatin1String("Branches: Expanding..."));
|
2015-01-28 13:45:02 +01:00
|
|
|
emit requestBranchList();
|
2014-06-24 10:53:03 +03:00
|
|
|
}
|
|
|
|
|
|
2013-05-30 12:15:22 +02:00
|
|
|
///////////////////////////////// DiffEditor //////////////////////////////////
|
2013-05-07 14:02:08 +02:00
|
|
|
|
2014-08-21 21:12:04 +02:00
|
|
|
DiffEditor::DiffEditor(const QSharedPointer<DiffEditorDocument> &doc)
|
|
|
|
|
: m_document(doc)
|
2014-01-31 09:01:49 +01:00
|
|
|
, m_descriptionWidget(0)
|
2014-02-13 16:43:28 +01:00
|
|
|
, m_stackedWidget(0)
|
|
|
|
|
, m_sideBySideEditor(0)
|
|
|
|
|
, m_unifiedEditor(0)
|
|
|
|
|
, m_currentEditor(0)
|
2014-02-24 11:10:17 +01:00
|
|
|
, m_guiController(0)
|
2014-01-31 09:01:49 +01:00
|
|
|
, m_toolBar(0)
|
2013-12-16 16:19:40 +01:00
|
|
|
, m_entriesComboBox(0)
|
2014-01-30 17:04:54 +01:00
|
|
|
, m_toggleDescriptionAction(0)
|
2014-07-04 13:55:51 +02:00
|
|
|
, m_reloadAction(0)
|
2014-02-13 16:43:28 +01:00
|
|
|
, m_diffEditorSwitcher(0)
|
2013-05-07 14:02:08 +02:00
|
|
|
{
|
2015-01-30 11:35:31 +01:00
|
|
|
QTC_ASSERT(m_document, return);
|
|
|
|
|
|
2014-01-30 17:04:54 +01:00
|
|
|
QSplitter *splitter = new Core::MiniSplitter(Qt::Vertical);
|
2014-01-31 09:01:49 +01:00
|
|
|
|
2015-01-30 14:33:39 +01:00
|
|
|
m_descriptionWidget = new DescriptionEditorWidget(splitter);
|
2014-01-30 17:04:54 +01:00
|
|
|
m_descriptionWidget->setReadOnly(true);
|
|
|
|
|
splitter->addWidget(m_descriptionWidget);
|
2014-01-31 09:01:49 +01:00
|
|
|
|
2014-02-13 16:43:28 +01:00
|
|
|
m_stackedWidget = new QStackedWidget(splitter);
|
|
|
|
|
splitter->addWidget(m_stackedWidget);
|
|
|
|
|
|
|
|
|
|
m_sideBySideEditor = new SideBySideDiffEditorWidget(m_stackedWidget);
|
|
|
|
|
m_stackedWidget->addWidget(m_sideBySideEditor);
|
|
|
|
|
|
|
|
|
|
m_unifiedEditor = new UnifiedDiffEditorWidget(m_stackedWidget);
|
|
|
|
|
m_stackedWidget->addWidget(m_unifiedEditor);
|
2014-01-31 09:01:49 +01:00
|
|
|
|
2014-01-30 17:04:54 +01:00
|
|
|
setWidget(splitter);
|
|
|
|
|
|
|
|
|
|
|
2015-01-30 11:36:02 +01:00
|
|
|
DiffEditorController *control = controller();
|
|
|
|
|
m_guiController = new DiffEditorGuiController(control, this);
|
2014-01-31 09:01:49 +01:00
|
|
|
|
2015-01-30 14:33:39 +01:00
|
|
|
connect(m_descriptionWidget, &DescriptionEditorWidget::requestBranchList,
|
2015-01-30 11:36:02 +01:00
|
|
|
control, &DiffEditorController::expandBranchesRequested);
|
|
|
|
|
connect(control, &DiffEditorController::cleared, this, &DiffEditor::slotCleared);
|
|
|
|
|
connect(control, &DiffEditorController::diffFilesChanged,
|
2015-01-28 13:45:02 +01:00
|
|
|
this, &DiffEditor::slotDiffFilesChanged);
|
2015-01-30 11:36:02 +01:00
|
|
|
connect(control, &DiffEditorController::descriptionChanged,
|
2015-01-28 13:45:02 +01:00
|
|
|
this, &DiffEditor::slotDescriptionChanged);
|
2015-01-30 11:36:02 +01:00
|
|
|
connect(control, &DiffEditorController::descriptionEnablementChanged,
|
2015-01-28 13:45:02 +01:00
|
|
|
this, &DiffEditor::slotDescriptionVisibilityChanged);
|
|
|
|
|
connect(m_guiController, &DiffEditorGuiController::descriptionVisibilityChanged,
|
|
|
|
|
this, &DiffEditor::slotDescriptionVisibilityChanged);
|
|
|
|
|
connect(m_guiController, &DiffEditorGuiController::currentDiffFileIndexChanged,
|
|
|
|
|
this, &DiffEditor::activateEntry);
|
2014-01-31 09:01:49 +01:00
|
|
|
|
2015-01-30 11:36:02 +01:00
|
|
|
slotDescriptionChanged(control->description());
|
2014-01-31 09:01:49 +01:00
|
|
|
slotDescriptionVisibilityChanged();
|
2014-02-13 16:43:28 +01:00
|
|
|
|
|
|
|
|
showDiffEditor(readCurrentDiffEditorSetting());
|
|
|
|
|
|
|
|
|
|
toolBar();
|
2013-05-07 14:02:08 +02:00
|
|
|
}
|
|
|
|
|
|
2013-05-30 12:15:22 +02:00
|
|
|
DiffEditor::~DiffEditor()
|
2013-05-07 14:02:08 +02:00
|
|
|
{
|
2014-01-31 09:01:49 +01:00
|
|
|
delete m_toolBar;
|
2015-01-30 11:18:08 +01:00
|
|
|
delete m_widget;
|
2013-05-07 14:02:08 +02:00
|
|
|
}
|
|
|
|
|
|
2014-01-31 09:01:49 +01:00
|
|
|
Core::IEditor *DiffEditor::duplicate()
|
|
|
|
|
{
|
2014-08-21 21:12:04 +02:00
|
|
|
return new DiffEditor(m_document);
|
2014-01-31 09:01:49 +01:00
|
|
|
}
|
|
|
|
|
|
2014-02-13 16:43:28 +01:00
|
|
|
bool DiffEditor::open(QString *errorString,
|
|
|
|
|
const QString &fileName,
|
|
|
|
|
const QString &realFileName)
|
2013-05-07 14:02:08 +02:00
|
|
|
{
|
|
|
|
|
Q_UNUSED(realFileName)
|
2014-10-23 16:47:20 +02:00
|
|
|
return m_document->open(errorString, fileName);
|
2013-05-07 14:02:08 +02:00
|
|
|
}
|
|
|
|
|
|
2013-05-30 12:15:22 +02:00
|
|
|
Core::IDocument *DiffEditor::document()
|
2013-05-07 14:02:08 +02:00
|
|
|
{
|
2014-01-31 09:01:49 +01:00
|
|
|
return m_document.data();
|
2013-05-07 14:02:08 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static QToolBar *createToolBar(const QWidget *someWidget)
|
|
|
|
|
{
|
|
|
|
|
// Create
|
|
|
|
|
QToolBar *toolBar = new QToolBar;
|
2013-08-29 13:08:34 +02:00
|
|
|
toolBar->setToolButtonStyle(Qt::ToolButtonIconOnly);
|
2013-05-07 14:02:08 +02:00
|
|
|
const int size = someWidget->style()->pixelMetric(QStyle::PM_SmallIconSize);
|
|
|
|
|
toolBar->setIconSize(QSize(size, size));
|
|
|
|
|
|
|
|
|
|
return toolBar;
|
|
|
|
|
}
|
|
|
|
|
|
2013-05-30 12:15:22 +02:00
|
|
|
QWidget *DiffEditor::toolBar()
|
2013-05-07 14:02:08 +02:00
|
|
|
{
|
2014-01-31 09:01:49 +01:00
|
|
|
if (m_toolBar)
|
|
|
|
|
return m_toolBar;
|
2013-05-07 14:02:08 +02:00
|
|
|
|
2015-01-30 11:36:02 +01:00
|
|
|
DiffEditorController *control = controller();
|
|
|
|
|
|
2013-05-07 14:02:08 +02:00
|
|
|
// Create
|
2014-02-13 16:43:28 +01:00
|
|
|
m_toolBar = createToolBar(m_sideBySideEditor);
|
2013-05-21 13:51:26 +02:00
|
|
|
|
2013-05-23 13:36:27 +02:00
|
|
|
m_entriesComboBox = new QComboBox;
|
|
|
|
|
m_entriesComboBox->setMinimumContentsLength(20);
|
|
|
|
|
// Make the combo box prefer to expand
|
|
|
|
|
QSizePolicy policy = m_entriesComboBox->sizePolicy();
|
|
|
|
|
policy.setHorizontalPolicy(QSizePolicy::Expanding);
|
|
|
|
|
m_entriesComboBox->setSizePolicy(policy);
|
2015-01-28 13:45:02 +01:00
|
|
|
connect(m_entriesComboBox, static_cast<void (QComboBox::*)(int)>(&QComboBox::activated),
|
|
|
|
|
this, &DiffEditor::entryActivated);
|
2014-01-31 09:01:49 +01:00
|
|
|
m_toolBar->addWidget(m_entriesComboBox);
|
2013-05-07 14:02:08 +02:00
|
|
|
|
2014-01-31 09:01:49 +01:00
|
|
|
QToolButton *whitespaceButton = new QToolButton(m_toolBar);
|
2013-05-15 16:26:27 +02:00
|
|
|
whitespaceButton->setText(tr("Ignore Whitespace"));
|
2013-05-07 14:02:08 +02:00
|
|
|
whitespaceButton->setCheckable(true);
|
2015-01-30 11:36:02 +01:00
|
|
|
whitespaceButton->setChecked(control->isIgnoreWhitespace());
|
2014-10-24 14:11:12 +02:00
|
|
|
m_whitespaceButtonAction = m_toolBar->addWidget(whitespaceButton);
|
2013-05-07 14:02:08 +02:00
|
|
|
|
2014-01-31 09:01:49 +01:00
|
|
|
QLabel *contextLabel = new QLabel(m_toolBar);
|
2013-05-30 12:01:07 +02:00
|
|
|
contextLabel->setText(tr("Context Lines:"));
|
2013-06-04 10:53:03 +02:00
|
|
|
contextLabel->setContentsMargins(6, 0, 6, 0);
|
2014-10-24 14:11:12 +02:00
|
|
|
m_contextLabelAction = m_toolBar->addWidget(contextLabel);
|
2013-05-07 14:02:08 +02:00
|
|
|
|
2014-01-31 09:01:49 +01:00
|
|
|
QSpinBox *contextSpinBox = new QSpinBox(m_toolBar);
|
2014-02-13 16:43:28 +01:00
|
|
|
contextSpinBox->setRange(1, 100);
|
2015-01-30 11:36:02 +01:00
|
|
|
contextSpinBox->setValue(control->contextLinesNumber());
|
2013-05-30 12:01:07 +02:00
|
|
|
contextSpinBox->setFrame(false);
|
2014-02-13 16:43:28 +01:00
|
|
|
contextSpinBox->setSizePolicy(QSizePolicy::Minimum,
|
|
|
|
|
QSizePolicy::Expanding); // Mac Qt5
|
2014-10-24 14:11:12 +02:00
|
|
|
m_contextSpinBoxAction = m_toolBar->addWidget(contextSpinBox);
|
2013-05-07 14:02:08 +02:00
|
|
|
|
2014-01-31 09:01:49 +01:00
|
|
|
QToolButton *toggleDescription = new QToolButton(m_toolBar);
|
2014-02-13 16:43:28 +01:00
|
|
|
toggleDescription->setIcon(
|
|
|
|
|
QIcon(QLatin1String(Constants::ICON_TOP_BAR)));
|
2014-01-30 17:04:54 +01:00
|
|
|
toggleDescription->setCheckable(true);
|
2014-02-13 16:43:28 +01:00
|
|
|
toggleDescription->setChecked(m_guiController->isDescriptionVisible());
|
2014-01-31 09:01:49 +01:00
|
|
|
m_toggleDescriptionAction = m_toolBar->addWidget(toggleDescription);
|
2014-01-30 17:04:54 +01:00
|
|
|
slotDescriptionVisibilityChanged();
|
|
|
|
|
|
2014-02-13 16:43:28 +01:00
|
|
|
QToolButton *reloadButton = new QToolButton(m_toolBar);
|
2014-06-12 17:54:46 +02:00
|
|
|
reloadButton->setIcon(QIcon(QLatin1String(Core::Constants::ICON_RELOAD_GRAY)));
|
2014-02-13 16:43:28 +01:00
|
|
|
reloadButton->setToolTip(tr("Reload Editor"));
|
2014-07-04 13:55:51 +02:00
|
|
|
m_reloadAction = m_toolBar->addWidget(reloadButton);
|
2014-11-06 13:23:40 +01:00
|
|
|
slotReloaderChanged();
|
2014-07-04 13:55:51 +02:00
|
|
|
|
|
|
|
|
QToolButton *toggleSync = new QToolButton(m_toolBar);
|
|
|
|
|
toggleSync->setIcon(QIcon(QLatin1String(Core::Constants::ICON_LINK)));
|
|
|
|
|
toggleSync->setCheckable(true);
|
|
|
|
|
toggleSync->setChecked(m_guiController->horizontalScrollBarSynchronization());
|
|
|
|
|
toggleSync->setToolTip(tr("Synchronize Horizontal Scroll Bars"));
|
|
|
|
|
m_toolBar->addWidget(toggleSync);
|
2014-02-13 16:43:28 +01:00
|
|
|
|
|
|
|
|
m_diffEditorSwitcher = new QToolButton(m_toolBar);
|
|
|
|
|
m_toolBar->addWidget(m_diffEditorSwitcher);
|
|
|
|
|
updateDiffEditorSwitcher();
|
|
|
|
|
|
2015-01-28 13:45:02 +01:00
|
|
|
connect(whitespaceButton, &QToolButton::clicked,
|
2015-01-30 11:36:02 +01:00
|
|
|
control, &DiffEditorController::setIgnoreWhitespace);
|
|
|
|
|
connect(control, &DiffEditorController::ignoreWhitespaceChanged,
|
2015-01-28 13:45:02 +01:00
|
|
|
whitespaceButton, &QToolButton::setChecked);
|
|
|
|
|
connect(contextSpinBox, static_cast<void (QSpinBox::*)(int)>(&QSpinBox::valueChanged),
|
2015-01-30 11:36:02 +01:00
|
|
|
control, &DiffEditorController::setContextLinesNumber);
|
|
|
|
|
connect(control, &DiffEditorController::contextLinesNumberChanged,
|
2015-01-28 13:45:02 +01:00
|
|
|
contextSpinBox, &QSpinBox::setValue);
|
|
|
|
|
connect(toggleSync, &QAbstractButton::clicked,
|
|
|
|
|
m_guiController, &DiffEditorGuiController::setHorizontalScrollBarSynchronization);
|
|
|
|
|
connect(toggleDescription, &QAbstractButton::clicked,
|
|
|
|
|
m_guiController, &DiffEditorGuiController::setDescriptionVisible);
|
|
|
|
|
connect(m_diffEditorSwitcher, &QAbstractButton::clicked,
|
|
|
|
|
this, &DiffEditor::slotDiffEditorSwitched);
|
|
|
|
|
connect(reloadButton, &QAbstractButton::clicked,
|
2015-01-30 11:36:02 +01:00
|
|
|
control, &DiffEditorController::requestReload);
|
|
|
|
|
connect(control, &DiffEditorController::reloaderChanged,
|
2015-01-28 13:45:02 +01:00
|
|
|
this, &DiffEditor::slotReloaderChanged);
|
2015-01-30 11:36:02 +01:00
|
|
|
connect(control, &DiffEditorController::contextLinesNumberEnablementChanged,
|
2015-01-28 13:45:02 +01:00
|
|
|
this, &DiffEditor::slotReloaderChanged);
|
2013-12-16 16:19:40 +01:00
|
|
|
|
2014-01-31 09:01:49 +01:00
|
|
|
return m_toolBar;
|
2013-05-07 14:02:08 +02:00
|
|
|
}
|
|
|
|
|
|
2014-02-13 16:43:28 +01:00
|
|
|
DiffEditorController *DiffEditor::controller() const
|
2014-02-11 13:00:35 +01:00
|
|
|
{
|
2015-01-30 11:36:02 +01:00
|
|
|
return m_document->controller();
|
2014-02-11 13:00:35 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void DiffEditor::updateEntryToolTip()
|
|
|
|
|
{
|
|
|
|
|
const QString &toolTip = m_entriesComboBox->itemData(
|
|
|
|
|
m_entriesComboBox->currentIndex(), Qt::ToolTipRole).toString();
|
|
|
|
|
m_entriesComboBox->setToolTip(toolTip);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void DiffEditor::entryActivated(int index)
|
2013-05-23 13:36:27 +02:00
|
|
|
{
|
2014-02-11 13:00:35 +01:00
|
|
|
updateEntryToolTip();
|
2014-02-24 11:10:17 +01:00
|
|
|
m_guiController->setCurrentDiffFileIndex(index);
|
2014-02-11 13:00:35 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void DiffEditor::slotCleared(const QString &message)
|
|
|
|
|
{
|
|
|
|
|
Q_UNUSED(message)
|
|
|
|
|
|
|
|
|
|
m_entriesComboBox->clear();
|
|
|
|
|
updateEntryToolTip();
|
|
|
|
|
}
|
|
|
|
|
|
2014-02-13 16:43:28 +01:00
|
|
|
void DiffEditor::slotDiffFilesChanged(const QList<FileData> &diffFileList,
|
|
|
|
|
const QString &workingDirectory)
|
2014-02-11 13:00:35 +01:00
|
|
|
{
|
|
|
|
|
Q_UNUSED(workingDirectory)
|
|
|
|
|
|
2013-05-23 13:36:27 +02:00
|
|
|
m_entriesComboBox->clear();
|
|
|
|
|
const int count = diffFileList.count();
|
|
|
|
|
for (int i = 0; i < count; i++) {
|
2014-02-13 16:43:28 +01:00
|
|
|
const DiffFileInfo leftEntry = diffFileList.at(i).leftFileInfo;
|
|
|
|
|
const DiffFileInfo rightEntry = diffFileList.at(i).rightFileInfo;
|
2015-01-10 23:40:32 +02:00
|
|
|
const QString leftShortFileName = Utils::FileName::fromString(leftEntry.fileName).fileName();
|
|
|
|
|
const QString rightShortFileName = Utils::FileName::fromString(rightEntry.fileName).fileName();
|
2013-05-23 13:36:27 +02:00
|
|
|
QString itemText;
|
|
|
|
|
QString itemToolTip;
|
|
|
|
|
if (leftEntry.fileName == rightEntry.fileName) {
|
|
|
|
|
itemText = leftShortFileName;
|
|
|
|
|
|
|
|
|
|
if (leftEntry.typeInfo.isEmpty() && rightEntry.typeInfo.isEmpty()) {
|
|
|
|
|
itemToolTip = leftEntry.fileName;
|
|
|
|
|
} else {
|
|
|
|
|
itemToolTip = tr("[%1] vs. [%2] %3")
|
2014-02-13 16:43:28 +01:00
|
|
|
.arg(leftEntry.typeInfo,
|
|
|
|
|
rightEntry.typeInfo,
|
|
|
|
|
leftEntry.fileName);
|
2013-05-23 13:36:27 +02:00
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
if (leftShortFileName == rightShortFileName) {
|
|
|
|
|
itemText = leftShortFileName;
|
|
|
|
|
} else {
|
|
|
|
|
itemText = tr("%1 vs. %2")
|
2014-02-13 16:43:28 +01:00
|
|
|
.arg(leftShortFileName,
|
|
|
|
|
rightShortFileName);
|
2013-05-23 13:36:27 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (leftEntry.typeInfo.isEmpty() && rightEntry.typeInfo.isEmpty()) {
|
|
|
|
|
itemToolTip = tr("%1 vs. %2")
|
2014-02-13 16:43:28 +01:00
|
|
|
.arg(leftEntry.fileName,
|
|
|
|
|
rightEntry.fileName);
|
2013-05-23 13:36:27 +02:00
|
|
|
} else {
|
|
|
|
|
itemToolTip = tr("[%1] %2 vs. [%3] %4")
|
2014-02-13 16:43:28 +01:00
|
|
|
.arg(leftEntry.typeInfo,
|
|
|
|
|
leftEntry.fileName,
|
|
|
|
|
rightEntry.typeInfo,
|
|
|
|
|
rightEntry.fileName);
|
2013-05-23 13:36:27 +02:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
m_entriesComboBox->addItem(itemText);
|
2014-02-13 16:43:28 +01:00
|
|
|
m_entriesComboBox->setItemData(m_entriesComboBox->count() - 1,
|
|
|
|
|
itemToolTip, Qt::ToolTipRole);
|
2013-05-23 13:36:27 +02:00
|
|
|
}
|
|
|
|
|
updateEntryToolTip();
|
|
|
|
|
}
|
|
|
|
|
|
2013-05-30 12:15:22 +02:00
|
|
|
void DiffEditor::activateEntry(int index)
|
2013-05-23 13:36:27 +02:00
|
|
|
{
|
|
|
|
|
m_entriesComboBox->blockSignals(true);
|
|
|
|
|
m_entriesComboBox->setCurrentIndex(index);
|
|
|
|
|
m_entriesComboBox->blockSignals(false);
|
|
|
|
|
updateEntryToolTip();
|
|
|
|
|
}
|
|
|
|
|
|
2014-01-30 17:04:54 +01:00
|
|
|
void DiffEditor::slotDescriptionChanged(const QString &description)
|
|
|
|
|
{
|
|
|
|
|
m_descriptionWidget->setPlainText(description);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void DiffEditor::slotDescriptionVisibilityChanged()
|
|
|
|
|
{
|
2015-01-30 11:36:02 +01:00
|
|
|
const bool enabled = controller()->isDescriptionEnabled();
|
2014-02-24 11:10:17 +01:00
|
|
|
const bool visible = m_guiController->isDescriptionVisible();
|
2014-01-30 17:04:54 +01:00
|
|
|
|
|
|
|
|
m_descriptionWidget->setVisible(visible && enabled);
|
|
|
|
|
|
|
|
|
|
if (!m_toggleDescriptionAction)
|
|
|
|
|
return;
|
|
|
|
|
|
2014-01-31 09:01:49 +01:00
|
|
|
QWidget *toggle = m_toolBar->widgetForAction(m_toggleDescriptionAction);
|
2014-01-30 17:04:54 +01:00
|
|
|
if (visible)
|
|
|
|
|
toggle->setToolTip(tr("Hide Change Description"));
|
|
|
|
|
else
|
|
|
|
|
toggle->setToolTip(tr("Show Change Description"));
|
|
|
|
|
|
|
|
|
|
m_toggleDescriptionAction->setVisible(enabled);
|
2014-02-13 16:43:28 +01:00
|
|
|
}
|
|
|
|
|
|
2014-11-06 13:23:40 +01:00
|
|
|
void DiffEditor::slotReloaderChanged()
|
2014-07-04 13:55:51 +02:00
|
|
|
{
|
2015-01-30 11:36:02 +01:00
|
|
|
DiffEditorController *control = controller();
|
|
|
|
|
const DiffEditorReloader *reloader = control->reloader();
|
|
|
|
|
const bool contextVisible = control->isContextLinesNumberEnabled();
|
2014-11-06 13:23:40 +01:00
|
|
|
|
2014-10-24 14:11:12 +02:00
|
|
|
m_whitespaceButtonAction->setVisible(reloader);
|
2014-11-06 13:23:40 +01:00
|
|
|
m_contextLabelAction->setVisible(reloader && contextVisible);
|
|
|
|
|
m_contextSpinBoxAction->setVisible(reloader && contextVisible);
|
2014-07-04 13:55:51 +02:00
|
|
|
m_reloadAction->setVisible(reloader);
|
|
|
|
|
}
|
|
|
|
|
|
2014-02-13 16:43:28 +01:00
|
|
|
void DiffEditor::slotDiffEditorSwitched()
|
|
|
|
|
{
|
|
|
|
|
QWidget *oldEditor = m_currentEditor;
|
|
|
|
|
QWidget *newEditor = 0;
|
|
|
|
|
if (oldEditor == m_sideBySideEditor)
|
|
|
|
|
newEditor = m_unifiedEditor;
|
|
|
|
|
else if (oldEditor == m_unifiedEditor)
|
|
|
|
|
newEditor = m_sideBySideEditor;
|
|
|
|
|
else
|
|
|
|
|
newEditor = readCurrentDiffEditorSetting();
|
|
|
|
|
|
|
|
|
|
showDiffEditor(newEditor);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void DiffEditor::updateDiffEditorSwitcher()
|
|
|
|
|
{
|
|
|
|
|
if (!m_diffEditorSwitcher)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
QIcon actionIcon;
|
|
|
|
|
QString actionToolTip;
|
|
|
|
|
if (m_currentEditor == m_unifiedEditor) {
|
|
|
|
|
actionIcon = QIcon(QLatin1String(Constants::ICON_SIDE_BY_SIDE_DIFF));
|
|
|
|
|
actionToolTip = tr("Switch to Side By Side Diff Editor");
|
|
|
|
|
} else if (m_currentEditor == m_sideBySideEditor) {
|
|
|
|
|
actionIcon = QIcon(QLatin1String(Constants::ICON_UNIFIED_DIFF));
|
|
|
|
|
actionToolTip = tr("Switch to Unified Diff Editor");
|
|
|
|
|
}
|
2014-01-30 17:04:54 +01:00
|
|
|
|
2014-02-13 16:43:28 +01:00
|
|
|
m_diffEditorSwitcher->setIcon(actionIcon);
|
|
|
|
|
m_diffEditorSwitcher->setToolTip(actionToolTip);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void DiffEditor::showDiffEditor(QWidget *newEditor)
|
|
|
|
|
{
|
|
|
|
|
if (m_currentEditor == newEditor)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
if (m_currentEditor == m_sideBySideEditor)
|
|
|
|
|
m_sideBySideEditor->setDiffEditorGuiController(0);
|
|
|
|
|
else if (m_currentEditor == m_unifiedEditor)
|
|
|
|
|
m_unifiedEditor->setDiffEditorGuiController(0);
|
|
|
|
|
|
|
|
|
|
m_currentEditor = newEditor;
|
|
|
|
|
|
|
|
|
|
if (m_currentEditor == m_unifiedEditor)
|
|
|
|
|
m_unifiedEditor->setDiffEditorGuiController(m_guiController);
|
|
|
|
|
else if (m_currentEditor == m_sideBySideEditor)
|
|
|
|
|
m_sideBySideEditor->setDiffEditorGuiController(m_guiController);
|
|
|
|
|
|
|
|
|
|
m_stackedWidget->setCurrentWidget(m_currentEditor);
|
|
|
|
|
|
|
|
|
|
writeCurrentDiffEditorSetting(m_currentEditor);
|
|
|
|
|
updateDiffEditorSwitcher();
|
2014-08-13 13:28:37 +02:00
|
|
|
widget()->setFocusProxy(m_currentEditor);
|
2014-02-13 16:43:28 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QWidget *DiffEditor::readLegacyCurrentDiffEditorSetting()
|
|
|
|
|
{
|
|
|
|
|
QSettings *s = Core::ICore::settings();
|
|
|
|
|
|
|
|
|
|
s->beginGroup(QLatin1String(legacySettingsGroupC));
|
|
|
|
|
const bool legacyExists = s->contains(QLatin1String(useDiffEditorKeyC));
|
|
|
|
|
const bool legacyEditor = s->value(
|
|
|
|
|
QLatin1String(useDiffEditorKeyC), true).toBool();
|
|
|
|
|
if (legacyExists)
|
|
|
|
|
s->remove(QLatin1String(useDiffEditorKeyC));
|
|
|
|
|
s->endGroup();
|
|
|
|
|
|
|
|
|
|
QWidget *currentEditor = m_sideBySideEditor;
|
|
|
|
|
if (!legacyEditor)
|
|
|
|
|
currentEditor = m_unifiedEditor;
|
|
|
|
|
|
|
|
|
|
if (legacyExists && currentEditor == m_unifiedEditor)
|
|
|
|
|
writeCurrentDiffEditorSetting(currentEditor);
|
|
|
|
|
|
|
|
|
|
return currentEditor;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QWidget *DiffEditor::readCurrentDiffEditorSetting()
|
|
|
|
|
{
|
|
|
|
|
// replace it with m_sideBySideEditor when dropping legacy stuff
|
|
|
|
|
QWidget *defaultEditor = readLegacyCurrentDiffEditorSetting();
|
|
|
|
|
|
|
|
|
|
QSettings *s = Core::ICore::settings();
|
|
|
|
|
s->beginGroup(QLatin1String(settingsGroupC));
|
|
|
|
|
const QString editorString = s->value(
|
|
|
|
|
QLatin1String(diffEditorTypeKeyC)).toString();
|
|
|
|
|
s->endGroup();
|
|
|
|
|
if (editorString == QLatin1String(unifiedDiffEditorValueC))
|
|
|
|
|
return m_unifiedEditor;
|
|
|
|
|
|
|
|
|
|
if (editorString == QLatin1String(sideBySideDiffEditorValueC))
|
|
|
|
|
return m_sideBySideEditor;
|
|
|
|
|
|
|
|
|
|
return defaultEditor;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void DiffEditor::writeCurrentDiffEditorSetting(QWidget *currentEditor)
|
|
|
|
|
{
|
|
|
|
|
const QString editorString = currentEditor == m_unifiedEditor
|
|
|
|
|
? QLatin1String(unifiedDiffEditorValueC)
|
|
|
|
|
: QLatin1String(sideBySideDiffEditorValueC);
|
|
|
|
|
QSettings *s = Core::ICore::settings();
|
|
|
|
|
s->beginGroup(QLatin1String(settingsGroupC));
|
|
|
|
|
s->setValue(QLatin1String(diffEditorTypeKeyC), editorString);
|
|
|
|
|
s->endGroup();
|
2014-01-30 17:04:54 +01:00
|
|
|
}
|
|
|
|
|
|
2015-01-30 14:59:41 +01:00
|
|
|
} // namespace Internal
|
2013-05-07 14:02:08 +02:00
|
|
|
} // namespace DiffEditor
|
2014-01-30 17:04:54 +01:00
|
|
|
|
|
|
|
|
#include "diffeditor.moc"
|