aboutsummaryrefslogtreecommitdiffstats
path: root/src/plugins/git/stashdialog.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'src/plugins/git/stashdialog.cpp')
-rw-r--r--src/plugins/git/stashdialog.cpp411
1 files changed, 411 insertions, 0 deletions
diff --git a/src/plugins/git/stashdialog.cpp b/src/plugins/git/stashdialog.cpp
new file mode 100644
index 00000000000..a39559ad597
--- /dev/null
+++ b/src/plugins/git/stashdialog.cpp
@@ -0,0 +1,411 @@
+/**************************************************************************
+**
+** This file is part of Qt Creator
+**
+** Copyright (c) 2009 Nokia Corporation and/or its subsidiary(-ies).
+**
+** Contact: Nokia Corporation ([email protected])
+**
+** Commercial Usage
+**
+** Licensees holding valid Qt Commercial licenses may use this file in
+** accordance with the Qt Commercial License Agreement provided with the
+** Software or, alternatively, in accordance with the terms contained in
+** a written agreement between you and Nokia.
+**
+** GNU Lesser General Public License Usage
+**
+** Alternatively, this file may be used under the terms of the GNU Lesser
+** General Public License version 2.1 as published by the Free Software
+** Foundation and appearing in the file LICENSE.LGPL included in the
+** packaging of this file. Please review the following information to
+** ensure the GNU Lesser General Public License version 2.1 requirements
+** will be met: http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.
+**
+** If you are unsure which license is appropriate for your use, please
+** contact the sales department at http://qt.nokia.com/contact.
+**
+**************************************************************************/
+
+#include "stashdialog.h"
+#include "gitclient.h"
+#include "gitplugin.h"
+#include "gitutils.h"
+#include "ui_stashdialog.h"
+
+#include <utils/qtcassert.h>
+#include <vcsbase/vcsbaseoutputwindow.h>
+
+#include <QtCore/QDebug>
+#include <QtCore/QModelIndex>
+#include <QtCore/QDateTime>
+#include <QtGui/QStandardItemModel>
+#include <QtGui/QSortFilterProxyModel>
+#include <QtGui/QItemSelectionModel>
+#include <QtGui/QMessageBox>
+#include <QtGui/QPushButton>
+
+enum { NameColumn, BranchColumn, MessageColumn, ColumnCount };
+
+namespace Git {
+namespace Internal {
+
+static inline GitClient *gitClient()
+{
+ return GitPlugin::instance()->gitClient();
+}
+
+static inline QList<QStandardItem*> stashModelRowItems(const Stash &s)
+{
+ Qt::ItemFlags itemFlags = Qt::ItemIsSelectable|Qt::ItemIsEnabled;
+ QStandardItem *nameItem = new QStandardItem(s.name);
+ nameItem->setFlags(itemFlags);
+ QStandardItem *branchItem = new QStandardItem(s.branch);
+ branchItem->setFlags(itemFlags);
+ QStandardItem *messageItem = new QStandardItem(s.message);
+ messageItem->setFlags(itemFlags);
+ QList<QStandardItem*> rc;
+ rc << nameItem << branchItem << messageItem;
+ return rc;
+}
+
+// ----------- StashModel
+class StashModel : public QStandardItemModel {
+public:
+ explicit StashModel(QObject *parent = 0);
+
+ void setStashes(const QList<Stash> &stashes);
+ const Stash &at(int i) { return m_stashes.at(i); }
+
+private:
+ QList<Stash> m_stashes;
+};
+
+StashModel::StashModel(QObject *parent) :
+ QStandardItemModel(0, ColumnCount, parent)
+{
+ QStringList headers;
+ headers << StashDialog::tr("Name") << StashDialog::tr("Branch") << StashDialog::tr("Message");
+ setHorizontalHeaderLabels(headers);
+}
+
+void StashModel::setStashes(const QList<Stash> &stashes)
+{
+ m_stashes = stashes;
+ if (const int rows = rowCount())
+ removeRows(0, rows);
+ foreach(const Stash &s, stashes)
+ appendRow(stashModelRowItems(s));
+}
+
+// ---------- StashDialog
+StashDialog::StashDialog(QWidget *parent) :
+ QDialog(parent),
+ ui(new Ui::StashDialog),
+ m_model(new StashModel),
+ m_proxyModel(new QSortFilterProxyModel),
+ m_deleteAllButton(new QPushButton(tr("Delete all..."))),
+ m_deleteSelectionButton(new QPushButton(tr("Delete..."))),
+ m_showCurrentButton(new QPushButton(tr("Show"))),
+ m_restoreCurrentButton(new QPushButton(tr("Restore..."))),
+ m_restoreCurrentInBranchButton(new QPushButton(tr("Restore to branch..."))),
+ m_refreshButton(new QPushButton(tr("Refresh")))
+{
+ setWindowFlags(windowFlags() & ~Qt::WindowContextHelpButtonHint);
+ ui->setupUi(this);
+ // Buttons
+ ui->buttonBox->addButton(m_showCurrentButton, QDialogButtonBox::ActionRole);
+ connect(m_showCurrentButton, SIGNAL(clicked()), this, SLOT(showCurrent()));
+ ui->buttonBox->addButton(m_refreshButton, QDialogButtonBox::ActionRole);
+ connect(m_refreshButton, SIGNAL(clicked()), this, SLOT(forceRefresh()));
+ ui->buttonBox->addButton(m_restoreCurrentButton, QDialogButtonBox::ActionRole);
+ connect(m_restoreCurrentButton, SIGNAL(clicked()), this, SLOT(restoreCurrent()));
+ ui->buttonBox->addButton(m_restoreCurrentInBranchButton, QDialogButtonBox::ActionRole);
+ connect(m_restoreCurrentInBranchButton, SIGNAL(clicked()), this, SLOT(restoreCurrentInBranch()));
+ ui->buttonBox->addButton(m_deleteSelectionButton, QDialogButtonBox::ActionRole);
+ connect(m_deleteSelectionButton, SIGNAL(clicked()), this, SLOT(deleteSelection()));
+ ui->buttonBox->addButton(m_deleteAllButton, QDialogButtonBox::ActionRole);
+ connect(m_deleteAllButton, SIGNAL(clicked()), this, SLOT(deleteAll()));
+ // Models
+ m_proxyModel->setSourceModel(m_model);
+ m_proxyModel->setFilterKeyColumn(-1);
+ m_proxyModel->setFilterCaseSensitivity(Qt::CaseInsensitive);
+ ui->stashView->setModel(m_proxyModel);
+ ui->stashView->setSelectionMode(QAbstractItemView::MultiSelection);
+ connect(ui->filterLineEdit, SIGNAL(filterChanged(QString)), m_proxyModel, SLOT(setFilterFixedString(QString)));
+ connect(ui->stashView->selectionModel(), SIGNAL(currentRowChanged(QModelIndex,QModelIndex)),
+ this, SLOT(enableButtons()));
+ connect(ui->stashView->selectionModel(), SIGNAL(selectionChanged(QItemSelection,QItemSelection)),
+ this, SLOT(enableButtons()));
+ connect(ui->stashView, SIGNAL(doubleClicked(QModelIndex)), this, SLOT(showCurrent()));
+ ui->stashView->setFocus();
+}
+
+StashDialog::~StashDialog()
+{
+ delete ui;
+}
+
+void StashDialog::changeEvent(QEvent *e)
+{
+ QDialog::changeEvent(e);
+ switch (e->type()) {
+ case QEvent::LanguageChange:
+ ui->retranslateUi(this);
+ break;
+ default:
+ break;
+ }
+}
+
+void StashDialog::refresh(const QString &repository, bool force)
+{
+ if (m_repository == repository && !force)
+ return;
+ // Refresh
+ m_repository = repository;
+ if (m_repository.isEmpty()) {
+ ui->repositoryLabel->setText(tr("<No repository>"));
+ m_model->setStashes(QList<Stash>());
+ } else {
+ ui->repositoryLabel->setText(tr("Repository: %1").arg(repository));
+ QList<Stash> stashes;
+ gitClient()->synchronousStashList(m_repository, &stashes);
+ m_model->setStashes(stashes);
+ if (!stashes.isEmpty()) {
+ for(int c = 0; c < ColumnCount; c++)
+ ui->stashView->resizeColumnToContents(c);
+ }
+ }
+ enableButtons();
+}
+
+void StashDialog::deleteAll()
+{
+ const QString title = tr("Delete stashes");
+ if (!ask(title, tr("Do you want to delete all stashes?")))
+ return;
+ QString errorMessage;
+ if (gitClient()->synchronousStashRemove(m_repository, QString(), &errorMessage)) {
+ refresh(m_repository, true);
+ } else {
+ warning(title, errorMessage);
+ }
+}
+
+void StashDialog::deleteSelection()
+{
+ const QList<int> rows = selectedRows();
+ QTC_ASSERT(!rows.isEmpty(), return)
+ const QString title = tr("Delete stashes");
+ if (!ask(title, tr("Do you want to delete %n stash(es)?", 0, rows.size())))
+ return;
+ QString errorMessage;
+ QStringList errors;
+ // Delete in reverse order as stashes rotate
+ for (int r = rows.size() - 1; r >= 0; r--)
+ if (!gitClient()->synchronousStashRemove(m_repository, m_model->at(r).name, &errorMessage))
+ errors.push_back(errorMessage);
+ refresh(m_repository, true);
+ if (!errors.isEmpty())
+ warning(title, errors.join(QString(QLatin1Char('\n'))));
+}
+
+void StashDialog::showCurrent()
+{
+ const int index = currentRow();
+ QTC_ASSERT(index >= 0, return)
+ gitClient()->show(m_repository, m_model->at(index).name);
+}
+
+// Suggest Branch name to restore 'stash@{0}' -> 'stash0-date'
+static inline QString stashRestoreDefaultBranch(QString stash)
+{
+ stash.remove(QLatin1Char('{'));
+ stash.remove(QLatin1Char('}'));
+ stash.remove(QLatin1Char('@'));
+ stash += QLatin1Char('-');
+ stash += QDateTime::currentDateTime().toString(QLatin1String("yyMMddhhmmss"));
+ return stash;
+}
+
+// Return next stash id 'stash@{0}' -> 'stash@{1}'
+static inline QString nextStash(const QString &stash)
+{
+ const int openingBracePos = stash.indexOf(QLatin1Char('{'));
+ if (openingBracePos == -1)
+ return QString();
+ const int closingBracePos = stash.indexOf(QLatin1Char('}'), openingBracePos + 2);
+ if (closingBracePos == -1)
+ return QString();
+ bool ok;
+ const int n = stash.mid(openingBracePos + 1, closingBracePos - openingBracePos - 1).toInt(&ok);
+ if (!ok)
+ return QString();
+ QString rc = stash.left(openingBracePos + 1);
+ rc += QString::number(n + 1);
+ rc += QLatin1Char('}');
+ return rc;
+}
+
+StashDialog::ModifiedRepositoryAction StashDialog::promptModifiedRepository(const QString &stash)
+{
+ QMessageBox box(QMessageBox::Question,
+ tr("Repository modified"),
+ tr("%1 cannot be restored since the repository is modified.\n"
+ "You can choose between stashing the changes or discarding them.").arg(stash),
+ QMessageBox::Cancel, this);
+ QPushButton *stashButton = box.addButton(tr("Stash"), QMessageBox::AcceptRole);
+ QPushButton *discardButton = box.addButton(tr("Discard"), QMessageBox::AcceptRole);
+ box.exec();
+ const QAbstractButton *clickedButton = box.clickedButton();
+ if (clickedButton == stashButton)
+ return ModifiedRepositoryStash;
+ if (clickedButton == discardButton)
+ return ModifiedRepositoryDiscard;
+ return ModifiedRepositoryCancel;
+}
+
+// Prompt for restore: Make sure repository is unmodified,
+// prompt for a branch if desired or just ask to restore.
+// Note that the stash to be restored changes if the user
+// chooses to stash away modified repository.
+bool StashDialog::promptForRestore(QString *stash,
+ QString *branch /* = 0*/,
+ QString *errorMessage)
+{
+ const QString stashIn = *stash;
+ bool modifiedPromptShown = false;
+ switch (gitClient()->gitStatus(m_repository, false, 0, errorMessage)) {
+ case GitClient::StatusFailed:
+ return false;
+ case GitClient::StatusChanged: {
+ switch (promptModifiedRepository(*stash)) {
+ case ModifiedRepositoryCancel:
+ return false;
+ case ModifiedRepositoryStash:
+ if (gitClient()->synchronousStash(m_repository, QString(), GitClient::StashPromptDescription).isEmpty())
+ return false;
+ *stash = nextStash(*stash); // Our stash id to be restored changed
+ QTC_ASSERT(!stash->isEmpty(), return false)
+ break;
+ case ModifiedRepositoryDiscard:
+ if (!gitClient()->synchronousReset(m_repository))
+ return false;
+ break;
+ }
+ modifiedPromptShown = true;
+ }
+ break;
+ case GitClient::StatusUnchanged:
+ break;
+ }
+ // Prompt for branch or just ask.
+ if (branch) {
+ *branch = stashRestoreDefaultBranch(*stash);
+ if (!inputText(this, tr("Restore Stash to Branch"), tr("Branch:"), branch)
+ || branch->isEmpty())
+ return false;
+ } else {
+ if (!modifiedPromptShown && !ask(tr("Stash Restore"), tr("Would you like to restore %1?").arg(stashIn)))
+ return false;
+ }
+ return true;
+}
+
+static inline QString msgRestoreFailedTitle(const QString &stash)
+{
+ return StashDialog::tr("Error restoring %1").arg(stash);
+}
+
+void StashDialog::restoreCurrent()
+{
+ const int index = currentRow();
+ QTC_ASSERT(index >= 0, return)
+ QString errorMessage;
+ QString name = m_model->at(index).name;
+ // Make sure repository is not modified, restore. The command will
+ // output to window on success.
+ const bool success = promptForRestore(&name, 0, &errorMessage)
+ && gitClient()->synchronousStashRestore(m_repository, name, QString(), &errorMessage);
+ if (success) {
+ refresh(m_repository, true); // Might have stashed away local changes.
+ } else {
+ if (!errorMessage.isEmpty())
+ warning(msgRestoreFailedTitle(name), errorMessage);
+ }
+}
+
+void StashDialog::restoreCurrentInBranch()
+{
+ const int index = currentRow();
+ QTC_ASSERT(index >= 0, return)
+ QString errorMessage;
+ QString branch;
+ QString name = m_model->at(index).name;
+ const bool success = promptForRestore(&name, &branch, &errorMessage)
+ && gitClient()->synchronousStashRestore(m_repository, name, branch, &errorMessage);
+ if (success) {
+ refresh(m_repository, true); // git deletes the stash, unfortunately.
+ } else {
+ if (!errorMessage.isEmpty())
+ warning(msgRestoreFailedTitle(name), errorMessage);
+ }
+}
+
+int StashDialog::currentRow() const
+{
+ const QModelIndex proxyIndex = ui->stashView->currentIndex();
+ if (proxyIndex.isValid()) {
+ const QModelIndex index = m_proxyModel->mapToSource(proxyIndex);
+ if (index.isValid())
+ return index.row();
+ }
+ return -1;
+}
+
+QList<int> StashDialog::selectedRows() const
+{
+ QList<int> rc;
+ foreach(const QModelIndex &proxyIndex, ui->stashView->selectionModel()->selectedRows()) {
+ const QModelIndex index = m_proxyModel->mapToSource(proxyIndex);
+ if (index.isValid())
+ rc.push_back(index.row());
+ }
+ qSort(rc);
+ return rc;
+}
+
+void StashDialog::forceRefresh()
+{
+ refresh(m_repository, true);
+}
+
+void StashDialog::enableButtons()
+{
+ const bool hasStashes = m_model->rowCount();
+ const bool hasCurrentRow = hasStashes && currentRow() >= 0;
+ m_deleteAllButton->setEnabled(hasStashes);
+ m_showCurrentButton->setEnabled(hasCurrentRow);
+ m_restoreCurrentButton->setEnabled(hasCurrentRow);
+ m_restoreCurrentInBranchButton->setEnabled(hasCurrentRow);
+ const bool hasSelection = !ui->stashView->selectionModel()->selectedRows().isEmpty();
+ m_deleteSelectionButton->setEnabled(hasSelection);
+}
+
+void StashDialog::warning(const QString &title, const QString &what, const QString &details)
+{
+ QMessageBox msgBox(QMessageBox::Warning, title, what, QMessageBox::Ok, this);
+ if (!details.isEmpty())
+ msgBox.setDetailedText(details);
+ msgBox.exec();
+}
+
+bool StashDialog::ask(const QString &title, const QString &what, bool defaultButton)
+{
+ return QMessageBox::question(this, title, what, QMessageBox::Yes|QMessageBox::No,
+ defaultButton ? QMessageBox::Yes : QMessageBox::No) == QMessageBox::Yes;
+}
+
+} // namespace Internal
+} // namespace Git