Fixes: Make git diff work for staged files, do not launch submit if file list empty, make file model read only, add conveniences

Details: Add a type data field to the git file model
This commit is contained in:
Friedemann Kleint
2009-01-13 10:06:06 +01:00
parent 278ff78040
commit 4741a7282d
14 changed files with 171 additions and 32 deletions
+10
View File
@@ -234,6 +234,16 @@ void SubmitEditorWidget::setFileNameColumn(int c)
m_d->m_fileNameColumn = c; m_d->m_fileNameColumn = c;
} }
QAbstractItemView::SelectionMode SubmitEditorWidget::fileListSelectionMode() const
{
return m_d->m_ui.fileView->selectionMode();
}
void SubmitEditorWidget::setFileListSelectionMode(QAbstractItemView::SelectionMode sm)
{
m_d->m_ui.fileView->setSelectionMode(sm);
}
void SubmitEditorWidget::setFileModel(QAbstractItemModel *model) void SubmitEditorWidget::setFileModel(QAbstractItemModel *model)
{ {
m_d->m_ui.fileView->clearSelection(); // trigger the change signals m_d->m_ui.fileView->clearSelection(); // trigger the change signals
+5
View File
@@ -37,6 +37,7 @@
#include "utils_global.h" #include "utils_global.h"
#include <QtGui/QWidget> #include <QtGui/QWidget>
#include <QtGui/QAbstractItemView>
QT_BEGIN_NAMESPACE QT_BEGIN_NAMESPACE
class QPlainTextEdit; class QPlainTextEdit;
@@ -75,6 +76,7 @@ class QWORKBENCH_UTILS_EXPORT SubmitEditorWidget : public QWidget
Q_DISABLE_COPY(SubmitEditorWidget) Q_DISABLE_COPY(SubmitEditorWidget)
Q_PROPERTY(QString descriptionText READ descriptionText WRITE setDescriptionText DESIGNABLE true) Q_PROPERTY(QString descriptionText READ descriptionText WRITE setDescriptionText DESIGNABLE true)
Q_PROPERTY(int fileNameColumn READ fileNameColumn WRITE setFileNameColumn DESIGNABLE false) Q_PROPERTY(int fileNameColumn READ fileNameColumn WRITE setFileNameColumn DESIGNABLE false)
Q_PROPERTY(QAbstractItemView::SelectionMode fileListSelectionMode READ fileListSelectionMode WRITE setFileListSelectionMode DESIGNABLE true)
public: public:
explicit SubmitEditorWidget(QWidget *parent = 0); explicit SubmitEditorWidget(QWidget *parent = 0);
virtual ~SubmitEditorWidget(); virtual ~SubmitEditorWidget();
@@ -92,6 +94,9 @@ public:
int fileNameColumn() const; int fileNameColumn() const;
void setFileNameColumn(int c); void setFileNameColumn(int c);
QAbstractItemView::SelectionMode fileListSelectionMode() const;
void setFileListSelectionMode(QAbstractItemView::SelectionMode sm);
void setFileModel(QAbstractItemModel *model); void setFileModel(QAbstractItemModel *model);
QAbstractItemModel *fileModel() const; QAbstractItemModel *fileModel() const;
+6 -1
View File
@@ -130,6 +130,11 @@ static inline bool addStateFileSpecification(const QString &line, QList<CommitDa
\endcode \endcode
*/ */
bool CommitData::filesEmpty() const
{
return stagedFiles.empty() && unstagedFiles.empty() && untrackedFiles.empty();
}
bool CommitData::parseFilesFromStatus(const QString &output) bool CommitData::parseFilesFromStatus(const QString &output)
{ {
enum State { None, CommitFiles, NotUpdatedFiles, UntrackedFiles }; enum State { None, CommitFiles, NotUpdatedFiles, UntrackedFiles };
@@ -183,7 +188,7 @@ bool CommitData::parseFilesFromStatus(const QString &output)
} }
} }
} }
return !stagedFiles.empty() || !unstagedFiles.empty() || !untrackedFiles.empty(); return true;
} }
// Convert a spec pair list to a list of file names, optionally // Convert a spec pair list to a list of file names, optionally
+2
View File
@@ -77,6 +77,8 @@ struct CommitData
// from a git status output // from a git status output
bool parseFilesFromStatus(const QString &output); bool parseFilesFromStatus(const QString &output);
bool filesEmpty() const;
// Convenience to retrieve the file names from // Convenience to retrieve the file names from
// the specification list. Optionally filter for a certain state // the specification list. Optionally filter for a certain state
QStringList stagedFileNames(const QString &stateFilter = QString()) const; QStringList stagedFileNames(const QString &stateFilter = QString()) const;
+12 -4
View File
@@ -184,12 +184,14 @@ VCSBase::VCSBaseEditor
return rc; return rc;
} }
void GitClient::diff(const QString &workingDirectory, const QStringList &fileNames) void GitClient::diff(const QString &workingDirectory,
const QStringList &diffArgs,
const QStringList &fileNames)
{ {
if (Git::Constants::debug) if (Git::Constants::debug)
qDebug() << "diff" << workingDirectory << fileNames; qDebug() << "diff" << workingDirectory << fileNames;
QStringList arguments; QStringList arguments;
arguments << QLatin1String("diff") << QLatin1String("--") << fileNames; arguments << QLatin1String("diff") << diffArgs << QLatin1String("--") << fileNames;
const QString kind = QLatin1String(Git::Constants::GIT_DIFF_EDITOR_KIND); const QString kind = QLatin1String(Git::Constants::GIT_DIFF_EDITOR_KIND);
const QString title = tr("Git Diff"); const QString title = tr("Git Diff");
@@ -199,14 +201,16 @@ void GitClient::diff(const QString &workingDirectory, const QStringList &fileNam
} }
void GitClient::diff(const QString &workingDirectory, const QString &fileName) void GitClient::diff(const QString &workingDirectory,
const QStringList &diffArgs,
const QString &fileName)
{ {
if (Git::Constants::debug) if (Git::Constants::debug)
qDebug() << "diff" << workingDirectory << fileName; qDebug() << "diff" << workingDirectory << fileName;
QStringList arguments; QStringList arguments;
arguments << QLatin1String("diff"); arguments << QLatin1String("diff");
if (!fileName.isEmpty()) if (!fileName.isEmpty())
arguments << QLatin1String("--") << fileName; arguments << diffArgs << QLatin1String("--") << fileName;
const QString kind = QLatin1String(Git::Constants::GIT_DIFF_EDITOR_KIND); const QString kind = QLatin1String(Git::Constants::GIT_DIFF_EDITOR_KIND);
const QString title = tr("Git Diff %1").arg(fileName); const QString title = tr("Git Diff %1").arg(fileName);
@@ -710,6 +714,10 @@ bool GitClient::getCommitData(const QString &workingDirectory,
} }
// Filter out untracked files that are not part of the project // Filter out untracked files that are not part of the project
filterUntrackedFilesOfProject(repoDirectory, &d->untrackedFiles); filterUntrackedFilesOfProject(repoDirectory, &d->untrackedFiles);
if (d->filesEmpty()) {
*errorMessage = msgNoChangedFiles();
return false;
}
d->panelData.author = readConfigValue(workingDirectory, QLatin1String("user.name")); d->panelData.author = readConfigValue(workingDirectory, QLatin1String("user.name"));
d->panelData.email = readConfigValue(workingDirectory, QLatin1String("user.email")); d->panelData.email = readConfigValue(workingDirectory, QLatin1String("user.email"));
+2 -2
View File
@@ -78,8 +78,8 @@ public:
static QString findRepositoryForFile(const QString &fileName); static QString findRepositoryForFile(const QString &fileName);
static QString findRepositoryForDirectory(const QString &dir); static QString findRepositoryForDirectory(const QString &dir);
void diff(const QString &workingDirectory, const QString &fileName); void diff(const QString &workingDirectory, const QStringList &diffArgs, const QString &fileName);
void diff(const QString &workingDirectory, const QStringList &fileNames); void diff(const QString &workingDirectory, const QStringList &diffArgs, const QStringList &fileNames);
void status(const QString &workingDirectory); void status(const QString &workingDirectory);
void log(const QString &workingDirectory, const QString &fileName); void log(const QString &workingDirectory, const QString &fileName);
+16 -10
View File
@@ -450,19 +450,24 @@ void GitPlugin::extensionsInitialized()
m_projectExplorer = ExtensionSystem::PluginManager::instance()->getObject<ProjectExplorer::ProjectExplorerPlugin>(); m_projectExplorer = ExtensionSystem::PluginManager::instance()->getObject<ProjectExplorer::ProjectExplorerPlugin>();
} }
void GitPlugin::submitEditorDiff(const QStringList &files) void GitPlugin::submitEditorDiffStaged(const QStringList &files)
{ {
if (files.empty()) if (!files.empty())
return; m_gitClient->diff(m_submitRepository, QStringList(QLatin1String("--cached")), files);
m_gitClient->diff(m_submitRepository, files); }
void GitPlugin::submitEditorDiffUnstaged(const QStringList &files)
{
if (!files.empty())
m_gitClient->diff(m_submitRepository, QStringList(), files);
} }
void GitPlugin::diffCurrentFile() void GitPlugin::diffCurrentFile()
{ {
QFileInfo fileInfo = currentFile(); const QFileInfo fileInfo = currentFile();
QString fileName = fileInfo.fileName(); const QString fileName = fileInfo.fileName();
QString workingDirectory = fileInfo.absolutePath(); const QString workingDirectory = fileInfo.absolutePath();
m_gitClient->diff(workingDirectory, fileName); m_gitClient->diff(workingDirectory, QStringList(), fileName);
} }
void GitPlugin::diffCurrentProject() void GitPlugin::diffCurrentProject()
@@ -470,7 +475,7 @@ void GitPlugin::diffCurrentProject()
QString workingDirectory = getWorkingDirectory(); QString workingDirectory = getWorkingDirectory();
if (workingDirectory.isEmpty()) if (workingDirectory.isEmpty())
return; return;
m_gitClient->diff(workingDirectory, QString()); m_gitClient->diff(workingDirectory, QStringList(), QString());
} }
QFileInfo GitPlugin::currentFile() const QFileInfo GitPlugin::currentFile() const
@@ -640,7 +645,8 @@ Core::IEditor *GitPlugin::openSubmitEditor(const QString &fileName, const Commit
m_undoAction->setEnabled(false); m_undoAction->setEnabled(false);
m_redoAction->setEnabled(false); m_redoAction->setEnabled(false);
submitEditor->setCommitData(cd); submitEditor->setCommitData(cd);
connect(submitEditor, SIGNAL(diffSelectedFiles(QStringList)), this, SLOT(submitEditorDiff(QStringList))); connect(submitEditor, SIGNAL(diffStaged(QStringList)), this, SLOT(submitEditorDiffStaged(QStringList)));
connect(submitEditor, SIGNAL(diffUnstaged(QStringList)), this, SLOT(submitEditorDiffUnstaged(QStringList)));
return editor; return editor;
} }
+2 -1
View File
@@ -108,7 +108,8 @@ public slots:
private slots: private slots:
void diffCurrentFile(); void diffCurrentFile();
void diffCurrentProject(); void diffCurrentProject();
void submitEditorDiff(const QStringList &); void submitEditorDiffUnstaged(const QStringList &);
void submitEditorDiffStaged(const QStringList &);
void submitCurrentLog(); void submitCurrentLog();
void statusFile(); void statusFile();
void statusProject(); void statusProject();
+51 -12
View File
@@ -43,10 +43,22 @@
namespace Git { namespace Git {
namespace Internal { namespace Internal {
enum { FileTypeRole = Qt::UserRole + 1 };
enum FileType { StagedFile , UnstagedFile, UntrackedFile };
/* The problem with git is that no diff can be obtained to for a random
* multiselection of staged/unstaged files; it requires the --cached
* option for staged files. So, we set the file list to
* single selection and sort the files manual according to a type
* flag we add to the model. */
GitSubmitEditor::GitSubmitEditor(const VCSBase::VCSBaseSubmitEditorParameters *parameters, QWidget *parent) : GitSubmitEditor::GitSubmitEditor(const VCSBase::VCSBaseSubmitEditorParameters *parameters, QWidget *parent) :
VCSBaseSubmitEditor(parameters, new GitSubmitEditorWidget(parent)) VCSBaseSubmitEditor(parameters, new GitSubmitEditorWidget(parent)),
m_model(0)
{ {
setDisplayName(tr("Git Commit")); setDisplayName(tr("Git Commit"));
setFileListSelectionMode(QAbstractItemView::SingleSelection);
connect(this, SIGNAL(diffSelectedFiles(QStringList)), this, SLOT(slotDiffSelected(QStringList)));
} }
GitSubmitEditorWidget *GitSubmitEditor::submitEditorWidget() GitSubmitEditorWidget *GitSubmitEditor::submitEditorWidget()
@@ -54,14 +66,20 @@ GitSubmitEditorWidget *GitSubmitEditor::submitEditorWidget()
return static_cast<GitSubmitEditorWidget *>(widget()); return static_cast<GitSubmitEditorWidget *>(widget());
} }
static void addStateFileListToModel(const QList<CommitData::StateFilePair> &l, // Utility to add a list of state/file pairs to the model
VCSBase::SubmitFileModel *model, // setting a file type.
bool checked) static void addStateFileListToModel(const QList<CommitData::StateFilePair> &l,
bool checked, FileType ft,
VCSBase::SubmitFileModel *model)
{ {
typedef QList<CommitData::StateFilePair>::const_iterator ConstIterator; typedef QList<CommitData::StateFilePair>::const_iterator ConstIterator;
const ConstIterator cend = l.constEnd(); if (!l.empty()) {
for (ConstIterator it = l.constBegin(); it != cend; ++it) const ConstIterator cend = l.constEnd();
model->addFile(it->second, it->first, checked); const QVariant fileTypeData(ft);
for (ConstIterator it = l.constBegin(); it != cend; ++it)
model->addFile(it->second, it->first, checked).front()->setData(fileTypeData, FileTypeRole);
}
} }
void GitSubmitEditor::setCommitData(const CommitData &d) void GitSubmitEditor::setCommitData(const CommitData &d)
@@ -69,16 +87,37 @@ void GitSubmitEditor::setCommitData(const CommitData &d)
submitEditorWidget()->setPanelData(d.panelData); submitEditorWidget()->setPanelData(d.panelData);
submitEditorWidget()->setPanelInfo(d.panelInfo); submitEditorWidget()->setPanelInfo(d.panelInfo);
VCSBase::SubmitFileModel *model = new VCSBase::SubmitFileModel(this); m_model = new VCSBase::SubmitFileModel(this);
addStateFileListToModel(d.stagedFiles, model, true); addStateFileListToModel(d.stagedFiles, true, StagedFile, m_model);
addStateFileListToModel(d.unstagedFiles, model, false); addStateFileListToModel(d.unstagedFiles, false, UnstagedFile, m_model);
if (!d.untrackedFiles.empty()) { if (!d.untrackedFiles.empty()) {
const QString untrackedSpec = QLatin1String("untracked"); const QString untrackedSpec = QLatin1String("untracked");
const QVariant fileTypeData(UntrackedFile);
const QStringList::const_iterator cend = d.untrackedFiles.constEnd(); const QStringList::const_iterator cend = d.untrackedFiles.constEnd();
for (QStringList::const_iterator it = d.untrackedFiles.constBegin(); it != cend; ++it) for (QStringList::const_iterator it = d.untrackedFiles.constBegin(); it != cend; ++it)
model->addFile(*it, untrackedSpec, false); m_model->addFile(*it, untrackedSpec, false).front()->setData(fileTypeData, FileTypeRole);
} }
setFileModel(model); setFileModel(m_model);
}
void GitSubmitEditor::slotDiffSelected(const QStringList &files)
{
QList<QStandardItem *> fileRow = m_model->findRow(files.front(), fileNameColumn());
if (fileRow.empty())
return;
const FileType ft = static_cast<FileType>(fileRow.front()->data(FileTypeRole).toInt());
switch (ft) {
case StagedFile:
emit diffStaged(files);
break;
case UnstagedFile:
emit diffUnstaged(files);
break;
case UntrackedFile:
break;
}
} }
GitSubmitEditorPanelData GitSubmitEditor::panelData() const GitSubmitEditorPanelData GitSubmitEditor::panelData() const
+13
View File
@@ -38,6 +38,10 @@
#include <QtCore/QStringList> #include <QtCore/QStringList>
namespace VCSBase {
class SubmitFileModel;
}
namespace Git { namespace Git {
namespace Internal { namespace Internal {
@@ -54,8 +58,17 @@ public:
void setCommitData(const CommitData &); void setCommitData(const CommitData &);
GitSubmitEditorPanelData panelData() const; GitSubmitEditorPanelData panelData() const;
signals:
void diffStaged(const QStringList &);
void diffUnstaged(const QStringList &);
private slots:
void slotDiffSelected(const QStringList &);
private: private:
inline GitSubmitEditorWidget *submitEditorWidget(); inline GitSubmitEditorWidget *submitEditorWidget();
VCSBase::SubmitFileModel *m_model;
}; };
} // namespace Internal } // namespace Internal
+28 -1
View File
@@ -48,20 +48,47 @@ SubmitFileModel::SubmitFileModel(QObject *parent) :
setHorizontalHeaderLabels(headerLabels); setHorizontalHeaderLabels(headerLabels);
} }
QList<QStandardItem *> SubmitFileModel::addFile(const QString &fileName, const QString &status, bool checked) QList<QStandardItem *> SubmitFileModel::createFileRow(const QString &fileName, const QString &status, bool checked)
{ {
if (VCSBase::Constants::Internal::debug) if (VCSBase::Constants::Internal::debug)
qDebug() << Q_FUNC_INFO << fileName << status << checked; qDebug() << Q_FUNC_INFO << fileName << status << checked;
QStandardItem *statusItem = new QStandardItem(status); QStandardItem *statusItem = new QStandardItem(status);
statusItem->setCheckable(true); statusItem->setCheckable(true);
statusItem->setCheckState(checked ? Qt::Checked : Qt::Unchecked); statusItem->setCheckState(checked ? Qt::Checked : Qt::Unchecked);
statusItem->setFlags(Qt::ItemIsSelectable|Qt::ItemIsUserCheckable|Qt::ItemIsEnabled);
QStandardItem *fileItem = new QStandardItem(fileName); QStandardItem *fileItem = new QStandardItem(fileName);
fileItem->setFlags(Qt::ItemIsSelectable|Qt::ItemIsEnabled);
QList<QStandardItem *> row; QList<QStandardItem *> row;
row << statusItem << fileItem; row << statusItem << fileItem;
return row;
}
QList<QStandardItem *> SubmitFileModel::addFile(const QString &fileName, const QString &status, bool checked)
{
const QList<QStandardItem *> row = createFileRow(fileName, status, checked);
appendRow(row); appendRow(row);
return row; return row;
} }
QList<QStandardItem *> SubmitFileModel::rowAt(int row) const
{
const int colCount = columnCount();
QList<QStandardItem *> rc;
for (int c = 0; c < colCount; c++)
rc.push_back(item(row, c));
return rc;
}
QList<QStandardItem *> SubmitFileModel::findRow(const QString &text, int column) const
{
// Single item
const QList<QStandardItem *> items = findItems(text, Qt::MatchExactly, column);
if (items.empty())
return items;
// Compile row
return rowAt(items.front()->row());
}
unsigned SubmitFileModel::filter(const QStringList &filter, int column) unsigned SubmitFileModel::filter(const QStringList &filter, int column)
{ {
unsigned rc = 0; unsigned rc = 0;
+8 -1
View File
@@ -49,9 +49,16 @@ class VCSBASE_EXPORT SubmitFileModel : public QStandardItemModel
public: public:
explicit SubmitFileModel(QObject *parent = 0); explicit SubmitFileModel(QObject *parent = 0);
// Convenience to add a file plus status text. // Convenience to create and add rows containing a file plus status text.
static QList<QStandardItem *> createFileRow(const QString &fileName, const QString &status = QString(), bool checked = true);
QList<QStandardItem *> addFile(const QString &fileName, const QString &status = QString(), bool checked = true); QList<QStandardItem *> addFile(const QString &fileName, const QString &status = QString(), bool checked = true);
// Find convenience that returns the whole row (as opposed to QStandardItemModel::find).
QList<QStandardItem *> findRow(const QString &text, int column = 0) const;
// Convenience to obtain a row
QList<QStandardItem *> rowAt(int row) const;
// Filter for entries contained in the filter list. Returns the // Filter for entries contained in the filter list. Returns the
// number of deleted entries. // number of deleted entries.
unsigned filter(const QStringList &filter, int column); unsigned filter(const QStringList &filter, int column);
@@ -139,6 +139,17 @@ void VCSBaseSubmitEditor::setFileNameColumn(int c)
m_d->m_widget->setFileNameColumn(c); m_d->m_widget->setFileNameColumn(c);
} }
QAbstractItemView::SelectionMode VCSBaseSubmitEditor::fileListSelectionMode() const
{
return m_d->m_widget->fileListSelectionMode();
}
void VCSBaseSubmitEditor::setFileListSelectionMode(QAbstractItemView::SelectionMode sm)
{
m_d->m_widget->setFileListSelectionMode(sm);
}
void VCSBaseSubmitEditor::slotDescriptionChanged() void VCSBaseSubmitEditor::slotDescriptionChanged()
{ {
} }
@@ -39,6 +39,7 @@
#include <coreplugin/editormanager/ieditor.h> #include <coreplugin/editormanager/ieditor.h>
#include <QtCore/QList> #include <QtCore/QList>
#include <QtGui/QAbstractItemView>
QT_BEGIN_NAMESPACE QT_BEGIN_NAMESPACE
class QIcon; class QIcon;
@@ -92,6 +93,7 @@ class VCSBASE_EXPORT VCSBaseSubmitEditor : public Core::IEditor
{ {
Q_OBJECT Q_OBJECT
Q_PROPERTY(int fileNameColumn READ fileNameColumn WRITE setFileNameColumn DESIGNABLE false) Q_PROPERTY(int fileNameColumn READ fileNameColumn WRITE setFileNameColumn DESIGNABLE false)
Q_PROPERTY(QAbstractItemView::SelectionMode fileListSelectionMode READ fileListSelectionMode WRITE setFileListSelectionMode DESIGNABLE true)
public: public:
typedef QList<int> Context; typedef QList<int> Context;
@@ -105,6 +107,9 @@ public:
int fileNameColumn() const; int fileNameColumn() const;
void setFileNameColumn(int c); void setFileNameColumn(int c);
QAbstractItemView::SelectionMode fileListSelectionMode() const;
void setFileListSelectionMode(QAbstractItemView::SelectionMode sm);
// Core::IEditor // Core::IEditor
virtual bool createNew(const QString &contents); virtual bool createNew(const QString &contents);
virtual bool open(const QString &fileName); virtual bool open(const QString &fileName);