2022-08-19 15:59:36 +02:00
|
|
|
// Copyright (C) 2016 The Qt Company Ltd.
|
2022-12-21 10:12:09 +01:00
|
|
|
// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR GPL-3.0-only WITH Qt-GPL-exception-1.0
|
2008-12-02 16:19:05 +01:00
|
|
|
|
|
|
|
|
#include "session.h"
|
|
|
|
|
|
2023-02-28 17:37:59 +01:00
|
|
|
#include "session_p.h"
|
|
|
|
|
|
2023-01-13 12:38:22 +01:00
|
|
|
#include "projectexplorertr.h"
|
2023-02-14 15:47:22 +01:00
|
|
|
#include "projectmanager.h"
|
2008-12-02 12:01:29 +01:00
|
|
|
|
2017-12-04 10:41:46 +01:00
|
|
|
#include <coreplugin/coreconstants.h>
|
|
|
|
|
#include <coreplugin/editormanager/editormanager.h>
|
2021-10-21 15:39:28 +02:00
|
|
|
#include <coreplugin/foldernavigationwidget.h>
|
2008-12-02 12:01:29 +01:00
|
|
|
#include <coreplugin/icore.h>
|
2015-02-26 13:38:54 +01:00
|
|
|
#include <coreplugin/idocument.h>
|
2008-12-02 12:01:29 +01:00
|
|
|
#include <coreplugin/imode.h>
|
|
|
|
|
#include <coreplugin/modemanager.h>
|
2017-12-04 10:41:46 +01:00
|
|
|
#include <coreplugin/progressmanager/progressmanager.h>
|
2008-12-02 12:01:29 +01:00
|
|
|
|
2014-09-26 09:14:03 +02:00
|
|
|
#include <texteditor/texteditor.h>
|
2008-12-02 12:01:29 +01:00
|
|
|
|
2014-07-07 19:02:26 +02:00
|
|
|
#include <utils/algorithm.h>
|
2022-11-09 16:07:09 +01:00
|
|
|
#include <utils/filepath.h>
|
2022-07-27 14:18:48 +02:00
|
|
|
#include <utils/qtcassert.h>
|
2012-03-12 13:03:26 +01:00
|
|
|
#include <utils/stylehelper.h>
|
2022-07-27 12:27:41 +02:00
|
|
|
#include <utils/qtcassert.h>
|
2008-12-09 15:25:01 +01:00
|
|
|
|
2012-02-15 10:42:41 +01:00
|
|
|
#include <QDebug>
|
|
|
|
|
#include <QMessageBox>
|
|
|
|
|
#include <QPushButton>
|
2008-12-02 12:01:29 +01:00
|
|
|
|
2011-09-02 17:15:18 +02:00
|
|
|
using namespace Core;
|
2013-09-05 11:46:07 +02:00
|
|
|
using namespace Utils;
|
2008-12-02 12:01:29 +01:00
|
|
|
|
2013-09-05 11:46:07 +02:00
|
|
|
namespace ProjectExplorer {
|
|
|
|
|
|
2021-12-21 16:40:59 +01:00
|
|
|
const char DEFAULT_SESSION[] = "default";
|
2022-10-20 17:53:09 +02:00
|
|
|
const char LAST_ACTIVE_TIMES_KEY[] = "LastActiveTimes";
|
2023-05-12 15:25:28 +02:00
|
|
|
const char STARTUPSESSION_KEY[] = "ProjectExplorer/SessionToRestore";
|
|
|
|
|
const char LASTSESSION_KEY[] = "ProjectExplorer/StartupSession";
|
2021-12-21 16:40:59 +01:00
|
|
|
|
2011-04-14 12:58:14 +02:00
|
|
|
/*!
|
|
|
|
|
\class ProjectExplorer::SessionManager
|
|
|
|
|
|
2013-06-05 14:29:24 +02:00
|
|
|
\brief The SessionManager class manages sessions.
|
2011-04-14 12:58:14 +02:00
|
|
|
|
|
|
|
|
TODO the interface of this class is not really great.
|
2013-09-10 17:16:10 +02:00
|
|
|
The implementation suffers from that all the functions from the
|
|
|
|
|
public interface just wrap around functions which do the actual work.
|
2011-04-14 12:58:14 +02:00
|
|
|
This could be improved.
|
|
|
|
|
*/
|
2008-12-02 12:01:29 +01:00
|
|
|
|
2023-02-28 17:37:59 +01:00
|
|
|
static SessionManager *m_instance = nullptr;
|
|
|
|
|
SessionManagerPrivate *sb_d = nullptr;
|
2017-09-26 17:19:00 +02:00
|
|
|
|
2023-02-14 15:47:22 +01:00
|
|
|
SessionManager::SessionManager()
|
2008-12-02 12:01:29 +01:00
|
|
|
{
|
2023-02-28 17:37:59 +01:00
|
|
|
m_instance = this;
|
2023-02-14 15:47:22 +01:00
|
|
|
sb_d = new SessionManagerPrivate;
|
2013-09-05 11:46:07 +02:00
|
|
|
|
2015-02-23 14:06:58 +01:00
|
|
|
connect(ModeManager::instance(), &ModeManager::currentModeChanged,
|
|
|
|
|
this, &SessionManager::saveActiveMode);
|
2010-03-03 15:54:32 +01:00
|
|
|
|
2022-10-20 17:53:09 +02:00
|
|
|
connect(ICore::instance(), &ICore::saveSettingsRequested, this, [] {
|
2023-05-12 15:25:28 +02:00
|
|
|
QtcSettings *s = ICore::settings();
|
2022-10-20 17:53:09 +02:00
|
|
|
QVariantMap times;
|
2023-02-14 15:47:22 +01:00
|
|
|
for (auto it = sb_d->m_lastActiveTimes.cbegin(); it != sb_d->m_lastActiveTimes.cend(); ++it)
|
2022-10-20 17:53:09 +02:00
|
|
|
times.insert(it.key(), it.value());
|
2023-05-12 15:25:28 +02:00
|
|
|
s->setValue(LAST_ACTIVE_TIMES_KEY, times);
|
|
|
|
|
if (SessionManager::isDefaultVirgin()) {
|
|
|
|
|
s->remove(STARTUPSESSION_KEY);
|
|
|
|
|
} else {
|
|
|
|
|
s->setValue(STARTUPSESSION_KEY, SessionManager::activeSession());
|
|
|
|
|
s->setValue(LASTSESSION_KEY, SessionManager::activeSession());
|
|
|
|
|
}
|
2022-10-20 17:53:09 +02:00
|
|
|
});
|
|
|
|
|
|
2015-02-23 14:06:58 +01:00
|
|
|
connect(EditorManager::instance(), &EditorManager::editorOpened,
|
2018-05-25 17:06:42 +02:00
|
|
|
this, &SessionManager::markSessionFileDirty);
|
2015-02-23 14:06:58 +01:00
|
|
|
connect(EditorManager::instance(), &EditorManager::editorsClosed,
|
2018-05-25 17:06:42 +02:00
|
|
|
this, &SessionManager::markSessionFileDirty);
|
2009-01-20 17:14:00 +01:00
|
|
|
}
|
2008-12-02 12:01:29 +01:00
|
|
|
|
|
|
|
|
SessionManager::~SessionManager()
|
2023-02-14 15:47:22 +01:00
|
|
|
{
|
2023-02-28 17:37:59 +01:00
|
|
|
emit m_instance->aboutToUnloadSession(sb_d->m_sessionName);
|
2023-02-14 15:47:22 +01:00
|
|
|
delete sb_d->m_writer;
|
|
|
|
|
delete sb_d;
|
|
|
|
|
sb_d = nullptr;
|
|
|
|
|
}
|
|
|
|
|
|
2014-10-09 16:46:13 +02:00
|
|
|
SessionManager *SessionManager::instance()
|
2013-09-05 11:46:07 +02:00
|
|
|
{
|
|
|
|
|
return m_instance;
|
|
|
|
|
}
|
2008-12-02 12:01:29 +01:00
|
|
|
|
2013-09-05 11:46:07 +02:00
|
|
|
bool SessionManager::isDefaultVirgin()
|
2008-12-02 12:01:29 +01:00
|
|
|
{
|
2023-02-14 15:47:22 +01:00
|
|
|
return isDefaultSession(sb_d->m_sessionName) && sb_d->m_virginSession;
|
2008-12-02 12:01:29 +01:00
|
|
|
}
|
|
|
|
|
|
2013-09-05 11:46:07 +02:00
|
|
|
bool SessionManager::isDefaultSession(const QString &session)
|
2008-12-02 12:01:29 +01:00
|
|
|
{
|
2021-12-21 16:40:59 +01:00
|
|
|
return session == QLatin1String(DEFAULT_SESSION);
|
2008-12-02 12:01:29 +01:00
|
|
|
}
|
|
|
|
|
|
2016-03-25 20:00:19 +01:00
|
|
|
void SessionManager::saveActiveMode(Id mode)
|
2008-12-02 12:01:29 +01:00
|
|
|
{
|
2016-03-25 20:00:19 +01:00
|
|
|
if (mode != Core::Constants::MODE_WELCOME)
|
|
|
|
|
setValue(QLatin1String("ActiveMode"), mode.toString());
|
2008-12-02 12:01:29 +01:00
|
|
|
}
|
|
|
|
|
|
2012-05-31 13:02:08 +02:00
|
|
|
bool SessionManager::loadingSession()
|
|
|
|
|
{
|
2023-02-14 15:47:22 +01:00
|
|
|
return sb_d->m_loadingSession;
|
2012-05-31 13:02:08 +02:00
|
|
|
}
|
|
|
|
|
|
2011-04-14 12:58:14 +02:00
|
|
|
/*!
|
2013-09-10 17:16:10 +02:00
|
|
|
Lets other plugins store persistent values within the session file.
|
2011-04-14 12:58:14 +02:00
|
|
|
*/
|
|
|
|
|
|
2008-12-02 12:01:29 +01:00
|
|
|
void SessionManager::setValue(const QString &name, const QVariant &value)
|
|
|
|
|
{
|
2023-02-14 15:47:22 +01:00
|
|
|
if (sb_d->m_values.value(name) == value)
|
2010-03-03 15:54:32 +01:00
|
|
|
return;
|
2023-02-14 15:47:22 +01:00
|
|
|
sb_d->m_values.insert(name, value);
|
2008-12-02 12:01:29 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QVariant SessionManager::value(const QString &name)
|
|
|
|
|
{
|
2023-02-14 15:47:22 +01:00
|
|
|
auto it = sb_d->m_values.constFind(name);
|
|
|
|
|
return (it == sb_d->m_values.constEnd()) ? QVariant() : *it;
|
2008-12-02 12:01:29 +01:00
|
|
|
}
|
|
|
|
|
|
2013-09-05 11:46:07 +02:00
|
|
|
QString SessionManager::activeSession()
|
2008-12-02 12:01:29 +01:00
|
|
|
{
|
2023-02-14 15:47:22 +01:00
|
|
|
return sb_d->m_sessionName;
|
2008-12-02 12:01:29 +01:00
|
|
|
}
|
|
|
|
|
|
2013-09-05 11:46:07 +02:00
|
|
|
QStringList SessionManager::sessions()
|
2008-12-02 12:01:29 +01:00
|
|
|
{
|
2023-02-14 15:47:22 +01:00
|
|
|
if (sb_d->m_sessions.isEmpty()) {
|
2010-01-11 10:22:55 +01:00
|
|
|
// We are not initialized yet, so do that now
|
2022-01-27 10:30:09 +01:00
|
|
|
const FilePaths sessionFiles =
|
|
|
|
|
ICore::userResourcePath().dirEntries({{"*qws"}}, QDir::Time | QDir::Reversed);
|
2022-10-20 17:53:09 +02:00
|
|
|
const QVariantMap lastActiveTimes = ICore::settings()->value(LAST_ACTIVE_TIMES_KEY).toMap();
|
2021-06-29 17:51:56 +02:00
|
|
|
for (const FilePath &file : sessionFiles) {
|
|
|
|
|
const QString &name = file.completeBaseName();
|
2023-02-14 15:47:22 +01:00
|
|
|
sb_d->m_sessionDateTimes.insert(name, file.lastModified());
|
2022-10-20 17:53:09 +02:00
|
|
|
const auto lastActiveTime = lastActiveTimes.find(name);
|
2023-02-14 15:47:22 +01:00
|
|
|
sb_d->m_lastActiveTimes.insert(name, lastActiveTime != lastActiveTimes.end()
|
2022-10-20 17:53:09 +02:00
|
|
|
? lastActiveTime->toDateTime()
|
|
|
|
|
: file.lastModified());
|
2021-12-21 16:40:59 +01:00
|
|
|
if (name != QLatin1String(DEFAULT_SESSION))
|
2023-02-14 15:47:22 +01:00
|
|
|
sb_d->m_sessions << name;
|
2009-06-18 11:36:39 +02:00
|
|
|
}
|
2023-02-14 15:47:22 +01:00
|
|
|
sb_d->m_sessions.prepend(QLatin1String(DEFAULT_SESSION));
|
2009-06-18 11:36:39 +02:00
|
|
|
}
|
2023-02-14 15:47:22 +01:00
|
|
|
return sb_d->m_sessions;
|
2008-12-02 12:01:29 +01:00
|
|
|
}
|
|
|
|
|
|
2016-09-23 19:14:50 +02:00
|
|
|
QDateTime SessionManager::sessionDateTime(const QString &session)
|
|
|
|
|
{
|
2023-02-14 15:47:22 +01:00
|
|
|
return sb_d->m_sessionDateTimes.value(session);
|
2016-09-23 19:14:50 +02:00
|
|
|
}
|
|
|
|
|
|
2022-10-20 17:53:09 +02:00
|
|
|
QDateTime SessionManager::lastActiveTime(const QString &session)
|
|
|
|
|
{
|
2023-02-14 15:47:22 +01:00
|
|
|
return sb_d->m_lastActiveTimes.value(session);
|
2022-10-20 17:53:09 +02:00
|
|
|
}
|
|
|
|
|
|
2019-05-28 13:49:26 +02:00
|
|
|
FilePath SessionManager::sessionNameToFileName(const QString &session)
|
2008-12-02 12:01:29 +01:00
|
|
|
{
|
2021-04-26 15:46:09 +02:00
|
|
|
return ICore::userResourcePath(session + ".qws");
|
2008-12-02 12:01:29 +01:00
|
|
|
}
|
|
|
|
|
|
2011-04-14 12:58:14 +02:00
|
|
|
/*!
|
2013-09-10 17:16:10 +02:00
|
|
|
Creates \a session, but does not actually create the file.
|
2011-04-14 12:58:14 +02:00
|
|
|
*/
|
|
|
|
|
|
2008-12-02 12:01:29 +01:00
|
|
|
bool SessionManager::createSession(const QString &session)
|
|
|
|
|
{
|
|
|
|
|
if (sessions().contains(session))
|
|
|
|
|
return false;
|
2023-02-14 15:47:22 +01:00
|
|
|
Q_ASSERT(sb_d->m_sessions.size() > 0);
|
|
|
|
|
sb_d->m_sessions.insert(1, session);
|
|
|
|
|
sb_d->m_lastActiveTimes.insert(session, QDateTime::currentDateTime());
|
2008-12-02 12:01:29 +01:00
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
2010-04-21 16:08:13 +02:00
|
|
|
bool SessionManager::renameSession(const QString &original, const QString &newName)
|
|
|
|
|
{
|
|
|
|
|
if (!cloneSession(original, newName))
|
|
|
|
|
return false;
|
|
|
|
|
if (original == activeSession())
|
2023-02-14 15:47:22 +01:00
|
|
|
ProjectManager::loadSession(newName);
|
2021-11-23 17:04:13 +01:00
|
|
|
emit instance()->sessionRenamed(original, newName);
|
2010-04-21 16:08:13 +02:00
|
|
|
return deleteSession(original);
|
|
|
|
|
}
|
|
|
|
|
|
2013-05-23 15:51:50 +02:00
|
|
|
|
|
|
|
|
/*!
|
|
|
|
|
\brief Shows a dialog asking the user to confirm deleting the session \p session
|
|
|
|
|
*/
|
2019-05-09 16:21:42 +02:00
|
|
|
bool SessionManager::confirmSessionDelete(const QStringList &sessions)
|
2013-05-23 15:51:50 +02:00
|
|
|
{
|
2023-01-13 12:38:22 +01:00
|
|
|
const QString title = sessions.size() == 1 ? Tr::tr("Delete Session") : Tr::tr("Delete Sessions");
|
2019-05-09 16:21:42 +02:00
|
|
|
const QString question = sessions.size() == 1
|
2023-01-13 12:38:22 +01:00
|
|
|
? Tr::tr("Delete session %1?").arg(sessions.first())
|
|
|
|
|
: Tr::tr("Delete these sessions?\n %1").arg(sessions.join("\n "));
|
2020-06-02 09:10:40 +02:00
|
|
|
return QMessageBox::question(ICore::dialogParent(),
|
2019-05-09 16:21:42 +02:00
|
|
|
title,
|
|
|
|
|
question,
|
2013-05-23 15:51:50 +02:00
|
|
|
QMessageBox::Yes | QMessageBox::No) == QMessageBox::Yes;
|
|
|
|
|
}
|
|
|
|
|
|
2011-04-14 12:58:14 +02:00
|
|
|
/*!
|
2013-09-10 17:16:10 +02:00
|
|
|
Deletes \a session name from session list and the file from disk.
|
2011-04-14 12:58:14 +02:00
|
|
|
*/
|
2008-12-02 12:01:29 +01:00
|
|
|
bool SessionManager::deleteSession(const QString &session)
|
|
|
|
|
{
|
2023-02-14 15:47:22 +01:00
|
|
|
if (!sb_d->m_sessions.contains(session))
|
2008-12-02 12:01:29 +01:00
|
|
|
return false;
|
2023-02-14 15:47:22 +01:00
|
|
|
sb_d->m_sessions.removeOne(session);
|
|
|
|
|
sb_d->m_lastActiveTimes.remove(session);
|
2021-11-23 17:04:13 +01:00
|
|
|
emit instance()->sessionRemoved(session);
|
2022-11-09 16:07:09 +01:00
|
|
|
FilePath sessionFile = sessionNameToFileName(session);
|
|
|
|
|
if (sessionFile.exists())
|
|
|
|
|
return sessionFile.removeFile();
|
2008-12-02 12:01:29 +01:00
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
2019-05-09 16:21:42 +02:00
|
|
|
void SessionManager::deleteSessions(const QStringList &sessions)
|
|
|
|
|
{
|
|
|
|
|
for (const QString &session : sessions)
|
|
|
|
|
deleteSession(session);
|
|
|
|
|
}
|
|
|
|
|
|
2008-12-02 12:01:29 +01:00
|
|
|
bool SessionManager::cloneSession(const QString &original, const QString &clone)
|
|
|
|
|
{
|
2023-02-14 15:47:22 +01:00
|
|
|
if (!sb_d->m_sessions.contains(original))
|
2008-12-02 12:01:29 +01:00
|
|
|
return false;
|
|
|
|
|
|
2022-11-09 16:07:09 +01:00
|
|
|
FilePath sessionFile = sessionNameToFileName(original);
|
2008-12-02 12:01:29 +01:00
|
|
|
// If the file does not exist, we can still clone
|
2022-11-09 16:07:09 +01:00
|
|
|
if (!sessionFile.exists() || sessionFile.copyFile(sessionNameToFileName(clone))) {
|
2023-02-14 15:47:22 +01:00
|
|
|
sb_d->m_sessions.insert(1, clone);
|
|
|
|
|
sb_d->m_sessionDateTimes.insert(clone, sessionNameToFileName(clone).lastModified());
|
2008-12-02 12:01:29 +01:00
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
2013-09-05 11:46:07 +02:00
|
|
|
void SessionManagerPrivate::restoreValues(const PersistentSettingsReader &reader)
|
2012-02-10 12:46:33 +01:00
|
|
|
{
|
2013-09-05 11:46:07 +02:00
|
|
|
const QStringList keys = reader.restoreValue(QLatin1String("valueKeys")).toStringList();
|
2022-05-03 16:48:36 +02:00
|
|
|
for (const QString &key : keys) {
|
2012-02-10 12:46:33 +01:00
|
|
|
QVariant value = reader.restoreValue(QLatin1String("value-") + key);
|
|
|
|
|
m_values.insert(key, value);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2013-09-05 11:46:07 +02:00
|
|
|
void SessionManagerPrivate::restoreEditors(const PersistentSettingsReader &reader)
|
2012-02-10 12:46:33 +01:00
|
|
|
{
|
2013-09-05 11:46:07 +02:00
|
|
|
const QVariant editorsettings = reader.restoreValue(QLatin1String("EditorSettings"));
|
2012-02-10 12:46:33 +01:00
|
|
|
if (editorsettings.isValid()) {
|
2013-08-29 17:17:24 +02:00
|
|
|
EditorManager::restoreState(QByteArray::fromBase64(editorsettings.toByteArray()));
|
2013-07-08 16:26:29 +02:00
|
|
|
sessionLoadingProgress();
|
2012-02-10 12:46:33 +01:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2020-01-08 10:51:06 +01:00
|
|
|
/*!
|
|
|
|
|
Returns the last session that was opened by the user.
|
|
|
|
|
*/
|
2013-09-05 11:46:07 +02:00
|
|
|
QString SessionManager::lastSession()
|
2008-12-02 12:01:29 +01:00
|
|
|
{
|
2023-05-12 15:25:28 +02:00
|
|
|
return ICore::settings()->value(LASTSESSION_KEY).toString();
|
2020-01-07 12:34:31 +01:00
|
|
|
}
|
|
|
|
|
|
2020-01-08 10:51:06 +01:00
|
|
|
/*!
|
|
|
|
|
Returns the session that was active when Qt Creator was last closed, if any.
|
|
|
|
|
*/
|
2020-01-07 12:34:31 +01:00
|
|
|
QString SessionManager::startupSession()
|
|
|
|
|
{
|
2023-05-12 15:25:28 +02:00
|
|
|
return ICore::settings()->value(STARTUPSESSION_KEY).toString();
|
2008-12-02 12:01:29 +01:00
|
|
|
}
|
|
|
|
|
|
2018-05-25 17:06:42 +02:00
|
|
|
void SessionManager::markSessionFileDirty()
|
2010-03-03 15:54:32 +01:00
|
|
|
{
|
2023-02-14 15:47:22 +01:00
|
|
|
sb_d->m_virginSession = false;
|
2010-03-03 15:54:32 +01:00
|
|
|
}
|
2008-12-02 12:01:29 +01:00
|
|
|
|
2013-09-05 11:46:07 +02:00
|
|
|
void SessionManagerPrivate::sessionLoadingProgress()
|
2012-02-06 16:45:03 +01:00
|
|
|
{
|
2012-02-06 17:26:31 +01:00
|
|
|
m_future.setProgressValue(m_future.progressValue() + 1);
|
2012-02-06 16:45:03 +01:00
|
|
|
QCoreApplication::processEvents(QEventLoop::ExcludeUserInputEvents);
|
|
|
|
|
}
|
2012-02-23 17:50:30 +01:00
|
|
|
|
2013-09-05 11:46:07 +02:00
|
|
|
} // namespace ProjectExplorer
|