kde-workspace/dolphin/plugins/svn/fileviewsvnplugin.cpp

432 lines
15 KiB
C++
Raw Normal View History

/***************************************************************************
* Copyright (C) 2009-2011 by Peter Penz <peter.penz19@gmail.com> *
* *
* This program is free software; you can redistribute it and/or modify *
* it under the terms of the GNU General Public License as published by *
* the Free Software Foundation; either version 2 of the License, or *
* (at your option) any later version. *
* *
* This program is distributed in the hope that it will be useful, *
* but WITHOUT ANY WARRANTY; without even the implied warranty of *
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the *
* GNU General Public License for more details. *
* *
* You should have received a copy of the GNU General Public License *
* along with this program; if not, write to the *
* Free Software Foundation, Inc., *
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA *
***************************************************************************/
#include "fileviewsvnplugin.h"
#include "fileviewsvnpluginsettings.h"
#include <kaction.h>
#include <kdialog.h>
#include <kfileitem.h>
#include <kicon.h>
#include <klocale.h>
#include <krun.h>
#include <kshell.h>
#include <kvbox.h>
#include <kpluginfactory.h>
#include <kpluginloader.h>
#include <QDir>
#include <QLabel>
#include <QPlainTextEdit>
#include <QProcess>
#include <QString>
#include <QStringList>
#include <QTextStream>
K_PLUGIN_FACTORY(FileViewSvnPluginFactory, registerPlugin<FileViewSvnPlugin>();)
K_EXPORT_PLUGIN(FileViewSvnPluginFactory("fileviewsvnplugin"))
FileViewSvnPlugin::FileViewSvnPlugin(QObject* parent, const QList<QVariant>& args) :
KVersionControlPlugin(parent),
m_pendingOperation(false),
m_versionInfoHash(),
m_updateAction(0),
m_commitAction(0),
m_addAction(0),
m_removeAction(0),
m_showUpdatesAction(0),
m_command(),
m_arguments(),
m_errorMsg(),
m_operationCompletedMsg(),
m_contextDir(),
m_contextItems(),
m_process(),
m_tempFile()
{
Q_UNUSED(args);
m_updateAction = new KAction(this);
m_updateAction->setIcon(KIcon("view-refresh"));
m_updateAction->setText(i18nc("@item:inmenu", "SVN Update"));
connect(m_updateAction, SIGNAL(triggered()),
this, SLOT(updateFiles()));
m_commitAction = new KAction(this);
m_commitAction->setIcon(KIcon("svn-commit"));
m_commitAction->setText(i18nc("@item:inmenu", "SVN Commit..."));
connect(m_commitAction, SIGNAL(triggered()),
this, SLOT(commitFiles()));
m_addAction = new KAction(this);
m_addAction->setIcon(KIcon("list-add"));
m_addAction->setText(i18nc("@item:inmenu", "SVN Add"));
connect(m_addAction, SIGNAL(triggered()),
this, SLOT(addFiles()));
m_removeAction = new KAction(this);
m_removeAction->setIcon(KIcon("list-remove"));
m_removeAction->setText(i18nc("@item:inmenu", "SVN Delete"));
connect(m_removeAction, SIGNAL(triggered()),
this, SLOT(removeFiles()));
m_showUpdatesAction = new KAction(this);
m_showUpdatesAction->setCheckable(true);
m_showUpdatesAction->setText(i18nc("@item:inmenu", "Show SVN Updates"));
m_showUpdatesAction->setChecked(FileViewSvnPluginSettings::showUpdates());
connect(m_showUpdatesAction, SIGNAL(toggled(bool)),
this, SLOT(slotShowUpdatesToggled(bool)));
connect(this, SIGNAL(setShowUpdatesChecked(bool)),
m_showUpdatesAction, SLOT(setChecked(bool)));
connect(&m_process, SIGNAL(finished(int, QProcess::ExitStatus)),
this, SLOT(slotOperationCompleted(int, QProcess::ExitStatus)));
connect(&m_process, SIGNAL(error(QProcess::ProcessError)),
this, SLOT(slotOperationError()));
}
FileViewSvnPlugin::~FileViewSvnPlugin()
{
}
QString FileViewSvnPlugin::fileName() const
{
return QLatin1String(".svn");
}
bool FileViewSvnPlugin::beginRetrieval(const QString& directory)
{
Q_ASSERT(directory.endsWith(QLatin1Char('/')));
// Clear all entries for this directory including the entries
// for sub directories
QMutableHashIterator<QString, ItemVersion> it(m_versionInfoHash);
while (it.hasNext()) {
it.next();
if (it.key().startsWith(directory)) {
it.remove();
}
}
QStringList arguments;
arguments << QLatin1String("status");
if (FileViewSvnPluginSettings::showUpdates()) {
arguments << QLatin1String("--show-updates");
}
arguments << QLatin1String("--no-ignore") << directory;
QProcess process;
process.start(QLatin1String("svn"), arguments);
while (process.waitForReadyRead()) {
char buffer[1024];
while (process.readLine(buffer, sizeof(buffer)) > 0) {
ItemVersion version = NormalVersion;
QString filePath(buffer);
switch (buffer[0]) {
case 'I':
case '?': version = UnversionedVersion; break;
case 'M': version = LocallyModifiedVersion; break;
case 'A': version = AddedVersion; break;
case 'D': version = RemovedVersion; break;
case 'C': version = ConflictingVersion; break;
default:
if (filePath.contains('*')) {
version = UpdateRequiredVersion;
}
break;
}
// Only values with a different version as 'NormalVersion'
// are added to the hash table. If a value is not in the
// hash table, it is automatically defined as 'NormalVersion'
// (see FileViewSvnPlugin::itemVersion()).
if (version != NormalVersion) {
int pos = filePath.indexOf('/');
const int length = filePath.length() - pos - 1;
filePath = filePath.mid(pos, length);
if (!filePath.isEmpty()) {
m_versionInfoHash.insert(filePath, version);
}
}
}
}
if ((process.exitCode() != 0 || process.exitStatus() != QProcess::NormalExit)) {
if (FileViewSvnPluginSettings::showUpdates()) {
// Network update failed. Unset ShowUpdates option, which triggers a refresh
emit infoMessage(i18nc("@info:status", "SVN status update failed. Disabling Option "
"\"Show SVN Updates\"."));
emit setShowUpdatesChecked(false);
// this is no fail, we just try again differently
// furthermore returning false shows an error message that would override our info
return true;
} else {
return false;
}
}
return true;
}
void FileViewSvnPlugin::endRetrieval()
{
}
KVersionControlPlugin::ItemVersion FileViewSvnPlugin::itemVersion(const KFileItem& item) const
{
const QString itemUrl = item.localPath();
if (m_versionInfoHash.contains(itemUrl)) {
return m_versionInfoHash.value(itemUrl);
}
if (!item.isDir()) {
// files that have not been listed by 'svn status' (= m_versionInfoHash)
// are under version control per definition
return NormalVersion;
}
// The item is a directory. Check whether an item listed by 'svn status' (= m_versionInfoHash)
// is part of this directory. In this case a local modification should be indicated in the
// directory already.
QHash<QString, ItemVersion>::const_iterator it = m_versionInfoHash.constBegin();
while (it != m_versionInfoHash.constEnd()) {
if (it.key().startsWith(itemUrl)) {
const ItemVersion version = m_versionInfoHash.value(it.key());
if (version == LocallyModifiedVersion) {
return LocallyModifiedVersion;
}
}
++it;
}
return NormalVersion;
}
QList<QAction*> FileViewSvnPlugin::actions(const KFileItemList& items) const
{
if (items.count() == 1 && items.first().isDir()) {
const QString directory = items.first().url().path(KUrl::AddTrailingSlash);
return directoryActions(directory);
}
foreach (const KFileItem& item, items) {
m_contextItems.append(item);
}
m_contextDir.clear();
const bool noPendingOperation = !m_pendingOperation;
if (noPendingOperation) {
// iterate all items and check the version version to know which
// actions can be enabled
const int itemsCount = items.count();
int versionedCount = 0;
int editingCount = 0;
foreach (const KFileItem& item, items) {
const ItemVersion version = itemVersion(item);
if (version != UnversionedVersion) {
++versionedCount;
}
switch (version) {
case LocallyModifiedVersion:
case ConflictingVersion:
++editingCount;
break;
default:
break;
}
}
m_commitAction->setEnabled(editingCount > 0);
m_addAction->setEnabled(versionedCount == 0);
m_removeAction->setEnabled(versionedCount == itemsCount);
} else {
m_commitAction->setEnabled(false);
m_addAction->setEnabled(false);
m_removeAction->setEnabled(false);
}
m_updateAction->setEnabled(noPendingOperation);
QList<QAction*> actions;
actions.append(m_updateAction);
actions.append(m_commitAction);
actions.append(m_addAction);
actions.append(m_removeAction);
actions.append(m_showUpdatesAction);
return actions;
}
void FileViewSvnPlugin::updateFiles()
{
execSvnCommand("update", QStringList(),
i18nc("@info:status", "Updating SVN repository..."),
i18nc("@info:status", "Update of SVN repository failed."),
i18nc("@info:status", "Updated SVN repository."));
}
void FileViewSvnPlugin::commitFiles()
{
KDialog dialog(0, Qt::Dialog);
KVBox* box = new KVBox(&dialog);
new QLabel(i18nc("@label", "Description:"), box);
QPlainTextEdit* editor = new QPlainTextEdit(box);
dialog.setMainWidget(box);
dialog.setCaption(i18nc("@title:window", "SVN Commit"));
dialog.setButtons(KDialog::Ok | KDialog::Cancel);
dialog.setDefaultButton(KDialog::Ok);
dialog.setButtonText(KDialog::Ok, i18nc("@action:button", "Commit"));
KConfigGroup dialogConfig(KSharedConfig::openConfig("dolphinrc"),
"SvnCommitDialog");
dialog.restoreDialogSize(dialogConfig);
if (dialog.exec() == QDialog::Accepted) {
// Write the commit description into a temporary file, so
// that it can be read by the command "svn commit -F". The temporary
// file must stay alive until slotOperationCompleted() is invoked and will
// be destroyed when the version plugin is destructed.
if (!m_tempFile.open()) {
emit errorMessage(i18nc("@info:status", "Commit of SVN changes failed."));
return;
}
QTextStream out(&m_tempFile);
const QString fileName = m_tempFile.fileName();
out << editor->toPlainText();
m_tempFile.close();
QStringList arguments;
arguments << "-F" << fileName;
execSvnCommand("commit", arguments,
i18nc("@info:status", "Committing SVN changes..."),
i18nc("@info:status", "Commit of SVN changes failed."),
i18nc("@info:status", "Committed SVN changes."));
}
dialog.saveDialogSize(dialogConfig, KConfigBase::Persistent);
}
void FileViewSvnPlugin::addFiles()
{
execSvnCommand(QLatin1String("add"), QStringList(),
i18nc("@info:status", "Adding files to SVN repository..."),
i18nc("@info:status", "Adding of files to SVN repository failed."),
i18nc("@info:status", "Added files to SVN repository."));
}
void FileViewSvnPlugin::removeFiles()
{
execSvnCommand(QLatin1String("remove"), QStringList(),
i18nc("@info:status", "Removing files from SVN repository..."),
i18nc("@info:status", "Removing of files from SVN repository failed."),
i18nc("@info:status", "Removed files from SVN repository."));
}
void FileViewSvnPlugin::slotOperationCompleted(int exitCode, QProcess::ExitStatus exitStatus)
{
m_pendingOperation = false;
if ((exitStatus != QProcess::NormalExit) || (exitCode != 0)) {
emit errorMessage(m_errorMsg);
} else if (m_contextItems.isEmpty()) {
emit operationCompletedMessage(m_operationCompletedMsg);
emit itemVersionsChanged();
} else {
startSvnCommandProcess();
}
}
void FileViewSvnPlugin::slotOperationError()
{
// don't do any operation on other items anymore
m_contextItems.clear();
m_pendingOperation = false;
emit errorMessage(m_errorMsg);
}
void FileViewSvnPlugin::slotShowUpdatesToggled(bool checked)
{
FileViewSvnPluginSettings* settings = FileViewSvnPluginSettings::self();
Q_ASSERT(settings != 0);
settings->setShowUpdates(checked);
settings->writeConfig();
emit itemVersionsChanged();
}
void FileViewSvnPlugin::execSvnCommand(const QString& svnCommand,
const QStringList& arguments,
const QString& infoMsg,
const QString& errorMsg,
const QString& operationCompletedMsg)
{
emit infoMessage(infoMsg);
m_command = svnCommand;
m_arguments = arguments;
m_errorMsg = errorMsg;
m_operationCompletedMsg = operationCompletedMsg;
startSvnCommandProcess();
}
void FileViewSvnPlugin::startSvnCommandProcess()
{
Q_ASSERT(m_process.state() == QProcess::NotRunning);
m_pendingOperation = true;
const QString program(QLatin1String("svn"));
QStringList arguments;
arguments << m_command << m_arguments;
if (!m_contextDir.isEmpty()) {
arguments << m_contextDir;
m_contextDir.clear();
} else {
const KFileItem item = m_contextItems.takeLast();
arguments << item.localPath();
// the remaining items of m_contextItems will be executed
// after the process has finished (see slotOperationFinished())
}
m_process.start(program, arguments);
}
QList<QAction*> FileViewSvnPlugin::directoryActions(const QString& directory) const
{
m_contextDir = directory;
m_contextItems.clear();
// Only enable the SVN actions if no SVN commands are
// executed currently (see slotOperationCompleted() and
// startSvnCommandProcess()).
const bool enabled = !m_pendingOperation;
m_updateAction->setEnabled(enabled);
m_commitAction->setEnabled(enabled);
QList<QAction*> actions;
actions.append(m_updateAction);
actions.append(m_commitAction);
actions.append(m_showUpdatesAction);
return actions;
}
#include "moc_fileviewsvnplugin.cpp"