| 
									
										
										
										
											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
 |