2010-03-12 15:54:09 +01:00
|
|
|
/**************************************************************************
|
|
|
|
|
**
|
|
|
|
|
** This file is part of Qt Creator
|
|
|
|
|
**
|
|
|
|
|
** Copyright (c) 2010 Nokia Corporation and/or its subsidiary(-ies).
|
|
|
|
|
**
|
|
|
|
|
** Contact: Nokia Corporation (qt-info@nokia.com)
|
|
|
|
|
**
|
|
|
|
|
** 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 "cleandialog.h"
|
|
|
|
|
#include "ui_cleandialog.h"
|
|
|
|
|
#include "vcsbaseoutputwindow.h"
|
|
|
|
|
|
|
|
|
|
#include <coreplugin/editormanager/editormanager.h>
|
|
|
|
|
#include <coreplugin/icore.h>
|
|
|
|
|
#include <coreplugin/progressmanager/progressmanager.h>
|
|
|
|
|
|
|
|
|
|
#include <QtGui/QStandardItemModel>
|
|
|
|
|
#include <QtGui/QMessageBox>
|
|
|
|
|
#include <QtGui/QApplication>
|
|
|
|
|
#include <QtGui/QStyle>
|
|
|
|
|
#include <QtGui/QIcon>
|
|
|
|
|
|
|
|
|
|
#include <QtCore/QDir>
|
|
|
|
|
#include <QtCore/QFile>
|
|
|
|
|
#include <QtCore/QFileInfo>
|
|
|
|
|
#include <QtCore/QDebug>
|
|
|
|
|
#include <QtCore/QDateTime>
|
|
|
|
|
#include <QtCore/QFuture>
|
|
|
|
|
#include <QtCore/QtConcurrentRun>
|
|
|
|
|
|
|
|
|
|
enum { nameColumn, columnCount };
|
|
|
|
|
enum { fileNameRole = Qt::UserRole, isDirectoryRole = Qt::UserRole + 1 };
|
|
|
|
|
|
|
|
|
|
// Helper for recursively removing files.
|
|
|
|
|
static void removeFileRecursion(const QFileInfo &f, QString *errorMessage)
|
|
|
|
|
{
|
|
|
|
|
// The version control system might list files/directory in arbitrary
|
|
|
|
|
// order, causing files to be removed from parent directories.
|
|
|
|
|
if (!f.exists())
|
|
|
|
|
return;
|
|
|
|
|
if (f.isDir()) {
|
|
|
|
|
const QDir dir(f.absoluteFilePath());
|
|
|
|
|
foreach(const QFileInfo &fi, dir.entryInfoList(QDir::AllEntries|QDir::NoDotAndDotDot|QDir::Hidden))
|
|
|
|
|
removeFileRecursion(fi, errorMessage);
|
|
|
|
|
QDir parent = f.absoluteDir();
|
|
|
|
|
if (!parent.rmdir(f.fileName()))
|
|
|
|
|
errorMessage->append(VCSBase::CleanDialog::tr("The directory %1 could not be deleted.").arg(f.absoluteFilePath()));
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
if (!QFile::remove(f.absoluteFilePath())) {
|
|
|
|
|
if (!errorMessage->isEmpty())
|
|
|
|
|
errorMessage->append(QLatin1Char('\n'));
|
|
|
|
|
errorMessage->append(VCSBase::CleanDialog::tr("The file %1 could not be deleted.").arg(f.absoluteFilePath()));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
namespace VCSBase {
|
|
|
|
|
|
|
|
|
|
// A QFuture task for cleaning files in the background.
|
|
|
|
|
// Emits error signal if not all files can be deleted.
|
|
|
|
|
class CleanFilesTask : public QObject {
|
|
|
|
|
Q_OBJECT
|
|
|
|
|
public:
|
|
|
|
|
explicit CleanFilesTask(const QString &repository, const QStringList &files);
|
|
|
|
|
|
|
|
|
|
void run();
|
|
|
|
|
|
|
|
|
|
signals:
|
|
|
|
|
void error(const QString &e);
|
|
|
|
|
|
|
|
|
|
private:
|
|
|
|
|
const QString m_repository;
|
|
|
|
|
const QStringList m_files;
|
|
|
|
|
|
|
|
|
|
QString m_errorMessage;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
CleanFilesTask::CleanFilesTask(const QString &repository, const QStringList &files) :
|
|
|
|
|
m_repository(repository), m_files(files)
|
|
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void CleanFilesTask::run()
|
|
|
|
|
{
|
|
|
|
|
foreach(const QString &name, m_files)
|
|
|
|
|
removeFileRecursion(QFileInfo(name), &m_errorMessage);
|
|
|
|
|
if (!m_errorMessage.isEmpty()) {
|
|
|
|
|
// Format and emit error.
|
|
|
|
|
const QString msg = CleanDialog::tr("There were errors when cleaning the repository %1:").arg(m_repository);
|
|
|
|
|
m_errorMessage.insert(0, QLatin1Char('\n'));
|
|
|
|
|
m_errorMessage.insert(0, msg);
|
|
|
|
|
emit error(m_errorMessage);
|
|
|
|
|
}
|
|
|
|
|
// Run in background, need to delete ourselves
|
|
|
|
|
this->deleteLater();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// ---------------- CleanDialogPrivate ----------------
|
|
|
|
|
struct CleanDialogPrivate {
|
|
|
|
|
CleanDialogPrivate();
|
|
|
|
|
|
|
|
|
|
Ui::CleanDialog ui;
|
|
|
|
|
QStandardItemModel *m_filesModel;
|
|
|
|
|
QString m_workingDirectory;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
CleanDialogPrivate::CleanDialogPrivate() : m_filesModel(new QStandardItemModel(0, columnCount))
|
|
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
CleanDialog::CleanDialog(QWidget *parent) :
|
|
|
|
|
QDialog(parent),
|
|
|
|
|
d(new CleanDialogPrivate)
|
|
|
|
|
{
|
|
|
|
|
setModal(true);
|
|
|
|
|
setWindowFlags(windowFlags() & ~Qt::WindowContextHelpButtonHint);
|
|
|
|
|
|
|
|
|
|
d->ui.setupUi(this);
|
|
|
|
|
d->ui.buttonBox->addButton(tr("Delete..."), QDialogButtonBox::AcceptRole);
|
|
|
|
|
|
|
|
|
|
d->m_filesModel->setHorizontalHeaderLabels(QStringList(tr("Name")));
|
|
|
|
|
d->ui.filesTreeView->setModel(d->m_filesModel);
|
|
|
|
|
d->ui.filesTreeView->setUniformRowHeights(true);
|
|
|
|
|
d->ui.filesTreeView->setSelectionMode(QAbstractItemView::NoSelection);
|
|
|
|
|
d->ui.filesTreeView->setAllColumnsShowFocus(true);
|
|
|
|
|
d->ui.filesTreeView->setRootIsDecorated(false);
|
|
|
|
|
connect(d->ui.filesTreeView, SIGNAL(doubleClicked(QModelIndex)),
|
|
|
|
|
this, SLOT(slotDoubleClicked(QModelIndex)));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
CleanDialog::~CleanDialog()
|
|
|
|
|
{
|
|
|
|
|
delete d;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void CleanDialog::setFileList(const QString &workingDirectory, const QStringList &l)
|
|
|
|
|
{
|
|
|
|
|
d->m_workingDirectory = workingDirectory;
|
|
|
|
|
d->ui.groupBox->setTitle(tr("Repository: %1").arg(workingDirectory));
|
|
|
|
|
if (const int oldRowCount = d->m_filesModel->rowCount())
|
|
|
|
|
d->m_filesModel->removeRows(0, oldRowCount);
|
|
|
|
|
|
|
|
|
|
QStyle *style = QApplication::style();
|
|
|
|
|
const QIcon folderIcon = style->standardIcon(QStyle::SP_DirIcon);
|
|
|
|
|
const QIcon fileIcon = style->standardIcon(QStyle::SP_FileIcon);
|
|
|
|
|
const QString diffSuffix = QLatin1String(".diff");
|
|
|
|
|
const QString patchSuffix = QLatin1String(".patch");
|
2010-03-12 15:59:50 +01:00
|
|
|
const QString proUserSuffix = QLatin1String(".pro.user");
|
2010-03-12 15:54:09 +01:00
|
|
|
const QChar slash = QLatin1Char('/');
|
2010-03-12 15:59:50 +01:00
|
|
|
// Do not initially check patches or 'pro.user' files for deletion.
|
2010-03-12 15:54:09 +01:00
|
|
|
foreach(const QString &fileName, l) {
|
|
|
|
|
const QFileInfo fi(workingDirectory + slash + fileName);
|
|
|
|
|
const bool isDir = fi.isDir();
|
|
|
|
|
QStandardItem *nameItem = new QStandardItem(QDir::toNativeSeparators(fileName));
|
|
|
|
|
nameItem->setFlags(Qt::ItemIsUserCheckable|Qt::ItemIsEnabled);
|
|
|
|
|
nameItem->setIcon(isDir ? folderIcon : fileIcon);
|
2010-03-12 15:59:50 +01:00
|
|
|
const bool saveFile = !isDir && (fileName.endsWith(diffSuffix)
|
|
|
|
|
|| fileName.endsWith(patchSuffix)
|
|
|
|
|
|| fileName.endsWith(proUserSuffix));
|
2010-03-12 15:54:09 +01:00
|
|
|
nameItem->setCheckable(true);
|
2010-03-12 15:59:50 +01:00
|
|
|
nameItem->setCheckState(saveFile ? Qt::Unchecked : Qt::Checked);
|
2010-03-12 15:54:09 +01:00
|
|
|
nameItem->setData(QVariant(fi.absoluteFilePath()), fileNameRole);
|
|
|
|
|
nameItem->setData(QVariant(isDir), isDirectoryRole);
|
|
|
|
|
// Tooltip with size information
|
|
|
|
|
if (fi.isFile()) {
|
|
|
|
|
const QString lastModified = fi.lastModified().toString(Qt::DefaultLocaleShortDate);
|
|
|
|
|
nameItem->setToolTip(tr("%1 bytes, last modified %2")
|
|
|
|
|
.arg(fi.size()).arg(lastModified));
|
|
|
|
|
}
|
|
|
|
|
d->m_filesModel->appendRow(nameItem);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for (int c = 0; c < d->m_filesModel->columnCount(); c++)
|
|
|
|
|
d->ui.filesTreeView->resizeColumnToContents(c);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QStringList CleanDialog::checkedFiles() const
|
|
|
|
|
{
|
|
|
|
|
QStringList rc;
|
|
|
|
|
if (const int rowCount = d->m_filesModel->rowCount()) {
|
|
|
|
|
for (int r = 0; r < rowCount; r++) {
|
|
|
|
|
const QStandardItem *item = d->m_filesModel->item(r, 0);
|
|
|
|
|
if (item->checkState() == Qt::Checked)
|
|
|
|
|
rc.push_back(item->data(fileNameRole).toString());
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return rc;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void CleanDialog::accept()
|
|
|
|
|
{
|
|
|
|
|
if (promptToDelete())
|
|
|
|
|
QDialog::accept();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool CleanDialog::promptToDelete()
|
|
|
|
|
{
|
|
|
|
|
// Prompt the user and delete files
|
|
|
|
|
const QStringList selectedFiles = checkedFiles();
|
|
|
|
|
if (selectedFiles.isEmpty())
|
|
|
|
|
return true;
|
|
|
|
|
|
|
|
|
|
if (QMessageBox::question(this, tr("Delete"),
|
|
|
|
|
tr("Do you want to delete %n files?", 0, selectedFiles.size()),
|
|
|
|
|
QMessageBox::Yes|QMessageBox::No, QMessageBox::Yes) != QMessageBox::Yes)
|
|
|
|
|
return false;
|
|
|
|
|
|
|
|
|
|
// Remove in background
|
|
|
|
|
CleanFilesTask *cleanTask = new CleanFilesTask(d->m_workingDirectory, selectedFiles);
|
|
|
|
|
connect(cleanTask, SIGNAL(error(QString)),
|
|
|
|
|
VCSBase::VCSBaseOutputWindow::instance(), SLOT(appendSilently(QString)),
|
|
|
|
|
Qt::QueuedConnection);
|
|
|
|
|
|
|
|
|
|
QFuture<void> task = QtConcurrent::run(cleanTask, &CleanFilesTask::run);
|
|
|
|
|
const QString taskName = tr("Cleaning %1").arg(d->m_workingDirectory);
|
|
|
|
|
Core::ICore::instance()->progressManager()->addTask(task, taskName,
|
|
|
|
|
QLatin1String("VCSBase.cleanRepository"));
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void CleanDialog::slotDoubleClicked(const QModelIndex &index)
|
|
|
|
|
{
|
|
|
|
|
// Open file on doubleclick
|
|
|
|
|
if (const QStandardItem *item = d->m_filesModel->itemFromIndex(index))
|
|
|
|
|
if (!item->data(isDirectoryRole).toBool()) {
|
|
|
|
|
const QString fname = item->data(fileNameRole).toString();
|
|
|
|
|
Core::EditorManager::instance()->openEditor(fname);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void CleanDialog::changeEvent(QEvent *e)
|
|
|
|
|
{
|
|
|
|
|
QDialog::changeEvent(e);
|
|
|
|
|
switch (e->type()) {
|
|
|
|
|
case QEvent::LanguageChange:
|
|
|
|
|
d->ui.retranslateUi(this);
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
} // namespace VCSBase
|
|
|
|
|
|
|
|
|
|
#include "cleandialog.moc"
|