2009-02-25 09:15:00 +01:00
|
|
|
/**************************************************************************
|
2008-12-02 12:01:29 +01:00
|
|
|
**
|
|
|
|
|
** This file is part of Qt Creator
|
|
|
|
|
**
|
2010-03-05 11:25:49 +01:00
|
|
|
** Copyright (c) 2010 Nokia Corporation and/or its subsidiary(-ies).
|
2008-12-02 12:01:29 +01:00
|
|
|
**
|
2009-06-17 00:01:27 +10:00
|
|
|
** Contact: Nokia Corporation (qt-info@nokia.com)
|
2008-12-02 12:01:29 +01:00
|
|
|
**
|
2009-02-25 09:15:00 +01:00
|
|
|
** Commercial Usage
|
2008-12-02 14:17:16 +01:00
|
|
|
**
|
2009-02-25 09:15:00 +01:00
|
|
|
** 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.
|
2008-12-02 14:17:16 +01:00
|
|
|
**
|
2009-02-25 09:15:00 +01:00
|
|
|
** GNU Lesser General Public License Usage
|
2008-12-02 14:17:16 +01:00
|
|
|
**
|
2009-02-25 09:15:00 +01:00
|
|
|
** 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.
|
2008-12-02 14:17:16 +01:00
|
|
|
**
|
2009-02-25 09:15:00 +01:00
|
|
|
** If you are unsure which license is appropriate for your use, please
|
2009-08-14 09:30:56 +02:00
|
|
|
** contact the sales department at http://qt.nokia.com/contact.
|
2008-12-02 12:01:29 +01:00
|
|
|
**
|
2009-02-25 09:15:00 +01:00
|
|
|
**************************************************************************/
|
2008-12-02 15:08:31 +01:00
|
|
|
|
2008-12-02 12:01:29 +01:00
|
|
|
#include "vcsmanager.h"
|
|
|
|
|
#include "iversioncontrol.h"
|
2009-12-21 11:08:20 +01:00
|
|
|
#include "icore.h"
|
|
|
|
|
#include "filemanager.h"
|
2008-12-02 12:01:29 +01:00
|
|
|
|
|
|
|
|
#include <extensionsystem/pluginmanager.h>
|
2010-01-11 15:22:17 +01:00
|
|
|
#include <utils/qtcassert.h>
|
2008-12-02 12:01:29 +01:00
|
|
|
|
|
|
|
|
#include <QtCore/QString>
|
|
|
|
|
#include <QtCore/QList>
|
|
|
|
|
#include <QtCore/QMap>
|
|
|
|
|
#include <QtCore/QCoreApplication>
|
|
|
|
|
|
|
|
|
|
#include <QtCore/QDebug>
|
|
|
|
|
#include <QtCore/QFileInfo>
|
|
|
|
|
#include <QtGui/QMessageBox>
|
|
|
|
|
|
2008-12-03 15:04:51 +01:00
|
|
|
enum { debug = 0 };
|
|
|
|
|
|
2008-12-02 12:01:29 +01:00
|
|
|
namespace Core {
|
|
|
|
|
|
2008-12-03 15:04:51 +01:00
|
|
|
typedef QList<IVersionControl *> VersionControlList;
|
2009-12-08 14:26:41 +01:00
|
|
|
typedef QMap<QString, IVersionControl *> VersionControlCache;
|
2008-12-03 15:04:51 +01:00
|
|
|
|
|
|
|
|
static inline VersionControlList allVersionControls()
|
|
|
|
|
{
|
|
|
|
|
return ExtensionSystem::PluginManager::instance()->getObjects<IVersionControl>();
|
|
|
|
|
}
|
|
|
|
|
|
2009-12-08 14:26:41 +01:00
|
|
|
// ---- VCSManagerPrivate:
|
|
|
|
|
// Maintains a cache of top-level directory->version control.
|
|
|
|
|
|
2008-12-02 12:01:29 +01:00
|
|
|
struct VCSManagerPrivate {
|
2009-12-08 14:26:41 +01:00
|
|
|
VersionControlCache m_cachedMatches;
|
2008-12-02 12:01:29 +01:00
|
|
|
};
|
|
|
|
|
|
2009-11-11 14:32:54 +01:00
|
|
|
VCSManager::VCSManager(QObject *parent) :
|
|
|
|
|
QObject(parent),
|
2008-12-02 12:01:29 +01:00
|
|
|
m_d(new VCSManagerPrivate)
|
|
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
VCSManager::~VCSManager()
|
|
|
|
|
{
|
|
|
|
|
delete m_d;
|
|
|
|
|
}
|
|
|
|
|
|
2009-11-11 14:32:54 +01:00
|
|
|
void VCSManager::extensionsInitialized()
|
|
|
|
|
{
|
|
|
|
|
// Change signal connections
|
2009-12-21 11:08:20 +01:00
|
|
|
FileManager *fileManager = ICore::instance()->fileManager();
|
2009-11-11 14:32:54 +01:00
|
|
|
foreach (IVersionControl *versionControl, allVersionControls()) {
|
|
|
|
|
connect(versionControl, SIGNAL(filesChanged(QStringList)),
|
2009-12-21 11:08:20 +01:00
|
|
|
fileManager, SIGNAL(filesChangedInternally(QStringList)));
|
2009-11-11 14:32:54 +01:00
|
|
|
connect(versionControl, SIGNAL(repositoryChanged(QString)),
|
|
|
|
|
this, SIGNAL(repositoryChanged(QString)));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2009-12-08 14:26:41 +01:00
|
|
|
IVersionControl* VCSManager::findVersionControlForDirectory(const QString &directory,
|
|
|
|
|
QString *topLevelDirectory)
|
2008-12-02 12:01:29 +01:00
|
|
|
{
|
2009-12-08 14:26:41 +01:00
|
|
|
typedef VersionControlCache::const_iterator VersionControlCacheConstIterator;
|
2010-05-20 16:24:39 +02:00
|
|
|
|
|
|
|
|
if (debug) {
|
|
|
|
|
qDebug(">findVersionControlForDirectory %s topLevelPtr %d",
|
|
|
|
|
qPrintable(directory), (topLevelDirectory != 0));
|
|
|
|
|
if (debug > 1) {
|
|
|
|
|
const VersionControlCacheConstIterator cend = m_d->m_cachedMatches.constEnd();
|
|
|
|
|
for (VersionControlCacheConstIterator it = m_d->m_cachedMatches.constBegin(); it != cend; ++it)
|
|
|
|
|
qDebug("Cache %s -> '%s'", qPrintable(it.key()), qPrintable(it.value()->displayName()));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
QTC_ASSERT(!directory.isEmpty(), return 0);
|
|
|
|
|
|
2009-12-08 14:26:41 +01:00
|
|
|
const VersionControlCacheConstIterator cacheEnd = m_d->m_cachedMatches.constEnd();
|
|
|
|
|
|
|
|
|
|
if (topLevelDirectory)
|
|
|
|
|
topLevelDirectory->clear();
|
|
|
|
|
|
|
|
|
|
// First check if the directory has an entry, meaning it is a top level
|
|
|
|
|
const VersionControlCacheConstIterator fullPathIt = m_d->m_cachedMatches.constFind(directory);
|
|
|
|
|
if (fullPathIt != cacheEnd) {
|
|
|
|
|
if (topLevelDirectory)
|
|
|
|
|
*topLevelDirectory = directory;
|
2010-05-20 16:24:39 +02:00
|
|
|
if (debug)
|
|
|
|
|
qDebug("<findVersionControlForDirectory: full cache match for VCS '%s'", qPrintable(fullPathIt.value()->displayName()));
|
2009-12-08 14:26:41 +01:00
|
|
|
return fullPathIt.value();
|
2008-12-02 12:01:29 +01:00
|
|
|
}
|
|
|
|
|
|
2010-05-20 16:24:39 +02:00
|
|
|
// Split the path, trying to find the matching repository. We start from the reverse
|
|
|
|
|
// in order to detected nested repositories correctly (say, a git checkout under SVN).
|
|
|
|
|
// Note that detection of a nested version control will still fail if the
|
|
|
|
|
// above-located version control is detected and entered into the cache first.
|
|
|
|
|
// The nested one can then no longer be found due to the splitting of the paths.
|
|
|
|
|
int pos = directory.size() - 1;
|
2008-12-03 15:04:51 +01:00
|
|
|
const QChar slash = QLatin1Char('/');
|
2008-12-09 11:07:24 +01:00
|
|
|
while (true) {
|
2010-05-20 16:24:39 +02:00
|
|
|
const int index = directory.lastIndexOf(slash, pos);
|
|
|
|
|
if (index <= 0) // Stop at '/' or not found
|
2008-12-02 12:01:29 +01:00
|
|
|
break;
|
2008-12-03 15:04:51 +01:00
|
|
|
const QString directoryPart = directory.left(index);
|
2009-12-08 14:26:41 +01:00
|
|
|
const VersionControlCacheConstIterator it = m_d->m_cachedMatches.constFind(directoryPart);
|
|
|
|
|
if (it != cacheEnd) {
|
|
|
|
|
if (topLevelDirectory)
|
|
|
|
|
*topLevelDirectory = it.key();
|
2010-05-20 16:24:39 +02:00
|
|
|
if (debug)
|
|
|
|
|
qDebug("<findVersionControlForDirectory: cache match for VCS '%s', topLevel: %s",
|
|
|
|
|
qPrintable(it.value()->displayName()), qPrintable(it.key()));
|
2008-12-02 12:01:29 +01:00
|
|
|
return it.value();
|
2009-12-08 14:26:41 +01:00
|
|
|
}
|
2010-05-20 16:24:39 +02:00
|
|
|
pos = index - 1;
|
2008-12-02 12:01:29 +01:00
|
|
|
}
|
|
|
|
|
|
2009-12-08 14:26:41 +01:00
|
|
|
// Nothing: ask the IVersionControls directly, insert the toplevel into the cache.
|
2008-12-03 15:04:51 +01:00
|
|
|
const VersionControlList versionControls = allVersionControls();
|
2008-12-09 11:07:24 +01:00
|
|
|
foreach (IVersionControl * versionControl, versionControls) {
|
2010-05-20 16:24:39 +02:00
|
|
|
QString topLevel;
|
|
|
|
|
if (versionControl->managesDirectory(directory, &topLevel)) {
|
2009-12-08 14:26:41 +01:00
|
|
|
m_d->m_cachedMatches.insert(topLevel, versionControl);
|
|
|
|
|
if (topLevelDirectory)
|
|
|
|
|
*topLevelDirectory = topLevel;
|
2010-05-20 16:24:39 +02:00
|
|
|
if (debug)
|
|
|
|
|
qDebug("<findVersionControlForDirectory: invocation of '%s' matches: %s",
|
|
|
|
|
qPrintable(versionControl->displayName()), qPrintable(topLevel));
|
2008-12-02 12:01:29 +01:00
|
|
|
return versionControl;
|
|
|
|
|
}
|
|
|
|
|
}
|
2010-05-20 16:24:39 +02:00
|
|
|
if (debug)
|
|
|
|
|
qDebug("<findVersionControlForDirectory: No match for %s", qPrintable(directory));
|
2008-12-02 12:01:29 +01:00
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2010-01-11 15:22:17 +01:00
|
|
|
bool VCSManager::promptToDelete(const QString &fileName)
|
2008-12-02 12:01:29 +01:00
|
|
|
{
|
2010-01-11 15:22:17 +01:00
|
|
|
if (IVersionControl *vc = findVersionControlForDirectory(QFileInfo(fileName).absolutePath()))
|
|
|
|
|
return promptToDelete(vc, fileName);
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
2010-09-29 12:16:37 +02:00
|
|
|
IVersionControl *VCSManager::checkout(const QString &versionControlType,
|
2010-09-29 12:16:35 +02:00
|
|
|
const QString &directory,
|
|
|
|
|
const QByteArray &url)
|
|
|
|
|
{
|
2010-09-29 12:16:37 +02:00
|
|
|
foreach (IVersionControl *versionControl, allVersionControls()) {
|
|
|
|
|
if (versionControl->displayName() == versionControlType
|
|
|
|
|
&& versionControl->supportsOperation(Core::IVersionControl::CheckoutOperation)) {
|
|
|
|
|
if (versionControl->vcsCheckout(directory, url)) {
|
2010-09-29 12:16:35 +02:00
|
|
|
m_d->m_cachedMatches.insert(directory, versionControl);
|
|
|
|
|
return versionControl;
|
|
|
|
|
}
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool VCSManager::findVersionControl(const QString &versionControlType)
|
|
|
|
|
{
|
|
|
|
|
foreach (IVersionControl * versionControl, allVersionControls()) {
|
2010-09-29 12:16:37 +02:00
|
|
|
if (versionControl->displayName() == versionControlType)
|
2010-09-29 12:16:35 +02:00
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QString VCSManager::getRepositoryURL(const QString &directory)
|
|
|
|
|
{
|
|
|
|
|
IVersionControl *vc = findVersionControlForDirectory(directory);
|
|
|
|
|
|
|
|
|
|
if (vc && vc->supportsOperation(Core::IVersionControl::GetRepositoryRootOperation))
|
|
|
|
|
return vc->vcsGetRepositoryURL(directory);
|
|
|
|
|
return QString();
|
|
|
|
|
}
|
|
|
|
|
|
2010-01-11 15:22:17 +01:00
|
|
|
bool VCSManager::promptToDelete(IVersionControl *vc, const QString &fileName)
|
|
|
|
|
{
|
|
|
|
|
QTC_ASSERT(vc, return true)
|
|
|
|
|
if (!vc->supportsOperation(IVersionControl::DeleteOperation))
|
2008-12-03 15:04:51 +01:00
|
|
|
return true;
|
2008-12-02 12:01:29 +01:00
|
|
|
const QString title = QCoreApplication::translate("VCSManager", "Version Control");
|
|
|
|
|
const QString msg = QCoreApplication::translate("VCSManager",
|
2008-12-03 15:04:51 +01:00
|
|
|
"Would you like to remove this file from the version control system (%1)?\n"
|
2010-01-07 18:17:24 +01:00
|
|
|
"Note: This might remove the local file.").arg(vc->displayName());
|
2008-12-02 12:01:29 +01:00
|
|
|
const QMessageBox::StandardButton button =
|
|
|
|
|
QMessageBox::question(0, title, msg, QMessageBox::Yes | QMessageBox::No, QMessageBox::Yes);
|
2008-12-03 15:04:51 +01:00
|
|
|
if (button != QMessageBox::Yes)
|
|
|
|
|
return true;
|
|
|
|
|
return vc->vcsDelete(fileName);
|
2008-12-02 12:01:29 +01:00
|
|
|
}
|
|
|
|
|
|
2009-12-08 14:26:41 +01:00
|
|
|
CORE_EXPORT QDebug operator<<(QDebug in, const VCSManager &v)
|
|
|
|
|
{
|
|
|
|
|
QDebug nospace = in.nospace();
|
|
|
|
|
const VersionControlCache::const_iterator cend = v.m_d->m_cachedMatches.constEnd();
|
|
|
|
|
for (VersionControlCache::const_iterator it = v.m_d->m_cachedMatches.constBegin(); it != cend; ++it)
|
2010-01-07 18:17:24 +01:00
|
|
|
nospace << "Directory: " << it.key() << ' ' << it.value()->displayName() << '\n';
|
2009-12-08 14:26:41 +01:00
|
|
|
nospace << '\n';
|
|
|
|
|
return in;
|
|
|
|
|
}
|
|
|
|
|
|
2008-12-02 15:08:31 +01:00
|
|
|
} // namespace Core
|