1
mirror of https://invent.kde.org/network/falkon.git synced 2024-11-15 03:22:11 +01:00
falkonOfficial/src/lib/session/sessionmanager.cpp

424 lines
14 KiB
C++

/* ============================================================
* Falkon - Qt web browser
* Copyright (C) 2017 Razi Alavizadeh <s.r.alavizadeh@gmail.com>
* Copyright (C) 2018 David Rosca <nowrep@gmail.com>
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
* ============================================================ */
#include "browserwindow.h"
#include "datapaths.h"
#include "mainapplication.h"
#include "restoremanager.h"
#include "sessionmanager.h"
#include "sessionmanagerdialog.h"
#include "settings.h"
#include <QAction>
#include <QComboBox>
#include <QDateTime>
#include <QDialogButtonBox>
#include <QDir>
#include <QFileSystemWatcher>
#include <QInputDialog>
#include <QLabel>
#include <QMenu>
#include <QMessageBox>
#include <QVBoxLayout>
#include <QSaveFile>
SessionManager::SessionManager(QObject* parent)
: QObject(parent)
, m_firstBackupSession(DataPaths::currentProfilePath() + QL1S("/session.dat.old"))
, m_secondBackupSession(DataPaths::currentProfilePath() + QL1S("/session.dat.old1"))
{
QFileSystemWatcher* sessionFilesWatcher = new QFileSystemWatcher({DataPaths::path(DataPaths::Sessions)}, this);
connect(sessionFilesWatcher, SIGNAL(directoryChanged(QString)), this, SLOT(sessionsDirectoryChanged()));
connect(sessionFilesWatcher, &QFileSystemWatcher::directoryChanged, this, &SessionManager::sessionsMetaDataChanged);
loadSettings();
}
void SessionManager::aboutToShowSessionsMenu()
{
QMenu* menu = qobject_cast<QMenu*>(sender());
menu->clear();
QActionGroup *group = new QActionGroup(menu);
const auto sessions = sessionMetaData(/*withBackups*/ false);
for (const SessionManager::SessionMetaData &metaData : sessions) {
QAction* action = menu->addAction(metaData.name);
action->setCheckable(true);
action->setChecked(metaData.isActive);
group->addAction(action);
connect(action, &QAction::triggered, this, [=]() {
switchToSession(metaData.filePath);
});
}
}
void SessionManager::sessionsDirectoryChanged()
{
m_sessionsMetaDataList.clear();
}
void SessionManager::openSession(QString sessionFilePath, SessionFlags flags)
{
if (sessionFilePath.isEmpty()) {
QAction* action = qobject_cast<QAction*>(sender());
if (!action)
return;
sessionFilePath = action->data().toString();
}
if (isActive(sessionFilePath)) {
return;
}
RestoreData sessionData;
RestoreManager::createFromFile(sessionFilePath, sessionData);
if (!sessionData.isValid())
return;
BrowserWindow* window = mApp->getWindow();
if (flags.testFlag(SwitchSession)) {
writeCurrentSession(m_lastActiveSessionPath);
window = mApp->createWindow(Qz::BW_OtherRestoredWindow);
for (BrowserWindow* win : mApp->windows()) {
if (win != window)
win->close();
}
if (!flags.testFlag(ReplaceSession)) {
m_lastActiveSessionPath = QFileInfo(sessionFilePath).canonicalFilePath();
m_sessionsMetaDataList.clear();
}
}
mApp->openSession(window, sessionData);
}
void SessionManager::renameSession(QString sessionFilePath, SessionFlags flags)
{
if (sessionFilePath.isEmpty()) {
QAction* action = qobject_cast<QAction*>(sender());
if (!action)
return;
sessionFilePath = action->data().toString();
}
bool ok;
const QString suggestedName = QFileInfo(sessionFilePath).completeBaseName() + (flags.testFlag(CloneSession) ? tr("_cloned") : tr("_renamed"));
QString newName = QInputDialog::getText(mApp->activeWindow(), (flags.testFlag(CloneSession) ? tr("Clone Session") : tr("Rename Session")),
tr("Please enter a new name:"), QLineEdit::Normal,
suggestedName, &ok);
if (!ok)
return;
const QString newSessionPath = QString("%1/%2.dat").arg(DataPaths::path(DataPaths::Sessions)).arg(newName);
if (QFile::exists(newSessionPath)) {
QMessageBox::information(mApp->activeWindow(), tr("Error!"), tr("The session file \"%1\" exists. Please enter another name.").arg(newName));
renameSession(sessionFilePath, flags);
return;
}
if (flags.testFlag(CloneSession)) {
if (!QFile::copy(sessionFilePath, newSessionPath)) {
QMessageBox::information(mApp->activeWindow(), tr("Error!"), tr("An error occurred when cloning session file."));
return;
}
} else {
if (!QFile::rename(sessionFilePath, newSessionPath)) {
QMessageBox::information(mApp->activeWindow(), tr("Error!"), tr("An error occurred when renaming session file."));
return;
}
if (isActive(sessionFilePath)) {
m_lastActiveSessionPath = newSessionPath;
m_sessionsMetaDataList.clear();
}
}
}
void SessionManager::saveSession()
{
bool ok;
QString sessionName = QInputDialog::getText(mApp->activeWindow(), tr("Save Session"),
tr("Please enter a name to save session:"), QLineEdit::Normal,
tr("Saved Session (%1)").arg(QDateTime::currentDateTime().toString("dd MMM yyyy HH-mm-ss")), &ok);
if (!ok)
return;
const QString filePath = QString("%1/%2.dat").arg(DataPaths::path(DataPaths::Sessions)).arg(sessionName);
if (QFile::exists(filePath)) {
QMessageBox::information(mApp->activeWindow(), tr("Error!"), tr("The session file \"%1\" exists. Please enter another name.").arg(sessionName));
saveSession();
return;
}
writeCurrentSession(filePath);
}
void SessionManager::replaceSession(const QString &filePath)
{
QMessageBox::StandardButton result = QMessageBox::information(mApp->activeWindow(), tr("Restore Backup"), tr("Are you sure you want to replace current session?"),
QMessageBox::Yes | QMessageBox::No);
if (result == QMessageBox::Yes) {
openSession(filePath, ReplaceSession);
}
}
void SessionManager::switchToSession(const QString &filePath)
{
openSession(filePath, SwitchSession);
}
void SessionManager::cloneSession(const QString &filePath)
{
renameSession(filePath, CloneSession);
}
void SessionManager::deleteSession(const QString &filePath)
{
QMessageBox::StandardButton result = QMessageBox::information(mApp->activeWindow(), tr("Delete Session"), tr("Are you sure you want to delete session '%1'?")
.arg(QFileInfo(filePath).completeBaseName()), QMessageBox::Yes | QMessageBox::No);
if (result == QMessageBox::Yes) {
QFile::remove(filePath);
}
}
void SessionManager::newSession()
{
bool ok;
QString sessionName = QInputDialog::getText(mApp->activeWindow(), tr("New Session"),
tr("Please enter a name to create new session:"), QLineEdit::Normal,
tr("New Session (%1)").arg(QDateTime::currentDateTime().toString("dd MMM yyyy HH-mm-ss")), &ok);
if (!ok)
return;
const QString filePath = QString("%1/%2.dat").arg(DataPaths::path(DataPaths::Sessions)).arg(sessionName);
if (QFile::exists(filePath)) {
QMessageBox::information(mApp->activeWindow(), tr("Error!"), tr("The session file \"%1\" exists. Please enter another name.").arg(sessionName));
newSession();
return;
}
writeCurrentSession(m_lastActiveSessionPath);
BrowserWindow* window = mApp->createWindow(Qz::BW_NewWindow);
for (BrowserWindow* win : mApp->windows()) {
if (win != window)
win->close();
}
m_lastActiveSessionPath = filePath;
autoSaveLastSession();
}
QList<SessionManager::SessionMetaData> SessionManager::sessionMetaData(bool withBackups)
{
fillSessionsMetaDataListIfNeeded();
auto out = m_sessionsMetaDataList;
if (withBackups && QFile::exists(m_firstBackupSession)) {
SessionMetaData data;
data.name = tr("Backup 1");
data.filePath = m_firstBackupSession;
data.isBackup = true;
out.append(data);
}
if (withBackups && QFile::exists(m_secondBackupSession)) {
SessionMetaData data;
data.name = tr("Backup 2");
data.filePath = m_secondBackupSession;
data.isBackup = true;
out.append(data);
}
return out;
}
bool SessionManager::isActive(const QString &filePath) const
{
return QFileInfo(filePath) == QFileInfo(m_lastActiveSessionPath);
}
bool SessionManager::isActive(const QFileInfo &fileInfo) const
{
return fileInfo == QFileInfo(m_lastActiveSessionPath);
}
void SessionManager::fillSessionsMetaDataListIfNeeded()
{
if (!m_sessionsMetaDataList.isEmpty())
return;
QDir dir(DataPaths::path(DataPaths::Sessions));
const QFileInfoList sessionFiles = QFileInfoList() << QFileInfo(defaultSessionPath()) << dir.entryInfoList({QSL("*.*")}, QDir::Files, QDir::Time);
QStringList fileNames;
for (int i = 0; i < sessionFiles.size(); ++i) {
const QFileInfo &fileInfo = sessionFiles.at(i);
if (!RestoreManager::validateFile(fileInfo.absoluteFilePath()))
continue;
SessionMetaData metaData;
metaData.name = fileInfo.completeBaseName();
if (fileInfo == QFileInfo(defaultSessionPath())) {
metaData.name = tr("Default Session");
metaData.isDefault = true;
} else if (fileNames.contains(fileInfo.completeBaseName())) {
metaData.name = fileInfo.fileName();
} else {
metaData.name = fileInfo.completeBaseName();
}
if (isActive(fileInfo)) {
metaData.isActive = true;
}
fileNames << metaData.name;
metaData.filePath = fileInfo.canonicalFilePath();
m_sessionsMetaDataList << metaData;
}
}
void SessionManager::loadSettings()
{
QDir sessionsDir(DataPaths::path(DataPaths::Sessions));
Settings settings;
settings.beginGroup("Web-Browser-Settings");
m_lastActiveSessionPath = settings.value("lastActiveSessionPath", defaultSessionPath()).toString();
settings.endGroup();
if (QDir::isRelativePath(m_lastActiveSessionPath)) {
m_lastActiveSessionPath = sessionsDir.absoluteFilePath(m_lastActiveSessionPath);
}
// Fallback to default session
if (!RestoreManager::validateFile(m_lastActiveSessionPath))
m_lastActiveSessionPath = defaultSessionPath();
}
void SessionManager::saveSettings()
{
QDir sessionsDir(DataPaths::path(DataPaths::Sessions));
Settings settings;
settings.beginGroup("Web-Browser-Settings");
settings.setValue("lastActiveSessionPath", sessionsDir.relativeFilePath(m_lastActiveSessionPath));
settings.endGroup();
}
QString SessionManager::defaultSessionPath()
{
return DataPaths::currentProfilePath() + QL1S("/session.dat");
}
QString SessionManager::lastActiveSessionPath() const
{
return m_lastActiveSessionPath;
}
void SessionManager::backupSavedSessions()
{
if (!QFile::exists(m_lastActiveSessionPath)) {
return;
}
if (QFile::exists(m_firstBackupSession)) {
QFile::remove(m_secondBackupSession);
QFile::copy(m_firstBackupSession, m_secondBackupSession);
}
QFile::remove(m_firstBackupSession);
QFile::copy(m_lastActiveSessionPath, m_firstBackupSession);
}
void SessionManager::writeCurrentSession(const QString &filePath)
{
QSaveFile file(filePath);
if (!file.open(QIODevice::WriteOnly) || file.write(mApp->saveState()) == -1) {
qWarning() << "Error! can not write the current session file: " << filePath << file.errorString();
return;
}
file.commit();
}
void SessionManager::openSessionManagerDialog()
{
SessionManagerDialog *dialog = new SessionManagerDialog(mApp->getWindow());
dialog->open();
}
void SessionManager::autoSaveLastSession()
{
if (mApp->isPrivate() || mApp->windowCount() == 0) {
return;
}
saveSettings();
writeCurrentSession(m_lastActiveSessionPath);
}
QString SessionManager::askSessionFromUser()
{
fillSessionsMetaDataListIfNeeded();
QDialog dialog(mApp->getWindow(), Qt::WindowStaysOnTopHint);
QLabel label(tr("Please select the startup session:"), &dialog);
QComboBox comboBox(&dialog);
QDialogButtonBox buttonBox(QDialogButtonBox::Ok | QDialogButtonBox::Cancel, &dialog);
connect(&buttonBox, &QDialogButtonBox::accepted, &dialog, &QDialog::accept);
connect(&buttonBox, &QDialogButtonBox::rejected, &dialog, &QDialog::reject);
QVBoxLayout layout;
layout.addWidget(&label);
layout.addWidget(&comboBox);
layout.addWidget(&buttonBox);
dialog.setLayout(&layout);
const QFileInfo lastActiveSessionFileInfo(m_lastActiveSessionPath);
for (const SessionMetaData &metaData : m_sessionsMetaDataList) {
if (QFileInfo(metaData.filePath) != lastActiveSessionFileInfo) {
comboBox.addItem(metaData.name, metaData.filePath);
}
else {
comboBox.insertItem(0, tr("%1 (last session)").arg(metaData.name), metaData.filePath);
}
}
comboBox.setCurrentIndex(0);
if (dialog.exec() == QDialog::Accepted) {
m_lastActiveSessionPath = comboBox.currentData().toString();
}
return m_lastActiveSessionPath;
}