/**************************************************************************** ** ** Copyright (C) 2016 The Qt Company Ltd. ** Contact: https://www.qt.io/licensing/ ** ** This file is part of Qt Creator. ** ** Commercial License Usage ** Licensees holding valid commercial Qt licenses may use this file in ** accordance with the commercial license agreement provided with the ** Software or, alternatively, in accordance with the terms contained in ** a written agreement between you and The Qt Company. For licensing terms ** and conditions see https://www.qt.io/terms-conditions. For further ** information use the contact form at https://www.qt.io/contact-us. ** ** GNU General Public License Usage ** Alternatively, this file may be used under the terms of the GNU ** General Public License version 3 as published by the Free Software ** Foundation with exceptions as appearing in the file LICENSE.GPL3-EXCEPT ** included in the packaging of this file. Please review the following ** information to ensure the GNU General Public License requirements will ** be met: https://www.gnu.org/licenses/gpl-3.0.html. ** ****************************************************************************/ #include "vcsprojectcache.h" #include "vcsbasesubmiteditor.h" #include #include #include #include #include using namespace Utils; namespace { class PathMatcher { public: ProjectExplorer::Project *project() { return m_project; } void match(ProjectExplorer::Project *project, const FileName &base, const FileName &child) { int count = std::numeric_limits::max(); if (child.isChildOf(base)) { const QString relative = child.toString().mid(base.count() + 1); count = relative.count(QLatin1Char('/')); } if (count < m_count) { m_count = count; m_project = project; } } private: int m_count = std::numeric_limits::max(); ProjectExplorer::Project *m_project = nullptr; }; } // namespace namespace VcsBase { namespace Internal { VcsProjectCache *VcsProjectCache::m_instance = nullptr; VcsProjectCache::VcsProjectCache() { QTC_ASSERT(!m_instance, return); m_instance = this; connect(ProjectExplorer::SessionManager::instance(), &ProjectExplorer::SessionManager::projectAdded, this, []() { VcsProjectCache::invalidate(); }); connect(ProjectExplorer::SessionManager::instance(), &ProjectExplorer::SessionManager::projectRemoved, this, []() { VcsProjectCache::invalidate(); }); } VcsProjectCache::~VcsProjectCache() { m_instance = nullptr; } ProjectExplorer::Project *VcsProjectCache::projectFor(const QString &repo) { ProjectExplorer::Project *project; const int pos = Utils::indexOf(m_instance->m_cache, [repo](const CacheNode &n) { return n.repository == repo; }); if (pos >= 0) { if (pos > 0) { m_instance->m_cache.prepend(m_instance->m_cache.at(pos)); m_instance->m_cache.removeAt(pos + 1); } return m_instance->m_cache.at(0).project; } project = projectForToplevel(FileName::fromString(repo)); m_instance->m_cache.prepend(CacheNode(repo, project)); while (m_instance->m_cache.count() > 10) m_instance->m_cache.removeLast(); return project; } void VcsProjectCache::invalidate() { m_instance->m_cache.clear(); } void VcsProjectCache::create() { new VcsProjectCache; } void VcsProjectCache::destroy() { delete m_instance; } ProjectExplorer::Project *VcsProjectCache::projectForToplevel(const FileName &vcsTopLevel) { PathMatcher parentMatcher; PathMatcher childMatcher; for (ProjectExplorer::Project *project : ProjectExplorer::SessionManager::projects()) { const FileName projectDir = project->projectDirectory(); if (projectDir == vcsTopLevel) return project; parentMatcher.match(project, vcsTopLevel, projectDir); childMatcher.match(project, projectDir, vcsTopLevel); } if (parentMatcher.project()) return parentMatcher.project(); return childMatcher.project(); } } // Internal } // namespace VcsBase