2016-08-12 13:38:53 +02:00
|
|
|
/****************************************************************************
|
|
|
|
|
**
|
|
|
|
|
** 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 "sessionmodel.h"
|
|
|
|
|
#include "session.h"
|
|
|
|
|
|
|
|
|
|
#include "sessiondialog.h"
|
|
|
|
|
|
2017-03-21 09:47:55 +01:00
|
|
|
#include <coreplugin/actionmanager/actionmanager.h>
|
|
|
|
|
#include <coreplugin/id.h>
|
|
|
|
|
|
2016-08-12 13:38:53 +02:00
|
|
|
#include <utils/algorithm.h>
|
|
|
|
|
#include <utils/fileutils.h>
|
|
|
|
|
#include <utils/stringutils.h>
|
|
|
|
|
|
|
|
|
|
#include <QFileInfo>
|
|
|
|
|
#include <QDir>
|
|
|
|
|
|
2017-03-21 09:47:55 +01:00
|
|
|
using namespace Core;
|
|
|
|
|
|
2016-08-12 13:38:53 +02:00
|
|
|
namespace ProjectExplorer {
|
|
|
|
|
namespace Internal {
|
|
|
|
|
|
|
|
|
|
SessionModel::SessionModel(QObject *parent)
|
2016-09-23 19:14:50 +02:00
|
|
|
: QAbstractTableModel(parent)
|
2016-08-12 13:38:53 +02:00
|
|
|
{
|
2019-10-01 16:47:45 +02:00
|
|
|
m_sortedSessions = SessionManager::sessions();
|
2016-08-12 13:38:53 +02:00
|
|
|
connect(SessionManager::instance(), &SessionManager::sessionLoaded,
|
|
|
|
|
this, &SessionModel::resetSessions);
|
|
|
|
|
}
|
|
|
|
|
|
2016-09-26 12:44:25 +02:00
|
|
|
int SessionModel::indexOfSession(const QString &session)
|
|
|
|
|
{
|
2019-10-01 16:47:45 +02:00
|
|
|
return m_sortedSessions.indexOf(session);
|
2016-09-26 12:44:25 +02:00
|
|
|
}
|
|
|
|
|
|
2019-05-09 16:21:42 +02:00
|
|
|
QString SessionModel::sessionAt(int row) const
|
2016-09-26 12:44:25 +02:00
|
|
|
{
|
2019-10-01 16:47:45 +02:00
|
|
|
return m_sortedSessions.value(row, QString());
|
2016-09-26 12:44:25 +02:00
|
|
|
}
|
|
|
|
|
|
2016-09-23 19:14:50 +02:00
|
|
|
QVariant SessionModel::headerData(int section, Qt::Orientation orientation, int role) const
|
|
|
|
|
{
|
|
|
|
|
QVariant result;
|
|
|
|
|
if (orientation == Qt::Horizontal) {
|
|
|
|
|
switch (role) {
|
|
|
|
|
case Qt::DisplayRole:
|
|
|
|
|
switch (section) {
|
|
|
|
|
case 0: result = tr("Session");
|
|
|
|
|
break;
|
|
|
|
|
case 1: result = tr("Last Modified");
|
|
|
|
|
break;
|
|
|
|
|
} // switch (section)
|
|
|
|
|
break;
|
|
|
|
|
} // switch (role) {
|
|
|
|
|
}
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int SessionModel::columnCount(const QModelIndex &) const
|
|
|
|
|
{
|
|
|
|
|
static int sectionCount = 0;
|
|
|
|
|
if (sectionCount == 0) {
|
|
|
|
|
// headers sections defining possible columns
|
|
|
|
|
while (!headerData(sectionCount, Qt::Horizontal, Qt::DisplayRole).isNull())
|
|
|
|
|
sectionCount++;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return sectionCount;
|
|
|
|
|
}
|
|
|
|
|
|
2016-08-12 13:38:53 +02:00
|
|
|
int SessionModel::rowCount(const QModelIndex &) const
|
|
|
|
|
{
|
2019-10-01 16:47:45 +02:00
|
|
|
return m_sortedSessions.count();
|
2016-08-12 13:38:53 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QStringList pathsToBaseNames(const QStringList &paths)
|
|
|
|
|
{
|
|
|
|
|
return Utils::transform(paths, [](const QString &path) {
|
|
|
|
|
return QFileInfo(path).completeBaseName();
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QStringList pathsWithTildeHomePath(const QStringList &paths)
|
|
|
|
|
{
|
|
|
|
|
return Utils::transform(paths, [](const QString &path) {
|
|
|
|
|
return Utils::withTildeHomePath(QDir::toNativeSeparators(path));
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QVariant SessionModel::data(const QModelIndex &index, int role) const
|
|
|
|
|
{
|
2016-08-12 13:41:48 +02:00
|
|
|
QVariant result;
|
|
|
|
|
if (index.isValid()) {
|
2019-10-01 16:47:45 +02:00
|
|
|
QString sessionName = m_sortedSessions.at(index.row());
|
2016-08-12 13:41:48 +02:00
|
|
|
|
|
|
|
|
switch (role) {
|
|
|
|
|
case Qt::DisplayRole:
|
|
|
|
|
switch (index.column()) {
|
|
|
|
|
case 0: result = sessionName;
|
|
|
|
|
break;
|
2016-09-23 19:14:50 +02:00
|
|
|
case 1: result = SessionManager::sessionDateTime(sessionName);
|
|
|
|
|
break;
|
2016-08-12 13:41:48 +02:00
|
|
|
} // switch (section)
|
|
|
|
|
break;
|
2016-09-23 20:45:03 +02:00
|
|
|
case Qt::FontRole: {
|
|
|
|
|
QFont font;
|
|
|
|
|
if (SessionManager::isDefaultSession(sessionName))
|
|
|
|
|
font.setItalic(true);
|
|
|
|
|
else
|
|
|
|
|
font.setItalic(false);
|
|
|
|
|
if (SessionManager::activeSession() == sessionName && !SessionManager::isDefaultVirgin())
|
|
|
|
|
font.setBold(true);
|
|
|
|
|
else
|
|
|
|
|
font.setBold(false);
|
|
|
|
|
result = font;
|
|
|
|
|
} break;
|
2016-08-12 13:41:48 +02:00
|
|
|
case DefaultSessionRole:
|
|
|
|
|
result = SessionManager::isDefaultSession(sessionName);
|
|
|
|
|
break;
|
|
|
|
|
case LastSessionRole:
|
|
|
|
|
result = SessionManager::lastSession() == sessionName;
|
|
|
|
|
break;
|
|
|
|
|
case ActiveSessionRole:
|
|
|
|
|
result = SessionManager::activeSession() == sessionName;
|
|
|
|
|
break;
|
|
|
|
|
case ProjectsPathRole:
|
|
|
|
|
result = pathsWithTildeHomePath(SessionManager::projectsForSessionName(sessionName));
|
|
|
|
|
break;
|
|
|
|
|
case ProjectsDisplayRole:
|
|
|
|
|
result = pathsToBaseNames(SessionManager::projectsForSessionName(sessionName));
|
|
|
|
|
break;
|
2017-03-21 09:47:55 +01:00
|
|
|
case ShortcutRole: {
|
|
|
|
|
const Id sessionBase = SESSION_BASE_ID;
|
|
|
|
|
if (Command *cmd = ActionManager::command(sessionBase.withSuffix(index.row() + 1)))
|
|
|
|
|
result = cmd->keySequence().toString(QKeySequence::NativeText);
|
|
|
|
|
} break;
|
2016-08-12 13:41:48 +02:00
|
|
|
} // switch (role)
|
2016-08-12 13:38:53 +02:00
|
|
|
}
|
2016-08-12 13:41:48 +02:00
|
|
|
|
|
|
|
|
return result;
|
2016-08-12 13:38:53 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QHash<int, QByteArray> SessionModel::roleNames() const
|
|
|
|
|
{
|
|
|
|
|
static QHash<int, QByteArray> extraRoles{
|
|
|
|
|
{Qt::DisplayRole, "sessionName"},
|
|
|
|
|
{DefaultSessionRole, "defaultSession"},
|
|
|
|
|
{ActiveSessionRole, "activeSession"},
|
|
|
|
|
{LastSessionRole, "lastSession"},
|
|
|
|
|
{ProjectsPathRole, "projectsPath"},
|
|
|
|
|
{ProjectsDisplayRole, "projectsName"}
|
|
|
|
|
};
|
2016-09-23 19:14:50 +02:00
|
|
|
return QAbstractTableModel::roleNames().unite(extraRoles);
|
2016-08-12 13:38:53 +02:00
|
|
|
}
|
|
|
|
|
|
2019-10-01 16:47:45 +02:00
|
|
|
void SessionModel::sort(int column, Qt::SortOrder order)
|
|
|
|
|
{
|
|
|
|
|
beginResetModel();
|
|
|
|
|
const auto cmp = [column, order](const QString &s1, const QString &s2) {
|
|
|
|
|
bool isLess;
|
|
|
|
|
if (column == 0)
|
|
|
|
|
isLess = s1 < s2;
|
|
|
|
|
else
|
|
|
|
|
isLess = SessionManager::sessionDateTime(s1) < SessionManager::sessionDateTime(s2);
|
|
|
|
|
if (order == Qt::DescendingOrder)
|
|
|
|
|
isLess = !isLess;
|
|
|
|
|
return isLess;
|
|
|
|
|
};
|
|
|
|
|
Utils::sort(m_sortedSessions, cmp);
|
|
|
|
|
endResetModel();
|
|
|
|
|
}
|
|
|
|
|
|
2016-08-12 13:38:53 +02:00
|
|
|
bool SessionModel::isDefaultVirgin() const
|
|
|
|
|
{
|
|
|
|
|
return SessionManager::isDefaultVirgin();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void SessionModel::resetSessions()
|
|
|
|
|
{
|
|
|
|
|
beginResetModel();
|
2019-10-01 16:47:45 +02:00
|
|
|
m_sortedSessions = SessionManager::sessions();
|
2016-08-12 13:38:53 +02:00
|
|
|
endResetModel();
|
|
|
|
|
}
|
|
|
|
|
|
2018-01-23 19:02:46 +01:00
|
|
|
void SessionModel::newSession(QWidget *parent)
|
2016-08-12 13:38:53 +02:00
|
|
|
{
|
2018-01-23 19:02:46 +01:00
|
|
|
SessionNameInputDialog sessionInputDialog(parent);
|
2017-05-29 17:46:59 +02:00
|
|
|
sessionInputDialog.setWindowTitle(tr("New Session Name"));
|
|
|
|
|
sessionInputDialog.setActionText(tr("&Create"), tr("Create and &Open"));
|
|
|
|
|
|
|
|
|
|
runSessionNameInputDialog(&sessionInputDialog, [](const QString &newName) {
|
2016-09-05 10:07:07 +02:00
|
|
|
SessionManager::createSession(newName);
|
|
|
|
|
});
|
2016-09-01 14:57:32 +02:00
|
|
|
}
|
2016-08-12 13:38:53 +02:00
|
|
|
|
2018-01-23 19:02:46 +01:00
|
|
|
void SessionModel::cloneSession(QWidget *parent, const QString &session)
|
2016-09-01 14:57:32 +02:00
|
|
|
{
|
2018-01-23 19:02:46 +01:00
|
|
|
SessionNameInputDialog sessionInputDialog(parent);
|
2017-05-29 17:46:59 +02:00
|
|
|
sessionInputDialog.setWindowTitle(tr("New Session Name"));
|
|
|
|
|
sessionInputDialog.setActionText(tr("&Clone"), tr("Clone and &Open"));
|
|
|
|
|
sessionInputDialog.setValue(session + " (2)");
|
|
|
|
|
|
|
|
|
|
runSessionNameInputDialog(&sessionInputDialog, [session](const QString &newName) {
|
2016-09-01 14:57:32 +02:00
|
|
|
SessionManager::cloneSession(session, newName);
|
|
|
|
|
});
|
2016-08-12 13:38:53 +02:00
|
|
|
}
|
|
|
|
|
|
2019-05-09 16:21:42 +02:00
|
|
|
void SessionModel::deleteSessions(const QStringList &sessions)
|
2016-08-12 13:38:53 +02:00
|
|
|
{
|
2019-05-09 16:21:42 +02:00
|
|
|
if (!SessionManager::confirmSessionDelete(sessions))
|
2016-08-12 13:38:53 +02:00
|
|
|
return;
|
|
|
|
|
beginResetModel();
|
2019-05-09 16:21:42 +02:00
|
|
|
SessionManager::deleteSessions(sessions);
|
2016-08-12 13:38:53 +02:00
|
|
|
endResetModel();
|
|
|
|
|
}
|
|
|
|
|
|
2018-01-23 19:02:46 +01:00
|
|
|
void SessionModel::renameSession(QWidget *parent, const QString &session)
|
2016-09-01 14:57:32 +02:00
|
|
|
{
|
2018-01-23 19:02:46 +01:00
|
|
|
SessionNameInputDialog sessionInputDialog(parent);
|
2017-05-29 17:46:59 +02:00
|
|
|
sessionInputDialog.setWindowTitle(tr("Rename Session"));
|
|
|
|
|
sessionInputDialog.setActionText(tr("&Rename"), tr("Rename and &Open"));
|
|
|
|
|
sessionInputDialog.setValue(session);
|
|
|
|
|
|
|
|
|
|
runSessionNameInputDialog(&sessionInputDialog, [session](const QString &newName) {
|
2016-09-28 12:56:04 +02:00
|
|
|
SessionManager::renameSession(session, newName);
|
2016-09-01 14:57:32 +02:00
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void SessionModel::switchToSession(const QString &session)
|
|
|
|
|
{
|
|
|
|
|
SessionManager::loadSession(session);
|
|
|
|
|
emit sessionSwitched();
|
|
|
|
|
}
|
|
|
|
|
|
2017-05-29 17:46:59 +02:00
|
|
|
void SessionModel::runSessionNameInputDialog(SessionNameInputDialog *sessionInputDialog, std::function<void(const QString &)> createSession)
|
2016-08-12 13:38:53 +02:00
|
|
|
{
|
2017-05-29 17:46:59 +02:00
|
|
|
if (sessionInputDialog->exec() == QDialog::Accepted) {
|
|
|
|
|
QString newSession = sessionInputDialog->value();
|
2016-08-12 13:38:53 +02:00
|
|
|
if (newSession.isEmpty() || SessionManager::sessions().contains(newSession))
|
|
|
|
|
return;
|
|
|
|
|
beginResetModel();
|
2016-09-01 14:57:32 +02:00
|
|
|
createSession(newSession);
|
2019-10-01 16:47:45 +02:00
|
|
|
m_sortedSessions = SessionManager::sessions();
|
2016-08-12 13:38:53 +02:00
|
|
|
endResetModel();
|
|
|
|
|
|
2017-05-29 17:46:59 +02:00
|
|
|
if (sessionInputDialog->isSwitchToRequested())
|
2016-09-01 14:57:32 +02:00
|
|
|
switchToSession(newSession);
|
2016-09-27 12:40:46 +02:00
|
|
|
emit sessionCreated(newSession);
|
2016-08-12 13:38:53 +02:00
|
|
|
}
|
|
|
|
|
}
|
2016-09-23 19:14:50 +02:00
|
|
|
|
2016-08-12 13:38:53 +02:00
|
|
|
} // namespace Internal
|
|
|
|
|
} // namespace ProjectExplorer
|