/* This file is part of the KDE project Copyright (C) 2005 Christoph Cullmann <cullmann@kde.org> This library is free software; you can redistribute it and/or modify it under the terms of the GNU Library General Public License version 2 as published by the Free Software Foundation. This library 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 Library General Public License for more details. You should have received a copy of the GNU Library General Public License along with this library; see the file COPYING.LIB. If not, write to the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. */ #include "katesession.h" #include "kateapp.h" #include "katemainwindow.h" #include "katedocmanager.h" #include <kstandarddirs.h> #include <tdelocale.h> #include <kdebug.h> #include <kdirwatch.h> #include <kinputdialog.h> #include <kiconloader.h> #include <tdemessagebox.h> #include <kmdcodec.h> #include <kstdguiitem.h> #include <kpushbutton.h> #include <tdepopupmenu.h> #include <tqdir.h> #include <tqfile.h> #include <tqlabel.h> #include <tqlayout.h> #include <tqvbox.h> #include <tqhbox.h> #include <tqdatetime.h> #include <tqmap.h> #include <unistd.h> #include <time.h> // FIXME general: need to keep doc list and current session's m_documents in synchro // all the time (doc open, doc closed, doc renamed). // To be done when doc list software is developed // String constants namespace { // Kate session const char *KS_COUNT = "Count"; const char *KS_DOCCOUNT = "Document count"; const char *KS_DOCLIST = "Document list"; const char *KS_GENERAL = "General"; const char *KS_NAME = "Name"; const char *KS_OPENDOC = "Open Documents"; const char *KS_READONLY = "ReadOnly"; const char *KS_OPEN_MAINWINDOWS = "Open MainWindows"; const char *KS_UNNAMED = "Unnamed"; // Kate session manager const char *KSM_DIR = "kate/sessions"; const char *KSM_FILE = "sessions.list"; const char *KSM_SESSIONS_COUNT = "Sessions count"; const char *KSM_LAST_SESSION_ID = "Last session id"; const char *KSM_SESSIONS_LIST = "Sessions list"; // Kate app const char *KAPP_GENERAL = "General"; const char *KAPP_LAST_SESSION = "Last Session"; const char *KAPP_STARTUP_SESSION = "Startup Session"; const char *KAPP_NEW = "new"; const char *KAPP_LAST = "last"; const char *KAPP_MANUAL = "manual"; const char *KAPP_SESSION_EXIT = "Session Exit"; const char *KAPP_DISCARD = "discard"; const char *KAPP_SAVE = "save"; const char *KAPP_ASK = "ask"; } //BEGIN Kate session KateSession::KateSession(const KateSessionManager &manager, const TQString &sessionName, const TQString &fileName) : m_manager(manager), m_sessionName(sessionName), m_filename(fileName), m_readOnly(false), m_documents(), m_config(NULL) { load(false); } //------------------------------------ KateSession::KateSession(const KateSession &session, const TQString &newSessionName) : m_manager(session.m_manager), m_sessionName(newSessionName), m_filename(), m_readOnly(false), m_documents(session.m_documents), m_config(NULL) { createFilename(); if (session.m_config) { m_config = new KSimpleConfig(m_filename); session.m_config->copyTo(m_filename, m_config); m_config->sync(); } } //------------------------------------ KateSession::~KateSession() { if (m_config) { delete m_config; } } //------------------------------------ void KateSession::setSessionName(const TQString &sessionName) { m_sessionName = sessionName.isEmpty() ? i18n(KS_UNNAMED) : sessionName; } //------------------------------------ bool KateSession::isStillVolatile() const { return m_filename.isEmpty() && m_sessionName == i18n(KS_UNNAMED); } //------------------------------------ void KateSession::load(bool includeGUIInfo) { if (m_config) { delete m_config; } if (TDEGlobal::dirs()->exists(m_filename)) { // Create config object if the session file already exists m_config = new KSimpleConfig(m_filename, m_readOnly); m_config->setGroup(KS_GENERAL); // Session general properties m_sessionName = m_config->readEntry(KS_NAME, i18n(KS_UNNAMED)); m_readOnly = m_config->readBoolEntry(KS_READONLY, false); // Document list if (m_config->hasGroup(KS_DOCLIST)) { // Read new style document list (from TDE R14.1.0) m_config->setGroup(KS_DOCLIST); int docCount = m_config->readNumEntry(KS_DOCCOUNT, 0); for (int i = 0; i < docCount; ++i) { TQString urlStr = m_config->readEntry(TQString("URL_%1").arg(i)); if (!urlStr.isEmpty()) { // Filter out empty URLs m_documents.append(urlStr); } } } else { // Create document list from old session configuration // to effortlessly import existing sessions m_config->setGroup(KS_OPENDOC); int docCount = m_config->readNumEntry(KS_COUNT, 0); for (int i = 0; i < docCount; ++i) { m_config->setGroup(TQString("Document %1").arg(i)); TQString urlStr = m_config->readEntry("URL"); if (!urlStr.isEmpty()) { // Filter out empty URLs m_documents.append(urlStr); } } } } else { m_filename = TQString::null; } if (m_sessionName.isEmpty()) { m_sessionName = i18n(KS_UNNAMED); } // Update all current documents if necessary if (includeGUIInfo) { activate(); } } //------------------------------------ void KateSession::save(bool saveGUIInfo, bool setReadOnly) { if (m_readOnly && !setReadOnly) { return; } // create a new session filename if needed if (m_filename.isEmpty()) { createFilename(); } // save session config info if (!m_config) { m_config = new KSimpleConfig(m_filename); } if (m_config->hasGroup(KS_GENERAL)) { m_config->deleteGroup(KS_GENERAL); } m_config->setGroup(KS_GENERAL); m_config->writeEntry(KS_NAME, m_sessionName); m_config->writeEntry(KS_READONLY, m_readOnly); if (m_config->hasGroup(KS_DOCLIST)) { m_config->deleteGroup(KS_DOCLIST); } m_config->setGroup(KS_DOCLIST); m_config->writeEntry(KS_DOCCOUNT, m_documents.count()); for (int i = 0; i < (int)m_documents.count(); ++i) { m_config->writeEntry(TQString("URL_%1").arg(i), m_documents[i]); } // save GUI elements info if (saveGUIInfo) { KateDocManager::self()->saveDocumentList(m_config); // save main windows info int mwCount = KateApp::self()->mainWindows(); m_config->setGroup(KS_OPEN_MAINWINDOWS); m_config->writeEntry(KS_COUNT, mwCount); for (int i = 0; i < mwCount; ++i) { m_config->setGroup(TQString("MainWindow%1").arg(i)); KateApp::self()->mainWindow(i)->saveProperties(m_config); } } m_config->sync(); } //------------------------------------ void KateSession::activate() { if (KateDocManager::self()->documents() > 0) { KateDocManager::self()->closeAllDocuments(); } Kate::Document::setOpenErrorDialogsActivated(false); if (m_config) { KateApp::self()->documentManager()->restoreDocumentList(m_config); // load main windows info, if it exists if (m_config->hasGroup(KS_OPEN_MAINWINDOWS)) { m_config->setGroup(KS_OPEN_MAINWINDOWS); int mwCount = m_config->readUnsignedNumEntry(KS_COUNT, 1); for (int i = 0; i < mwCount; ++i) { if (i >= (int)KateApp::self()->mainWindows()) { KateApp::self()->newMainWindow(m_config, TQString("MainWindow%1").arg(i)); } else { m_config->setGroup(TQString("MainWindow%1").arg(i)); KateApp::self()->mainWindow(i)->readProperties(m_config); } } } } Kate::Document::setOpenErrorDialogsActivated(true); } //------------------------------------ void KateSession::createFilename() { // create a new session filename if needed if (!m_filename.isEmpty()) { return; } int s = time(0); TQCString tname; TQString tmpName; while (true) { tname.setNum(s++); KMD5 md5(tname); tmpName = m_manager.getBaseDir() + TQString("%1.katesession").arg(md5.hexDigest().data()); if (!TDEGlobal::dirs()->exists(tmpName)) { m_filename = tmpName; break; } } } //END Kate session //BEGIN KateSessionManager //------------------------------------ KateSessionManager *KateSessionManager::ksm_instance = NULL; //------------------------------------ KateSessionManager* KateSessionManager::self() { if (!KateSessionManager::ksm_instance) { KateSessionManager::ksm_instance = new KateSessionManager(); } return KateSessionManager::ksm_instance; } //------------------------------------ KateSessionManager::KateSessionManager() : m_baseDir(locateLocal("data", KSM_DIR)+"/"), m_configFile(m_baseDir + KSM_FILE), m_activeSessionId(INVALID_SESSION), m_lastSessionId(INVALID_SESSION), m_sessions(), m_config(NULL), m_startupOption(STARTUP_NEW), m_switchOption(SWITCH_DISCARD) { // Session startup and switch options updateSessionOptions(SO_ALL); // Sessions configuration m_sessions.setAutoDelete(true); int sessionsCount = 0; if (TDEGlobal::dirs()->exists(m_configFile)) { // Read new style configuration (from TDE R14.1.0) m_config = new KSimpleConfig(m_configFile); m_config->setGroup(KSM_SESSIONS_LIST); sessionsCount = m_config->readNumEntry(KSM_SESSIONS_COUNT, 0); m_lastSessionId = m_config->readNumEntry(KSM_LAST_SESSION_ID, INVALID_SESSION); for (int i = 0; i < sessionsCount; ++i) { TQString urlStr = m_config->readEntry(TQString("URL_%1").arg(i)); if (!urlStr.isEmpty() && TDEGlobal::dirs()->exists(urlStr)) { // Filter out empty URLs or non existing sessions m_sessions.append(new KateSession(*this, TQString::null, urlStr)); } } } else { // Create sessions list from session files // to effortlessly import existing sessions TQDir sessionDir(m_baseDir, "*.katesession"); for (unsigned int i = 0; i < sessionDir.count(); ++i) { m_sessions.append(new KateSession(*this, TQString::null, m_baseDir+sessionDir[i])); } } sessionsCount = (int)m_sessions.count(); if (sessionsCount == 0) // In the worst case, there is no valid session at all { m_sessions.append(new KateSession(*this, TQString::null, TQString::null)); } if (m_lastSessionId < 0 || m_lastSessionId >= (int)m_sessions.count()) { m_lastSessionId = 0; // Invalid last session was detected. Use first in the list } //NOTE do not activate any session in the KateSessionManager costructor // since Kate's main window may not be ready yet. The initial session // will be activated by KateApp::startupKate() or KateApp::restoreKate() } //------------------------------------ KateSessionManager::~KateSessionManager() { if (m_config) { delete m_config; } m_sessions.clear(); } //------------------------------------ void KateSessionManager::updateSessionOptions(int optionType) { // Session startup and switch options TDEConfig *kateCfg = KateApp::self()->config(); kateCfg->setGroup(KAPP_GENERAL); if (optionType == SO_STARTUP || optionType == SO_ALL) { if (kateCfg->hasKey(KAPP_LAST_SESSION)) { // Delete no longer used entry (pre R14.1.0) kateCfg->deleteEntry(KAPP_LAST_SESSION); } TQString startupOption(kateCfg->readEntry(KAPP_STARTUP_SESSION, KAPP_MANUAL)); if (startupOption == KAPP_LAST) { m_startupOption = STARTUP_LAST; } else if (startupOption == KAPP_NEW) { m_startupOption = STARTUP_NEW; } else // startupOption == "manual" { m_startupOption = STARTUP_MANUAL; } } if (optionType == SO_SWITCH || optionType == SO_ALL) { TQString switchOption(kateCfg->readEntry(KAPP_SESSION_EXIT, KAPP_ASK)); if (switchOption == KAPP_DISCARD) { m_switchOption = SWITCH_DISCARD; } else if (switchOption == KAPP_SAVE) { m_switchOption = SWITCH_SAVE; } else // switchOption == "ask" { m_switchOption = SWITCH_ASK; } } } //------------------------------------ void KateSessionManager::saveSessionOptions(int optionType) { TDEConfig *kateCfg = KateApp::self()->config(); kateCfg->setGroup(KAPP_GENERAL); if (optionType == SO_STARTUP || optionType == SO_ALL) { if (m_startupOption == STARTUP_LAST) { kateCfg->writeEntry(KAPP_STARTUP_SESSION, KAPP_LAST); } else if (m_startupOption == STARTUP_NEW) { kateCfg->writeEntry(KAPP_STARTUP_SESSION, KAPP_NEW); } else // m_startupOption == STARTUP_MANUAL { kateCfg->writeEntry(KAPP_STARTUP_SESSION, KAPP_MANUAL); } } if (optionType == SO_SWITCH || optionType == SO_ALL) { if (m_switchOption == SWITCH_DISCARD) { kateCfg->writeEntry(KAPP_SESSION_EXIT, KAPP_DISCARD); } else if (m_switchOption == SWITCH_SAVE) { kateCfg->writeEntry(KAPP_SESSION_EXIT, KAPP_SAVE); } else // m_switchOption == SWITCH_ASK { kateCfg->writeEntry(KAPP_SESSION_EXIT, KAPP_ASK); } } kateCfg->sync(); } //------------------------------------ void KateSessionManager::saveConfig(bool saveSessions) { // Session startup and switch options updateSessionOptions(SO_ALL); saveSessionOptions(SO_ALL); // Sessions configuration if (!saveSessions) { // delete all session files if they exist for (int i = 0; i < (int)m_sessions.count(); ++i) { const TQString &filename = m_sessions[i]->getSessionFilename(); if (filename != TQString::null && TQFile::exists(filename)) { TQFile::remove(filename); } } m_sessions.clear(); m_activeSessionId = INVALID_SESSION; } if (!m_config) { m_config = new KSimpleConfig(m_configFile); } if (m_config->hasGroup(KSM_SESSIONS_LIST)) { m_config->deleteGroup(KSM_SESSIONS_LIST); } m_config->setGroup(KSM_SESSIONS_LIST); m_config->writeEntry(KSM_SESSIONS_COUNT, m_sessions.count()); m_config->writeEntry(KSM_LAST_SESSION_ID, m_activeSessionId); for (int i = 0; i < (int)m_sessions.count(); ++i) { saveSession(i, false, false); m_config->writeEntry(TQString("URL_%1").arg(i), m_sessions[i]->getSessionFilename()); } m_config->sync(); } //------------------------------------ const int KateSessionManager::getStartupOption() { updateSessionOptions(SO_STARTUP); return m_startupOption; } //------------------------------------ const int KateSessionManager::getSwitchOption() { updateSessionOptions(SO_SWITCH); return m_switchOption; } //------------------------------------ void KateSessionManager::setSwitchOption(int option) { m_switchOption = (option == SWITCH_DISCARD || option == SWITCH_SAVE) ? option : SWITCH_ASK; saveSessionOptions(SO_SWITCH); emit switchOptionChanged(); } //------------------------------------ const TQString& KateSessionManager::getSessionName(int sessionId) { if (sessionId < 0 || sessionId >= (int)m_sessions.count()) { return TQString::null; } return m_sessions[sessionId]->getSessionName(); } //------------------------------------ KateSession* KateSessionManager::getSessionFromId(int sessionId) { if (sessionId < 0 || sessionId >= (int)m_sessions.count()) { return NULL; } return m_sessions[sessionId]; } //------------------------------------ int KateSessionManager::getSessionIdFromName(const TQString &name) { if (name.isEmpty()) return INVALID_SESSION; for (int i = 0; i < (int)m_sessions.count(); ++i) { if (m_sessions[i]->getSessionName() == name) return i; } return INVALID_SESSION; } //------------------------------------ bool KateSessionManager::activateSession(int sessionId, bool saveCurr) { if (sessionId < 0 || sessionId >= (int)m_sessions.count()) { return false; } if (sessionId == m_activeSessionId) { return true; } int oldSessionId = m_activeSessionId; if (m_activeSessionId != INVALID_SESSION) { // Do this only if a session has already been activated earlier, if (KateApp::self()->activeMainWindow()) { // First check if all documents can be closed safely if (!KateApp::self()->activeMainWindow()->queryClose_internal()) return false; } if (saveCurr) { saveSession(m_activeSessionId, true); } else { // Delete current session before activating the new one const TQString &filename = m_sessions[m_activeSessionId]->getSessionFilename(); if (filename != TQString::null && TQFile::exists(filename)) { TQFile::remove(filename); } m_sessions.remove(m_activeSessionId); // this also deletes the KateSession item since auto-deletion is enabled m_activeSessionId = INVALID_SESSION; if (sessionId > oldSessionId) { --sessionId; } emit sessionDeleted(oldSessionId); oldSessionId = INVALID_SESSION; } } m_activeSessionId = sessionId; m_sessions[sessionId]->activate(); m_lastSessionId = INVALID_SESSION; emit sessionActivated(m_activeSessionId, oldSessionId); return true; } //------------------------------------ int KateSessionManager::newSession(const TQString &sessionName, bool saveCurr) { m_sessions.append(new KateSession(*this, sessionName, TQString::null)); int newSessionId = m_sessions.count() - 1; emit sessionCreated(newSessionId); activateSession(newSessionId, saveCurr); return newSessionId; } //------------------------------------ int KateSessionManager::cloneSession(int sessionId, const TQString &sessionName, bool activate, bool deleteCurr) { if (sessionId < 0 || sessionId >= (int)m_sessions.count()) { return INVALID_SESSION; } m_sessions.append(new KateSession(*m_sessions[sessionId], sessionName)); int newSessionId = m_sessions.count() - 1; emit sessionCreated(newSessionId); // If cloning the active session, the new session will contain the current status // and the original session will be restored to the last saved state (save as... functionality) saveSession(newSessionId, sessionId == m_activeSessionId); if (sessionId == m_activeSessionId) { reloadActiveSession(); } if (activate) { activateSession(newSessionId, m_activeSessionId != INVALID_SESSION && !deleteCurr); } return newSessionId; } //------------------------------------ bool KateSessionManager::restoreLastSession() { if (m_activeSessionId != INVALID_SESSION) { return false; } return activateSession(m_lastSessionId, false); } //------------------------------------------- void KateSessionManager::saveSession(int sessionId, bool saveGUIInfo, bool setReadOnly) { if (sessionId < 0 || sessionId >= (int)m_sessions.count()) { return; } m_sessions[sessionId]->save(saveGUIInfo, setReadOnly); emit sessionSaved(sessionId); } //------------------------------------------- bool KateSessionManager::deleteSession(int sessionId, int actSessId) { if (sessionId < 0 || sessionId >= (int)m_sessions.count()) { return false; } // delete session file if it exists const TQString &filename = m_sessions[sessionId]->getSessionFilename(); if (filename != TQString::null && TQFile::exists(filename)) { TQFile::remove(filename); } // delete session m_sessions.remove(sessionId); // this also deletes the KateSession item since auto-deletion is enabled if (m_activeSessionId > sessionId) { --m_activeSessionId; } else if (m_activeSessionId == sessionId) { m_activeSessionId = INVALID_SESSION; } emit sessionDeleted(sessionId); if (m_activeSessionId == INVALID_SESSION) { if (m_sessions.count() > 0 && actSessId >= 0 && actSessId < (int)m_sessions.count()) { activateSession(actSessId, false); } else { newSession(); } } return true; } //------------------------------------------- void KateSessionManager::swapSessionsPosition(int sessionId1, int sessionId2) { if (sessionId1 < 0 || sessionId1 >= (int)m_sessions.count() || sessionId2 < 0 || sessionId2 >= (int)m_sessions.count() || sessionId1 == sessionId2) { return; } int idxMin, idxMax; if (sessionId1 < sessionId2) { idxMin = sessionId1; idxMax = sessionId2; } else { idxMin = sessionId2; idxMax = sessionId1; } KateSession *sessMax = m_sessions.take(idxMax); KateSession *sessMin = m_sessions.take(idxMin); m_sessions.insert(idxMin, sessMax); m_sessions.insert(idxMax, sessMin); if (m_activeSessionId == sessionId1) { m_activeSessionId = sessionId2; } else if (m_activeSessionId == sessionId2) { m_activeSessionId = sessionId1; } emit sessionsSwapped(idxMin, idxMax); } //------------------------------------------- void KateSessionManager::moveSessionForward(int sessionId) { if (sessionId < 0 || sessionId >= ((int)m_sessions.count() - 1)) { return; } swapSessionsPosition(sessionId, sessionId + 1); } //------------------------------------------- void KateSessionManager::moveSessionBackward(int sessionId) { if (sessionId < 1 || sessionId >= (int)m_sessions.count()) { return; } swapSessionsPosition(sessionId, sessionId - 1); } //------------------------------------------- void KateSessionManager::renameSession(int sessionId, const TQString &newSessionName) { if (sessionId < 0 || sessionId >= (int)m_sessions.count()) { return; } m_sessions[sessionId]->setSessionName(newSessionName); emit sessionRenamed(sessionId); } //------------------------------------------- void KateSessionManager::setSessionReadOnlyStatus(int sessionId, bool readOnly) { if (sessionId < 0 || sessionId >= (int)m_sessions.count()) { return; } m_sessions[sessionId]->setReadOnly(readOnly); // Session is saved one last time when making it read only saveSession(sessionId, sessionId == m_activeSessionId, true); } //END KateSessionManager //BEGIN KateSessionChooser //------------------------------------------- KateSessionChooser::KateSessionChooser(TQWidget *parent) : KDialogBase(parent, "", true, i18n("Session Chooser"), KDialogBase::User1 | KDialogBase::User2 | KDialogBase::User3, KDialogBase::User2, true, KStdGuiItem::quit(), KGuiItem(i18n("Open Session"), "document-open"), KGuiItem(i18n("New Session"), "document-new")), m_listview(NULL) { TQHBox *page = new TQHBox(this); page->setMinimumSize(400, 200); setMainWidget(page); TQHBox *hb = new TQHBox(page); hb->setSpacing(KDialog::spacingHint()); TQLabel *label = new TQLabel(hb); label->setPixmap(UserIcon("sessionchooser")); label->setFrameStyle (TQFrame::Panel | TQFrame::Sunken); TQVBox *vb = new TQVBox(hb); vb->setSpacing (KDialog::spacingHint()); m_listview = new TDEListView(vb); m_listview->addColumn(i18n("Session Name")); m_listview->addColumn(i18n("Open Documents")); m_listview->setSelectionMode(TQListView::Single); m_listview->setAllColumnsShowFocus(true); m_listview->setSorting(-1); m_listview->setResizeMode(TQListView::LastColumn); connect (m_listview, TQ_SIGNAL(selectionChanged()), this, TQ_SLOT(slotSelectionChanged())); connect (m_listview, TQ_SIGNAL(executed(TQListViewItem*)), this, TQ_SLOT(slotUser2())); TQPtrList<KateSession> &sessions = KateSessionManager::self()->getSessionsList(); for (int idx = sessions.count()-1; idx >= 0; --idx) { new KateSessionChooserItem(m_listview, sessions[idx]->getSessionName(), TQString("%1").arg(sessions[idx]->getDocCount()), idx); } setResult(RESULT_NO_OP); slotSelectionChanged(); // update button status } //------------------------------------------- int KateSessionChooser::getSelectedSessionId() { KateSessionChooserItem *selectedItem = dynamic_cast<KateSessionChooserItem*>(m_listview->selectedItem()); if (!selectedItem) return KateSessionManager::INVALID_SESSION; return selectedItem->getSessionId(); } //------------------------------------------- void KateSessionChooser::slotUser1() { done(RESULT_QUIT_KATE); } //------------------------------------------- void KateSessionChooser::slotUser2() { done(RESULT_OPEN_EXISTING); } //------------------------------------------- void KateSessionChooser::slotUser3() { done(RESULT_OPEN_NEW); } //------------------------------------------- void KateSessionChooser::slotSelectionChanged() { enableButton(KDialogBase::User2, m_listview->selectedItem()); } //END KateSessionChooser #include "katesession.moc"