/**************************************************************************** ** ** 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 "session.h" #include "buildconfiguration.h" #include "deployconfiguration.h" #include "editorconfiguration.h" #include "foldernavigationwidget.h" #include "kit.h" #include "project.h" #include "projectexplorer.h" #include "projectnodes.h" #include "target.h" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #ifdef WITH_TESTS #include #include #include #endif using namespace Core; using namespace Utils; using namespace ProjectExplorer::Internal; namespace ProjectExplorer { /*! \class ProjectExplorer::SessionManager \brief The SessionManager class manages sessions. TODO the interface of this class is not really great. The implementation suffers from that all the functions from the public interface just wrap around functions which do the actual work. This could be improved. */ class SessionManagerPrivate { public: void restoreValues(const PersistentSettingsReader &reader); void restoreDependencies(const PersistentSettingsReader &reader); void restoreStartupProject(const PersistentSettingsReader &reader); void restoreEditors(const PersistentSettingsReader &reader); void restoreProjects(const QStringList &fileList); void askUserAboutFailedProjects(); void sessionLoadingProgress(); bool recursiveDependencyCheck(const QString &newDep, const QString &checkDep) const; QStringList dependencies(const QString &proName) const; QStringList dependenciesOrder() const; void dependencies(const QString &proName, QStringList &result) const; static QString windowTitleAddition(const QString &filePath); static QString sessionTitle(const QString &filePath); bool hasProjects() const { return !m_projects.isEmpty(); } QString m_sessionName = QLatin1String("default"); bool m_virginSession = true; bool m_loadingSession = false; bool m_casadeSetActive = false; mutable QStringList m_sessions; mutable QHash m_sessionDateTimes; Project *m_startupProject = nullptr; QList m_projects; QStringList m_failedProjects; QMap m_depMap; QMap m_values; QFutureInterface m_future; PersistentSettingsWriter *m_writer = nullptr; private: static QString locationInProject(const QString &filePath); }; static SessionManager *m_instance = nullptr; static SessionManagerPrivate *d = nullptr; static QString projectFolderId(Project *pro) { return pro->projectFilePath().toString(); } const int PROJECT_SORT_VALUE = 100; SessionManager::SessionManager(QObject *parent) : QObject(parent) { m_instance = this; d = new SessionManagerPrivate; connect(ModeManager::instance(), &ModeManager::currentModeChanged, this, &SessionManager::saveActiveMode); connect(EditorManager::instance(), &EditorManager::editorCreated, this, &SessionManager::configureEditor); connect(this, &SessionManager::projectAdded, EditorManager::instance(), &EditorManager::updateWindowTitles); connect(this, &SessionManager::projectRemoved, EditorManager::instance(), &EditorManager::updateWindowTitles); connect(this, &SessionManager::projectDisplayNameChanged, EditorManager::instance(), &EditorManager::updateWindowTitles); connect(EditorManager::instance(), &EditorManager::editorOpened, this, &SessionManager::markSessionFileDirty); connect(EditorManager::instance(), &EditorManager::editorsClosed, this, &SessionManager::markSessionFileDirty); EditorManager::setWindowTitleAdditionHandler(&SessionManagerPrivate::windowTitleAddition); EditorManager::setSessionTitleHandler(&SessionManagerPrivate::sessionTitle); } SessionManager::~SessionManager() { emit m_instance->aboutToUnloadSession(d->m_sessionName); delete d->m_writer; delete d; d = nullptr; } SessionManager *SessionManager::instance() { return m_instance; } bool SessionManager::isDefaultVirgin() { return isDefaultSession(d->m_sessionName) && d->m_virginSession; } bool SessionManager::isDefaultSession(const QString &session) { return session == QLatin1String("default"); } void SessionManager::saveActiveMode(Id mode) { if (mode != Core::Constants::MODE_WELCOME) setValue(QLatin1String("ActiveMode"), mode.toString()); } bool SessionManagerPrivate::recursiveDependencyCheck(const QString &newDep, const QString &checkDep) const { if (newDep == checkDep) return false; foreach (const QString &dependency, m_depMap.value(checkDep)) { if (!recursiveDependencyCheck(newDep, dependency)) return false; } return true; } /* * The dependency management exposes an interface based on projects, but * is internally purely string based. This is suboptimal. Probably it would be * nicer to map the filenames to projects on load and only map it back to * filenames when saving. */ QList SessionManager::dependencies(const Project *project) { const QString proName = project->projectFilePath().toString(); const QStringList proDeps = d->m_depMap.value(proName); QList projects; foreach (const QString &dep, proDeps) { const Utils::FilePath fn = Utils::FilePath::fromString(dep); Project *pro = Utils::findOrDefault(d->m_projects, [&fn](Project *p) { return p->projectFilePath() == fn; }); if (pro) projects += pro; } return projects; } bool SessionManager::hasDependency(const Project *project, const Project *depProject) { const QString proName = project->projectFilePath().toString(); const QString depName = depProject->projectFilePath().toString(); const QStringList proDeps = d->m_depMap.value(proName); return proDeps.contains(depName); } bool SessionManager::canAddDependency(const Project *project, const Project *depProject) { const QString newDep = project->projectFilePath().toString(); const QString checkDep = depProject->projectFilePath().toString(); return d->recursiveDependencyCheck(newDep, checkDep); } bool SessionManager::addDependency(Project *project, Project *depProject) { const QString proName = project->projectFilePath().toString(); const QString depName = depProject->projectFilePath().toString(); // check if this dependency is valid if (!d->recursiveDependencyCheck(proName, depName)) return false; QStringList proDeps = d->m_depMap.value(proName); if (!proDeps.contains(depName)) { proDeps.append(depName); d->m_depMap[proName] = proDeps; } emit m_instance->dependencyChanged(project, depProject); return true; } void SessionManager::removeDependency(Project *project, Project *depProject) { const QString proName = project->projectFilePath().toString(); const QString depName = depProject->projectFilePath().toString(); QStringList proDeps = d->m_depMap.value(proName); proDeps.removeAll(depName); if (proDeps.isEmpty()) d->m_depMap.remove(proName); else d->m_depMap[proName] = proDeps; emit m_instance->dependencyChanged(project, depProject); } bool SessionManager::isProjectConfigurationCascading() { return d->m_casadeSetActive; } void SessionManager::setProjectConfigurationCascading(bool b) { d->m_casadeSetActive = b; markSessionFileDirty(); } void SessionManager::setActiveTarget(Project *project, Target *target, SetActive cascade) { QTC_ASSERT(project, return); project->setActiveTarget(target); if (!target) // never cascade setting no target return; if (cascade != SetActive::Cascade || !d->m_casadeSetActive) return; Core::Id kitId = target->kit()->id(); for (Project *otherProject : SessionManager::projects()) { if (otherProject == project) continue; if (Target *otherTarget = Utils::findOrDefault(otherProject->targets(), [kitId](Target *t) { return t->kit()->id() == kitId; })) otherProject->setActiveTarget(otherTarget); } } void SessionManager::setActiveBuildConfiguration(Target *target, BuildConfiguration *bc, SetActive cascade) { QTC_ASSERT(target, return); target->setActiveBuildConfiguration(bc); if (!bc) return; if (cascade != SetActive::Cascade || !d->m_casadeSetActive) return; Core::Id kitId = target->kit()->id(); QString name = bc->displayName(); // We match on displayname for (Project *otherProject : SessionManager::projects()) { if (otherProject == target->project()) continue; Target *otherTarget = otherProject->activeTarget(); if (!otherTarget || otherTarget->kit()->id() != kitId) continue; foreach (BuildConfiguration *otherBc, otherTarget->buildConfigurations()) { if (otherBc->displayName() == name) { otherTarget->setActiveBuildConfiguration(otherBc); break; } } } } void SessionManager::setActiveDeployConfiguration(Target *target, DeployConfiguration *dc, SetActive cascade) { QTC_ASSERT(target, return); target->setActiveDeployConfiguration(dc); if (!dc) return; if (cascade != SetActive::Cascade || !d->m_casadeSetActive) return; Core::Id kitId = target->kit()->id(); QString name = dc->displayName(); // We match on displayname for (Project *otherProject : SessionManager::projects()) { if (otherProject == target->project()) continue; Target *otherTarget = otherProject->activeTarget(); if (!otherTarget || otherTarget->kit()->id() != kitId) continue; foreach (DeployConfiguration *otherDc, otherTarget->deployConfigurations()) { if (otherDc->displayName() == name) { otherTarget->setActiveDeployConfiguration(otherDc); break; } } } } void SessionManager::setStartupProject(Project *startupProject) { QTC_ASSERT((!startupProject && d->m_projects.isEmpty()) || (startupProject && d->m_projects.contains(startupProject)), return); if (d->m_startupProject == startupProject) return; d->m_startupProject = startupProject; if (d->m_startupProject && d->m_startupProject->needsConfiguration()) { ModeManager::activateMode(Constants::MODE_SESSION); ModeManager::setFocusToCurrentMode(); } emit m_instance->startupProjectChanged(startupProject); } Project *SessionManager::startupProject() { return d->m_startupProject; } void SessionManager::addProject(Project *pro) { QTC_ASSERT(pro, return); QTC_CHECK(!pro->displayName().isEmpty()); QTC_CHECK(pro->id().isValid()); d->m_virginSession = false; QTC_ASSERT(!d->m_projects.contains(pro), return); d->m_projects.append(pro); connect(pro, &Project::displayNameChanged, m_instance, [pro]() { emit m_instance->projectDisplayNameChanged(pro); }); emit m_instance->projectAdded(pro); const auto updateFolderNavigation = [pro] { // destructing projects might trigger changes, so check if the project is actually there if (QTC_GUARD(d->m_projects.contains(pro))) { const QIcon icon = pro->rootProjectNode() ? pro->rootProjectNode()->icon() : QIcon(); FolderNavigationWidgetFactory::insertRootDirectory({projectFolderId(pro), PROJECT_SORT_VALUE, pro->displayName(), pro->projectFilePath().parentDir(), icon}); } }; updateFolderNavigation(); configureEditors(pro); connect(pro, &Project::fileListChanged, m_instance, [pro, updateFolderNavigation]() { configureEditors(pro); updateFolderNavigation(); // update icon }); connect(pro, &Project::displayNameChanged, m_instance, updateFolderNavigation); if (!startupProject()) setStartupProject(pro); } void SessionManager::removeProject(Project *project) { d->m_virginSession = false; QTC_ASSERT(project, return); removeProjects({project}); } bool SessionManager::loadingSession() { return d->m_loadingSession; } bool SessionManager::save() { // do not save new virgin default sessions if (isDefaultVirgin()) return true; emit m_instance->aboutToSaveSession(); if (!d->m_writer || d->m_writer->fileName() != sessionNameToFileName(d->m_sessionName)) { delete d->m_writer; d->m_writer = new PersistentSettingsWriter(sessionNameToFileName(d->m_sessionName), QLatin1String("QtCreatorSession")); } QVariantMap data; // save the startup project if (d->m_startupProject) data.insert(QLatin1String("StartupProject"), d->m_startupProject->projectFilePath().toString()); QColor c = StyleHelper::requestedBaseColor(); if (c.isValid()) { QString tmp = QString::fromLatin1("#%1%2%3") .arg(c.red(), 2, 16, QLatin1Char('0')) .arg(c.green(), 2, 16, QLatin1Char('0')) .arg(c.blue(), 2, 16, QLatin1Char('0')); data.insert(QLatin1String("Color"), tmp); } QStringList projectFiles = Utils::transform(projects(), [](Project *p) { return p->projectFilePath().toString(); }); // Restore infromation on projects that failed to load: // don't readd projects to the list, which the user loaded foreach (const QString &failed, d->m_failedProjects) { if (!projectFiles.contains(failed)) projectFiles << failed; } data.insert(QLatin1String("ProjectList"), projectFiles); data.insert(QLatin1String("CascadeSetActive"), d->m_casadeSetActive); QMap depMap; auto i = d->m_depMap.constBegin(); while (i != d->m_depMap.constEnd()) { QString key = i.key(); QStringList values; foreach (const QString &value, i.value()) values << value; depMap.insert(key, values); ++i; } data.insert(QLatin1String("ProjectDependencies"), QVariant(depMap)); data.insert(QLatin1String("EditorSettings"), EditorManager::saveState().toBase64()); auto end = d->m_values.constEnd(); QStringList keys; for (auto it = d->m_values.constBegin(); it != end; ++it) { data.insert(QLatin1String("value-") + it.key(), it.value()); keys << it.key(); } data.insert(QLatin1String("valueKeys"), keys); bool result = d->m_writer->save(data, ICore::mainWindow()); if (result) { d->m_sessionDateTimes.insert(activeSession(), QDateTime::currentDateTime()); } else { QMessageBox::warning(ICore::dialogParent(), tr("Error while saving session"), tr("Could not save session to file %1").arg(d->m_writer->fileName().toUserOutput())); } return result; } /*! Closes all projects */ void SessionManager::closeAllProjects() { removeProjects(projects()); } const QList SessionManager::projects() { return d->m_projects; } bool SessionManager::hasProjects() { return d->hasProjects(); } bool SessionManager::hasProject(Project *p) { return d->m_projects.contains(p); } QStringList SessionManagerPrivate::dependencies(const QString &proName) const { QStringList result; dependencies(proName, result); return result; } void SessionManagerPrivate::dependencies(const QString &proName, QStringList &result) const { QStringList depends = m_depMap.value(proName); foreach (const QString &dep, depends) dependencies(dep, result); if (!result.contains(proName)) result.append(proName); } QString SessionManagerPrivate::sessionTitle(const QString &filePath) { if (SessionManager::isDefaultSession(d->m_sessionName)) { if (filePath.isEmpty()) { // use single project's name if there is only one loaded. const QList projects = SessionManager::projects(); if (projects.size() == 1) return projects.first()->displayName(); } } else { QString sessionName = d->m_sessionName; if (sessionName.isEmpty()) sessionName = SessionManager::tr("Untitled"); return sessionName; } return QString(); } QString SessionManagerPrivate::locationInProject(const QString &filePath) { const Project *project = SessionManager::projectForFile(Utils::FilePath::fromString(filePath)); if (!project) return QString(); const Utils::FilePath file = Utils::FilePath::fromString(filePath); const Utils::FilePath parentDir = file.parentDir(); if (parentDir == project->projectDirectory()) return "@ " + project->displayName(); if (file.isChildOf(project->projectDirectory())) { const Utils::FilePath dirInProject = parentDir.relativeChildPath(project->projectDirectory()); return "(" + dirInProject.toUserOutput() + " @ " + project->displayName() + ")"; } // For a file that is "outside" the project it belongs to, we display its // dir's full path because it is easier to read than a series of "../../.". // Example: /home/hugo/GenericProject/App.files lists /home/hugo/lib/Bar.cpp return "(" + parentDir.toUserOutput() + " @ " + project->displayName() + ")"; } QString SessionManagerPrivate::windowTitleAddition(const QString &filePath) { return locationInProject(filePath); } QStringList SessionManagerPrivate::dependenciesOrder() const { QList > unordered; QStringList ordered; // copy the map to a temporary list for (const Project *pro : m_projects) { const QString proName = pro->projectFilePath().toString(); unordered << QPair(proName, m_depMap.value(proName)); } while (!unordered.isEmpty()) { for (int i = (unordered.count() - 1); i >= 0; --i) { if (unordered.at(i).second.isEmpty()) { ordered << unordered.at(i).first; unordered.removeAt(i); } } // remove the handled projects from the dependency lists // of the remaining unordered projects for (int i = 0; i < unordered.count(); ++i) { foreach (const QString &pro, ordered) { QStringList depList = unordered.at(i).second; depList.removeAll(pro); unordered[i].second = depList; } } } return ordered; } QList SessionManager::projectOrder(const Project *project) { QList result; QStringList pros; if (project) pros = d->dependencies(project->projectFilePath().toString()); else pros = d->dependenciesOrder(); foreach (const QString &proFile, pros) { for (Project *pro : projects()) { if (pro->projectFilePath().toString() == proFile) { result << pro; break; } } } return result; } Project *SessionManager::projectForFile(const Utils::FilePath &fileName) { return Utils::findOrDefault(SessionManager::projects(), [&fileName](const Project *p) { return p->isKnownFile(fileName); }); } void SessionManager::configureEditor(IEditor *editor, const QString &fileName) { if (auto textEditor = qobject_cast(editor)) { Project *project = projectForFile(Utils::FilePath::fromString(fileName)); // Global settings are the default. if (project) project->editorConfiguration()->configureEditor(textEditor); } } void SessionManager::configureEditors(Project *project) { foreach (IDocument *document, DocumentModel::openedDocuments()) { if (project->isKnownFile(document->filePath())) { foreach (IEditor *editor, DocumentModel::editorsForDocument(document)) { if (auto textEditor = qobject_cast(editor)) { project->editorConfiguration()->configureEditor(textEditor); } } } } } void SessionManager::removeProjects(const QList &remove) { QMap resMap; for (Project *pro : remove) emit m_instance->aboutToRemoveProject(pro); // Refresh dependencies QSet projectFiles; for (Project *pro : projects()) { if (!remove.contains(pro)) projectFiles.insert(pro->projectFilePath().toString()); } auto i = projectFiles.begin(); while (i != projectFiles.end()) { QStringList dependencies; foreach (const QString &dependency, d->m_depMap.value(*i)) { if (projectFiles.contains(dependency)) dependencies << dependency; } if (!dependencies.isEmpty()) resMap.insert(*i, dependencies); ++i; } d->m_depMap = resMap; bool changeStartupProject = false; // Delete projects for (Project *pro : remove) { pro->saveSettings(); // Remove the project node: d->m_projects.removeOne(pro); if (pro == d->m_startupProject) changeStartupProject = true; FolderNavigationWidgetFactory::removeRootDirectory(projectFolderId(pro)); disconnect(pro, nullptr, m_instance, nullptr); emit m_instance->projectRemoved(pro); } if (changeStartupProject) setStartupProject(hasProjects() ? projects().first() : nullptr); qDeleteAll(remove); } /*! Lets other plugins store persistent values within the session file. */ void SessionManager::setValue(const QString &name, const QVariant &value) { if (d->m_values.value(name) == value) return; d->m_values.insert(name, value); } QVariant SessionManager::value(const QString &name) { auto it = d->m_values.constFind(name); return (it == d->m_values.constEnd()) ? QVariant() : *it; } QString SessionManager::activeSession() { return d->m_sessionName; } QStringList SessionManager::sessions() { if (d->m_sessions.isEmpty()) { // We are not initialized yet, so do that now QDir sessionDir(ICore::userResourcePath()); QFileInfoList sessionFiles = sessionDir.entryInfoList(QStringList() << QLatin1String("*.qws"), QDir::NoFilter, QDir::Time); foreach (const QFileInfo &fileInfo, sessionFiles) { const QString &name = fileInfo.completeBaseName(); d->m_sessionDateTimes.insert(name, fileInfo.lastModified()); if (name != QLatin1String("default")) d->m_sessions << name; } d->m_sessions.prepend(QLatin1String("default")); } return d->m_sessions; } QDateTime SessionManager::sessionDateTime(const QString &session) { return d->m_sessionDateTimes.value(session); } FilePath SessionManager::sessionNameToFileName(const QString &session) { return FilePath::fromString(ICore::userResourcePath() + QLatin1Char('/') + session + QLatin1String(".qws")); } /*! Creates \a session, but does not actually create the file. */ bool SessionManager::createSession(const QString &session) { if (sessions().contains(session)) return false; Q_ASSERT(d->m_sessions.size() > 0); d->m_sessions.insert(1, session); return true; } bool SessionManager::renameSession(const QString &original, const QString &newName) { if (!cloneSession(original, newName)) return false; if (original == activeSession()) loadSession(newName); return deleteSession(original); } /*! \brief Shows a dialog asking the user to confirm deleting the session \p session */ bool SessionManager::confirmSessionDelete(const QStringList &sessions) { const QString title = sessions.size() == 1 ? tr("Delete Session") : tr("Delete Sessions"); const QString question = sessions.size() == 1 ? tr("Delete session %1?").arg(sessions.first()) : tr("Delete these sessions?\n %1").arg(sessions.join("\n ")); return QMessageBox::question(ICore::mainWindow(), title, question, QMessageBox::Yes | QMessageBox::No) == QMessageBox::Yes; } /*! Deletes \a session name from session list and the file from disk. */ bool SessionManager::deleteSession(const QString &session) { if (!d->m_sessions.contains(session)) return false; d->m_sessions.removeOne(session); QFile fi(sessionNameToFileName(session).toString()); if (fi.exists()) return fi.remove(); return false; } void SessionManager::deleteSessions(const QStringList &sessions) { for (const QString &session : sessions) deleteSession(session); } bool SessionManager::cloneSession(const QString &original, const QString &clone) { if (!d->m_sessions.contains(original)) return false; QFile fi(sessionNameToFileName(original).toString()); // If the file does not exist, we can still clone if (!fi.exists() || fi.copy(sessionNameToFileName(clone).toString())) { d->m_sessions.insert(1, clone); d->m_sessionDateTimes.insert(clone, sessionNameToFileName(clone).toFileInfo().lastModified()); return true; } return false; } void SessionManagerPrivate::restoreValues(const PersistentSettingsReader &reader) { const QStringList keys = reader.restoreValue(QLatin1String("valueKeys")).toStringList(); foreach (const QString &key, keys) { QVariant value = reader.restoreValue(QLatin1String("value-") + key); m_values.insert(key, value); } } void SessionManagerPrivate::restoreDependencies(const PersistentSettingsReader &reader) { QMap depMap = reader.restoreValue(QLatin1String("ProjectDependencies")).toMap(); auto i = depMap.constBegin(); while (i != depMap.constEnd()) { const QString &key = i.key(); if (!m_failedProjects.contains(key)) { QStringList values; foreach (const QString &value, i.value().toStringList()) { if (!m_failedProjects.contains(value)) values << value; } m_depMap.insert(key, values); } ++i; } } void SessionManagerPrivate::askUserAboutFailedProjects() { QStringList failedProjects = m_failedProjects; if (!failedProjects.isEmpty()) { QString fileList = QDir::toNativeSeparators(failedProjects.join(QLatin1String("
"))); QMessageBox box(QMessageBox::Warning, SessionManager::tr("Failed to restore project files"), SessionManager::tr("Could not restore the following project files:
%1"). arg(fileList)); auto keepButton = new QPushButton(SessionManager::tr("Keep projects in Session"), &box); auto removeButton = new QPushButton(SessionManager::tr("Remove projects from Session"), &box); box.addButton(keepButton, QMessageBox::AcceptRole); box.addButton(removeButton, QMessageBox::DestructiveRole); box.exec(); if (box.clickedButton() == removeButton) m_failedProjects.clear(); } } void SessionManagerPrivate::restoreStartupProject(const PersistentSettingsReader &reader) { const QString startupProject = reader.restoreValue(QLatin1String("StartupProject")).toString(); if (!startupProject.isEmpty()) { for (Project *pro : m_projects) { if (pro->projectFilePath().toString() == startupProject) { m_instance->setStartupProject(pro); break; } } } if (!m_startupProject) { if (!startupProject.isEmpty()) qWarning() << "Could not find startup project" << startupProject; if (hasProjects()) m_instance->setStartupProject(m_projects.first()); } } void SessionManagerPrivate::restoreEditors(const PersistentSettingsReader &reader) { const QVariant editorsettings = reader.restoreValue(QLatin1String("EditorSettings")); if (editorsettings.isValid()) { EditorManager::restoreState(QByteArray::fromBase64(editorsettings.toByteArray())); sessionLoadingProgress(); } } /*! Loads a session, takes a session name (not filename). */ void SessionManagerPrivate::restoreProjects(const QStringList &fileList) { // indirectly adds projects to session // Keep projects that failed to load in the session! m_failedProjects = fileList; if (!fileList.isEmpty()) { ProjectExplorerPlugin::OpenProjectResult result = ProjectExplorerPlugin::openProjects(fileList); if (!result) ProjectExplorerPlugin::showOpenProjectError(result); foreach (Project *p, result.projects()) m_failedProjects.removeAll(p->projectFilePath().toString()); } } bool SessionManager::loadSession(const QString &session) { // Do nothing if we have that session already loaded, // exception if the session is the default virgin session // we still want to be able to load the default session if (session == d->m_sessionName && !isDefaultVirgin()) return true; if (!sessions().contains(session)) return false; QStringList fileList; // Try loading the file FilePath fileName = sessionNameToFileName(session); PersistentSettingsReader reader; if (fileName.exists()) { if (!reader.load(fileName)) { QMessageBox::warning(ICore::dialogParent(), tr("Error while restoring session"), tr("Could not restore session %1").arg(fileName.toUserOutput())); return false; } fileList = reader.restoreValue(QLatin1String("ProjectList")).toStringList(); } d->m_loadingSession = true; // Allow everyone to set something in the session and before saving emit m_instance->aboutToUnloadSession(d->m_sessionName); if (!save()) { d->m_loadingSession = false; return false; } // Clean up if (!EditorManager::closeAllEditors()) { d->m_loadingSession = false; return false; } // find a list of projects to close later const QList projectsToRemove = Utils::filtered(projects(), [&fileList](Project *p) { return !fileList.contains(p->projectFilePath().toString()); }); const QList openProjects = projects(); const QStringList projectPathsToLoad = Utils::filtered(fileList, [&openProjects](const QString &path) { return !Utils::contains(openProjects, [&path](Project *p) { return p->projectFilePath().toString() == path; }); }); d->m_failedProjects.clear(); d->m_depMap.clear(); d->m_values.clear(); d->m_casadeSetActive = false; d->m_sessionName = session; delete d->m_writer; d->m_writer = nullptr; EditorManager::updateWindowTitles(); if (fileName.exists()) { d->m_virginSession = false; ProgressManager::addTask(d->m_future.future(), tr("Loading Session"), "ProjectExplorer.SessionFile.Load"); d->m_future.setProgressRange(0, 1); d->m_future.setProgressValue(0); d->restoreValues(reader); emit m_instance->aboutToLoadSession(session); // retrieve all values before the following code could change them again Id modeId = Id::fromSetting(value(QLatin1String("ActiveMode"))); if (!modeId.isValid()) modeId = Id(Core::Constants::MODE_EDIT); QColor c = QColor(reader.restoreValue(QLatin1String("Color")).toString()); if (c.isValid()) StyleHelper::setBaseColor(c); d->m_future.setProgressRange(0, projectPathsToLoad.count() + 1/*initialization above*/ + 1/*editors*/); d->m_future.setProgressValue(1); QCoreApplication::processEvents(QEventLoop::ExcludeUserInputEvents); d->restoreProjects(projectPathsToLoad); d->sessionLoadingProgress(); d->restoreDependencies(reader); d->restoreStartupProject(reader); removeProjects(projectsToRemove); // only remove old projects now that the startup project is set! d->restoreEditors(reader); d->m_future.reportFinished(); d->m_future = QFutureInterface(); // Fall back to Project mode if the startup project is unconfigured and // use the mode saved in the session otherwise if (d->m_startupProject && d->m_startupProject->needsConfiguration()) modeId = Id(Constants::MODE_SESSION); ModeManager::activateMode(modeId); ModeManager::setFocusToCurrentMode(); } else { removeProjects(projects()); ModeManager::activateMode(Id(Core::Constants::MODE_EDIT)); ModeManager::setFocusToCurrentMode(); } d->m_casadeSetActive = reader.restoreValue(QLatin1String("CascadeSetActive"), false).toBool(); emit m_instance->sessionLoaded(session); // Starts a event loop, better do that at the very end d->askUserAboutFailedProjects(); d->m_loadingSession = false; return true; } QString SessionManager::lastSession() { return ICore::settings()->value(QLatin1String("ProjectExplorer/StartupSession")).toString(); } void SessionManager::reportProjectLoadingProgress() { d->sessionLoadingProgress(); } void SessionManager::markSessionFileDirty() { d->m_virginSession = false; } void SessionManagerPrivate::sessionLoadingProgress() { m_future.setProgressValue(m_future.progressValue() + 1); QCoreApplication::processEvents(QEventLoop::ExcludeUserInputEvents); } QStringList SessionManager::projectsForSessionName(const QString &session) { const FilePath fileName = sessionNameToFileName(session); PersistentSettingsReader reader; if (fileName.exists()) { if (!reader.load(fileName)) { qWarning() << "Could not restore session" << fileName.toUserOutput(); return QStringList(); } } return reader.restoreValue(QLatin1String("ProjectList")).toStringList(); } #ifdef WITH_TESTS void ProjectExplorerPlugin::testSessionSwitch() { QVERIFY(SessionManager::createSession("session1")); QVERIFY(SessionManager::createSession("session2")); QTemporaryFile cppFile("main.cpp"); QVERIFY(cppFile.open()); cppFile.close(); QTemporaryFile projectFile1("XXXXXX.pro"); QTemporaryFile projectFile2("XXXXXX.pro"); struct SessionSpec { SessionSpec(const QString &n, QTemporaryFile &f) : name(n), projectFile(f) {} const QString name; QTemporaryFile &projectFile; }; std::vector sessionSpecs{SessionSpec("session1", projectFile1), SessionSpec("session2", projectFile2)}; for (const SessionSpec &sessionSpec : sessionSpecs) { static const QByteArray proFileContents = "TEMPLATE = app\n" "CONFIG -= qt\n" "SOURCES = " + cppFile.fileName().toLocal8Bit(); QVERIFY(sessionSpec.projectFile.open()); sessionSpec.projectFile.write(proFileContents); sessionSpec.projectFile.close(); QVERIFY(SessionManager::loadSession(sessionSpec.name)); const OpenProjectResult openResult = ProjectExplorerPlugin::openProject(sessionSpec.projectFile.fileName()); if (openResult.errorMessage().contains("text/plain")) QSKIP("This test requires the presence of QmakeProjectManager to be fully functional"); QVERIFY(openResult); QCOMPARE(openResult.projects().count(), 1); QVERIFY(openResult.project()); QCOMPARE(SessionManager::projects().count(), 1); } for (int i = 0; i < 30; ++i) { QVERIFY(SessionManager::loadSession("session1")); QCOMPARE(SessionManager::activeSession(), "session1"); QCOMPARE(SessionManager::projects().count(), 1); QVERIFY(SessionManager::loadSession("session2")); QCOMPARE(SessionManager::activeSession(), "session2"); QCOMPARE(SessionManager::projects().count(), 1); } QVERIFY(SessionManager::loadSession("session1")); SessionManager::closeAllProjects(); QVERIFY(SessionManager::loadSession("session2")); SessionManager::closeAllProjects(); QVERIFY(SessionManager::deleteSession("session1")); QVERIFY(SessionManager::deleteSession("session2")); } #endif // WITH_TESTS } // namespace ProjectExplorer