diff --git a/shell/sessioncontroller.cpp b/shell/sessioncontroller.cpp index f8bfeea32c..c9fac677f9 100644 --- a/shell/sessioncontroller.cpp +++ b/shell/sessioncontroller.cpp @@ -1,1076 +1,1078 @@ /* This file is part of KDevelop Copyright 2008 Andreas Pakulat Copyright 2010 David Nolden This library is free software; you can redistribute it and/or modify it under the terms of the GNU Library General Public License as published by the Free Software Foundation; either version 2 of the License, or (at your option) any later version. 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 "sessioncontroller.h" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "session.h" #include "core.h" #include "uicontroller.h" #include "sessiondialog.h" #include "shellextension.h" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #if KDE_IS_VERSION(4,5,60) #define HAVE_RECOVERY_INTERFACE #include #endif const int recoveryStorageInterval = 10; ///@todo Make this configurable namespace KDevelop { namespace { int argc = 0; char** argv = 0; }; void SessionController::setArguments(int _argc, char** _argv) { argc = _argc; argv = _argv; } static QStringList standardArguments() { QStringList ret; for(int a = 0; a < argc; ++a) { QString arg = QString::fromLocal8Bit(argv[a]); kWarning() << "ARG:" << "" + arg + ""; /* if(arg.startsWith("--graphicssystem=") || arg.startsWith("--style=")) { ret << arg; }else */ if(arg.startsWith("-graphicssystem") || arg.startsWith("-style")) { ret << '-' + arg; if(a+1 < argc) ret << QString::fromLocal8Bit(argv[a+1]); } } kWarning() << "ARGUMENTS: " << ret << "from" << argc; return ret; } class SessionControllerPrivate : public QObject { Q_OBJECT public: SessionControllerPrivate( SessionController* s ) : q(s) , activeSession(0) , grp(0) , recoveryDirectoryIsOwn(false) { recoveryTimer.setInterval(recoveryStorageInterval * 1000); connect(&recoveryTimer, SIGNAL(timeout()), SLOT(recoveryStorageTimeout())); // Try the recovery only after the initialization has finished connect(ICore::self(), SIGNAL(initializationCompleted()), SLOT(lateInitialization()), Qt::QueuedConnection); recoveryTimer.setSingleShot(false); recoveryTimer.start(); } ~SessionControllerPrivate() { if (activeSession) { // when session was active, we deleted the folder already // in that case activeSession = 0 clearRecoveryDirectory(); } } Session* findSessionForName( const QString& name ) const { foreach( Session* s, sessionActions.keys() ) { if( s->name() == name ) return s; } return 0; } Session* findSessionForId(QString idString) { QUuid id(idString); foreach( Session* s, sessionActions.keys() ) { if( s->id() == id) return s; } return 0; } void newSession() { qsrand(QDateTime::currentDateTime().toTime_t()); Session* session = new Session( QUuid::createUuid() ); KProcess::startDetached(ShellExtension::getInstance()->binaryPath(), QStringList() << "-s" << session->id().toString() << standardArguments()); delete session; #if 0 //Terminate this instance of kdevelop if the user agrees foreach(Sublime::MainWindow* window, Core::self()->uiController()->controller()->mainWindows()) window->close(); #endif } void configureSessions() { SessionDialog dlg(ICore::self()->uiController()-> activeMainWindow()); dlg.exec(); } void deleteSession() { int choice = KMessageBox::warningContinueCancel(Core::self()->uiController()->activeMainWindow(), i18n("The current session and all contained settings will be deleted. The projects will stay unaffected. Do you really want to continue?")); if(choice == KMessageBox::Continue) { static_cast(q->activeSession())->deleteFromDisk(); q->emitQuitSession(); } } void renameSession() { KDialog dialog; dialog.setWindowTitle(i18n("Rename Session")); QGroupBox box; QHBoxLayout layout(&box); box.setTitle(i18n("New Session Name")); QLineEdit edit; layout.addWidget(&edit); dialog.setButtons(KDialog::Ok | KDialog::Cancel); edit.setText(q->activeSession()->name()); dialog.setMainWidget(&box); edit.setFocus(); if(dialog.exec() == QDialog::Accepted) { static_cast(q->activeSession())->setName(edit.text()); } } bool loadSessionExternally( Session* s ) { Q_ASSERT( s ); if( !SessionController::tryLockSession(s->id().toString()) ) { KMessageBox::error(ICore::self()->uiController()->activeMainWindow(), i18n("The selected session is already active in another running instance")); return false; }else{ KProcess::startDetached(ShellExtension::getInstance()->binaryPath(), QStringList() << "-s" << s->id().toString() << standardArguments()); return true; } } void activateSession( Session* s ) { Q_ASSERT( s ); KConfigGroup grp = KGlobal::config()->group( SessionController::cfgSessionGroup() ); grp.writeEntry( SessionController::cfgActiveSessionEntry(), s->id().toString() ); grp.sync(); activeSession = s; if (Core::self()->setupFlags() & Core::NoUi) return; QHash::iterator it = sessionActions.find(s); Q_ASSERT( it != sessionActions.end() ); (*it)->setCheckable(true); (*it)->setChecked(true); for(it = sessionActions.begin(); it != sessionActions.end(); ++it) { if(it.key() != s) (*it)->setCheckable(false); } } void loadSessionFromAction( QAction* a ) { foreach( Session* s, sessionActions.keys() ) { if( s->id() == QUuid( a->data().toString() ) && s != activeSession ) { loadSessionExternally( s ); break; } } } void addSession( Session* s ) { if (Core::self()->setupFlags() & Core::NoUi) { sessionActions[s] = 0; return; } KAction* a = new KAction( grp ); a->setText( s->description() ); a->setCheckable( false ); a->setData( s->id().toString() ); sessionActions[s] = a; q->actionCollection()->addAction( "session_"+s->id().toString(), a ); q->unplugActionList( "available_sessions" ); q->plugActionList( "available_sessions", grp->actions() ); connect(s, SIGNAL(nameChanged(QString,QString)), SLOT(nameChanged())); } SessionController* q; QHash sessionActions; ISession* activeSession; QActionGroup* grp; KLockFile::Ptr sessionLock; // Whether this process owns the recovery directory bool recoveryDirectoryIsOwn; QTimer recoveryTimer; QMap currentRecoveryFiles; QString ownSessionDirectory() const { Q_ASSERT(activeSession); return SessionController::sessionDirectory() + '/' + activeSession->id().toString(); } void clearRecoveryDirectory() { removeDirectory(ownSessionDirectory() + "/recovery"); } public slots: void documentSavedOrClosed( KDevelop::IDocument* document ) { if(currentRecoveryFiles.contains(document->url())) { kDebug() << "deleting recovery-info for" << document->url(); foreach(const QString& recoveryFileName, currentRecoveryFiles[document->url()]) { bool result = QFile::remove(recoveryFileName); kDebug() << "deleted" << recoveryFileName << result; } currentRecoveryFiles.remove(document->url()); } } private slots: void lateInitialization() { performRecovery(); connect(Core::self()->documentController(), SIGNAL(documentSaved(KDevelop::IDocument*)), SLOT(documentSavedOrClosed(KDevelop::IDocument*))); connect(Core::self()->documentController(), SIGNAL(documentClosed(KDevelop::IDocument*)), SLOT(documentSavedOrClosed(KDevelop::IDocument*))); } void performRecovery() { kDebug() << "Checking recovery"; QDir recoveryDir(ownSessionDirectory() + "/recovery"); if(recoveryDir.exists()) { kDebug() << "Have recovery directory, starting recovery"; QFile dateFile(recoveryDir.path() + "/date"); dateFile.open(QIODevice::ReadOnly); QString date = QString::fromUtf8(dateFile.readAll()); QDir recoverySubDir(recoveryDir.path() + "/current"); if(!recoverySubDir.exists()) recoverySubDir = QDir(recoveryDir.path() + "/backup"); if(recoverySubDir.exists()) { kWarning() << "Starting recovery from " << recoverySubDir.absolutePath(); QStringList urlList; for(uint num = 0; ; ++num) { QFile urlFile(recoverySubDir.path() + QString("/%1_url").arg(num)); if(!urlFile.exists()) break; urlFile.open(QIODevice::ReadOnly); KUrl originalFile(QString::fromUtf8(urlFile.readAll())); urlList << originalFile.pathOrUrl(); } if(!urlList.isEmpty()) { //Either recover, or delete the recovery directory ///TODO: user proper runtime locale for date, it might be different /// from what was used when the recovery file was saved KGuiItem recover = KStandardGuiItem::cont(); recover.setIcon(KIcon("edit-redo")); recover.setText(i18n("Recover")); KGuiItem discard = KStandardGuiItem::discard(); int choice = KMessageBox::warningContinueCancelList(qApp->activeWindow(), i18nc("%1: date of the last snapshot", "The session crashed the last time it was used. " "The following modified files can be recovered from a backup from %1.", date), urlList, i18n("Crash Recovery"), recover, discard ); if(choice == KMessageBox::Continue) { #if 0 { //Put the recovered documents into the "Review" area, and clear the working set ICore::self()->uiController()->switchToArea("review", KDevelop::IUiController::ThisWindow); Sublime::MainWindow* window = static_cast(ICore::self()->uiController()->activeMainWindow()); window->area()->setWorkingSet("recover"); window->area()->clearViews(); } #endif //Recover the files for(uint num = 0; ; ++num) { QFile urlFile(recoverySubDir.path() + QString("/%1_url").arg(num)); if(!urlFile.exists()) break; urlFile.open(QIODevice::ReadOnly); KUrl originalFile(QString::fromUtf8(urlFile.readAll())); QFile f(recoverySubDir.path() + '/' + QString("/%1_text").arg(num)); f.open(QIODevice::ReadOnly); QString text = QString::fromUtf8(f.readAll()); if(text.isEmpty()) { KMessageBox::error(ICore::self()->uiController()->activeMainWindow(), i18n("Could not recover %1, the recovery file is empty", originalFile.pathOrUrl()), i18n("Recovery")); continue; } kDebug() << "opening" << originalFile << "for recovery"; KDevelop::IDocument* doc = ICore::self()->documentController()->openDocument(originalFile); if(!doc || !doc->textDocument()) { kWarning() << "The document " << originalFile.prettyUrl() << " could not be opened as a text-document, creating a new document with the recovered contents"; doc = ICore::self()->documentController()->openDocumentFromText(text); }else{ #ifdef HAVE_RECOVERY_INTERFACE KTextEditor::RecoveryInterface* recovery = qobject_cast(doc->textDocument()); if(recovery && recovery->isDataRecoveryAvailable()) // Use the recovery from the kate swap-file if possible recovery->recoverData(); else // Use a simple recovery through "replace text" doc->textDocument()->setText(text); #else // Use a simple recovery through "replace text" doc->textDocument()->setText(text); #endif } } } } } } recoveryDirectoryIsOwn = true; } void nameChanged() { Q_ASSERT(qobject_cast(sender())); Session* s = static_cast(sender()); sessionActions[s]->setText( s->description() ); } void recoveryStorageTimeout() { if(!recoveryDirectoryIsOwn) return; currentRecoveryFiles.clear(); QDir recoveryDir(ownSessionDirectory() + "/recovery"); if(!recoveryDir.exists()) { // Try "taking" the recovery directory QDir sessionDir(ownSessionDirectory()); if(!sessionDir.mkdir("recovery")) return; } if (recoveryDir.exists("backup")) { // Clear the old backup recovery directory, as we will create a new one if (!removeDirectory(recoveryDir.absoluteFilePath("backup"))) { kWarning() << "RECOVERY ERROR: Removing the old recovery backup directory failed in " << recoveryDir; return; } } //Make the current recovery dir the backup dir, so we always have a recovery available //This may fail, because "current" might be nonexistent recoveryDir.rename("current", "backup"); { recoveryDir.mkdir("current_incomplete"); QDir recoveryCurrentDir(recoveryDir.path() + "/current_incomplete"); uint num = 0; foreach(KDevelop::IDocument* document, ICore::self()->documentController()->openDocuments()) { if(document->state() == IDocument::Modified || document->state() == IDocument::DirtyAndModified) { //This document was modified, create a recovery-backup if(document->textDocument()) { //Currently we can only back-up text documents QString text = document->textDocument()->text(); if(!text.isEmpty()) { QString urlFilePath = recoveryCurrentDir.path() + QString("/%1_url").arg(num); QFile urlFile(urlFilePath); urlFile.open(QIODevice::WriteOnly); urlFile.write(document->url().pathOrUrl().toUtf8()); urlFile.close(); QString textFilePath = recoveryCurrentDir.path() + '/' + QString("/%1_text").arg(num); QFile f(textFilePath); f.open(QIODevice::WriteOnly); f.write(text.toUtf8()); f.close(); currentRecoveryFiles[document->url()] = QStringList() << (recoveryDir.path() + "/current" + QString("/%1_url").arg(num)) << (recoveryDir.path() + "/current" + QString("/%1_text").arg(num)); if(urlFile.error() != QFile::NoError || f.error() != QFile::NoError) { kWarning() << "RECOVERY ERROR: Failed to write recovery for" << document->url() << "to" << textFilePath; KMessageBox::error(ICore::self()->uiController()->activeMainWindow(), i18n("Failed to write recovery copies to %1. Please make sure that your home directory is writable and not full. This application requires available space in the home directory to run stable. You may experience application crashes until you free up some space.", recoveryCurrentDir.path()), i18n("Recovery Error")); return; } ++num; } } } } } recoveryDir.rename("current_incomplete", "current"); { //Write down the date of the recovery QFile dateFile(recoveryDir.path() + "/date"); dateFile.open(QIODevice::WriteOnly); dateFile.write(QDateTime::currentDateTime().toString(Qt::DefaultLocaleShortDate).toUtf8()); } } }; bool SessionController::lockSession() { d->sessionLock = new KLockFile(d->ownSessionDirectory() + "/lock"); KLockFile::LockResult result = d->sessionLock->lock(KLockFile::NoBlockFlag | KLockFile::ForceFlag); return result == KLockFile::LockOK; } void SessionController::updateSessionDescriptions() { for(QHash< Session*, QAction* >::iterator it = d->sessionActions.begin(); it != d->sessionActions.end(); ++it) { it.key()->updateDescription(); if (*it) (*it)->setText(it.key()->description()); } } SessionController::SessionController( QObject *parent ) : QObject( parent ), d(new SessionControllerPrivate(this)) { setObjectName("SessionController"); setComponentData(KComponentData("kdevsession")); setXMLFile("kdevsessionui.rc"); QDBusConnection::sessionBus().registerObject( "/kdevelop/SessionController", this, QDBusConnection::ExportScriptableSlots ); if (Core::self()->setupFlags() & Core::NoUi) return; KAction* action = actionCollection()->addAction( "new_session", this, SLOT(newSession()) ); action->setText( i18nc("@action:inmenu", "Start New Session") ); action->setToolTip( i18nc("@info:tooltip", "Start a new KDevelop instance with an empty session") ); action->setIcon(KIcon("window-new")); action = actionCollection()->addAction( "rename_session", this, SLOT(renameSession()) ); action->setText( i18n("Rename Session...") ); action->setIcon(KIcon("edit-rename")); action = actionCollection()->addAction( "delete_session", this, SLOT(deleteSession()) ); action->setText( i18n("Delete Session...") ); action->setIcon(KIcon("edit-delete")); action = actionCollection()->addAction( "quit", this, SIGNAL(quitSession()) ); action->setText( i18n("Quit") ); action->setShortcut(Qt::CTRL | Qt::Key_Q); action->setIcon(KIcon("application-exit")); #if 0 action = actionCollection()->addAction( "configure_sessions", this, SLOT(configureSessions()) ); action->setText( i18n("Configure Sessions...") ); action->setToolTip( i18n("Create/Delete/Activate Sessions") ); action->setWhatsThis( i18n( "Configure Sessions

Shows a dialog to Create/Delete Sessions and set a new active session.

" ) ); #endif d->grp = new QActionGroup( this ); connect( d->grp, SIGNAL(triggered(QAction*)), this, SLOT(loadSessionFromAction(QAction*)) ); } SessionController::~SessionController() { delete d; } void SessionController::startNewSession() { d->newSession(); } void SessionController::cleanup() { d->recoveryTimer.stop(); ISession* active = d->activeSession; d->activeSession = 0; if (active->isTemporary()) { deleteSession(active->name()); } qDeleteAll(d->sessionActions); d->sessionActions.clear(); } void SessionController::initialize( const QString& session ) { QDir sessiondir( SessionController::sessionDirectory() ); foreach( const QString& s, sessiondir.entryList( QDir::AllDirs ) ) { QUuid id( s ); if( id.isNull() ) continue; // Only create sessions for directories that represent proper uuid's Session* ses = new Session( id, this ); //Delete sessions that have no name and are empty if( ses->containedProjects().isEmpty() && ses->name().isEmpty() && (session.isEmpty() || (ses->id().toString() != session && ses->name() != session)) ) { ///@todo Think about when we can do this. Another instance might still be using this session. // session->deleteFromDisk(); delete ses; }else{ d->addSession( ses ); } } loadDefaultSession( session ); connect(Core::self()->projectController(), SIGNAL(projectClosed(KDevelop::IProject*)), SLOT(updateSessionDescriptions())); connect(Core::self()->projectController(), SIGNAL(projectOpened(KDevelop::IProject*)), SLOT(updateSessionDescriptions())); updateSessionDescriptions(); } ISession* SessionController::activeSession() const { return d->activeSession; } void SessionController::loadSession( const QString& nameOrId ) { d->loadSessionExternally( session( nameOrId ) ); } QList SessionController::sessionNames() const { QStringList l; foreach( const Session* s, d->sessionActions.keys() ) { l << s->name(); } return l; } QList< const KDevelop::Session* > SessionController::sessions() const { QList< const KDevelop::Session* > ret; foreach( const Session* s, d->sessionActions.keys() ) { ret << s; } return ret; } Session* SessionController::createSession( const QString& name ) { Session* s; if(name.startsWith('{')) { s = new Session( QUuid(name) ); }else{ qsrand(QDateTime::currentDateTime().toTime_t()); s = new Session( QUuid::createUuid() ); s->setName( name ); } d->addSession( s ); return s; } void SessionController::deleteSession( const QString& nameOrId ) { Session* s = session(nameOrId); Q_ASSERT( s != d->activeSession ) ; QHash::iterator it = d->sessionActions.find(s); Q_ASSERT( it != d->sessionActions.end() ); unplugActionList( "available_sessions" ); actionCollection()->removeAction(*it); if (d->grp) { // happens in unit tests d->grp->removeAction(*it); plugActionList( "available_sessions", d->grp->actions() ); } s->deleteFromDisk(); emit sessionDeleted( s->name() ); d->sessionActions.remove(s); s->deleteLater(); } void SessionController::loadDefaultSession( const QString& session ) { QString load = session; if (load.isEmpty()) { KConfigGroup grp = KGlobal::config()->group( cfgSessionGroup() ); load = grp.readEntry( cfgActiveSessionEntry(), "default" ); } Session* s = this->session( load ); if( !s ) { s = createSession( load ); } d->activateSession( s ); } Session* SessionController::session( const QString& nameOrId ) const { Session* ret = d->findSessionForName( nameOrId ); if(ret) return ret; return d->findSessionForId( nameOrId ); } QString SessionController::cloneSession( const QString& nameOrid ) { Session* origSession = session( nameOrid ); qsrand(QDateTime::currentDateTime().toTime_t()); QUuid id = QUuid::createUuid(); KIO::NetAccess::dircopy( KUrl( sessionDirectory() + '/' + origSession->id().toString() ), KUrl( sessionDirectory() + '/' + id.toString() ), Core::self()->uiController()->activeMainWindow() ); Session* newSession = new Session( id ); newSession->setName( i18n( "Copy of %1", origSession->name() ) ); d->addSession(newSession); return newSession->name(); } void SessionController::plugActions() { unplugActionList( "available_sessions" ); plugActionList( "available_sessions", d->grp->actions() ); } QString SessionController::cfgSessionGroup() { return "Sessions"; } QString SessionController::cfgActiveSessionEntry() { return "Active Session ID"; } QList< SessionInfo > SessionController::availableSessionInfo() { QList< SessionInfo > available; QDir sessiondir( SessionController::sessionDirectory() ); foreach( const QString& s, sessiondir.entryList( QDir::AllDirs ) ) { QUuid id( s ); if( id.isNull() ) continue; // TODO: Refactor the code here and in session.cpp so its shared SessionInfo si; si.uuid = id; KSharedConfig::Ptr config = KSharedConfig::openConfig( sessiondir.absolutePath() + '/' + s +"/sessionrc" ); QString desc = config->group( "" ).readEntry( "SessionName", "" ); si.name = desc; si.projects = config->group( "General Options" ).readEntry( "Open Projects", QStringList() ); QString prettyContents = config->group("").readEntry( "SessionPrettyContents", "" ); if(!prettyContents.isEmpty()) { if(!desc.isEmpty()) desc += ": "; desc += prettyContents; } si.description = desc; available << si; } return available; } QString SessionController::sessionDirectory() { return KGlobal::mainComponent().dirs()->saveLocation( "data", KGlobal::mainComponent().componentName()+"/sessions", true ); } QString SessionController::sessionDir() { return sessionDirectory() + '/' + activeSession()->id().toString(); } SessionController::LockSessionState SessionController::tryLockSession(QString id) { LockSessionState ret; ret.lockFile = sessionDirectory() + '/' + id + "/lock"; if(!QFileInfo(ret.lockFile).exists()) { // Maybe the session doesn't exist yet ret.success = true; return ret; } KLockFile::Ptr lock(new KLockFile(ret.lockFile)); ret.success = lock->lock(KLockFile::NoBlockFlag | KLockFile::ForceFlag) == KLockFile::LockOK; if(!ret.success) { lock->getLockInfo(ret.holderPid, ret.holderHostname, ret.holderApp); } return ret; } // Internal helper class class SessionChooserDialog : public KDialog { Q_OBJECT public: SessionChooserDialog(QListView* view, QStandardItemModel* model); bool eventFilter(QObject* object, QEvent* event); public Q_SLOTS: void updateState(); void doubleClicked(QModelIndex); private Q_SLOTS: void deleteButtonPressed(); void showDeleteButton(); void itemEntered(const QModelIndex& index); private: QListView* m_view; QStandardItemModel* m_model; QTimer m_updateStateTimer; QPushButton* m_deleteButton; QTimer m_deleteButtonTimer; int m_deleteCandidateRow; }; SessionChooserDialog::SessionChooserDialog(QListView* view, QStandardItemModel* model) : m_view(view), m_model(model), m_deleteCandidateRow(-1) { m_updateStateTimer.setInterval(5000); m_updateStateTimer.setSingleShot(false); m_updateStateTimer.start(); connect(&m_updateStateTimer, SIGNAL(timeout()), SLOT(updateState())); connect(view, SIGNAL(doubleClicked(QModelIndex)), SLOT(doubleClicked(QModelIndex))); connect(view, SIGNAL(entered(QModelIndex)), SLOT(itemEntered(QModelIndex))); m_deleteButton = new KPushButton(view->viewport()); m_deleteButton->setIcon(KIcon("edit-delete")); m_deleteButton->setToolTip(i18nc("@info", "Delete session")); m_deleteButton->hide(); connect(m_deleteButton, SIGNAL(clicked(bool)), SLOT(deleteButtonPressed())); m_deleteButtonTimer.setInterval(500); m_deleteButtonTimer.setSingleShot(true); connect(&m_deleteButtonTimer, SIGNAL(timeout()), SLOT(showDeleteButton())); view->setMouseTracking(true); view->installEventFilter(this); } void SessionChooserDialog::doubleClicked(QModelIndex index) { QStandardItem* item = m_model->itemFromIndex(index); if(item && item->isEnabled()) accept(); } void SessionChooserDialog::updateState() { // Sometimes locking may take some time, so we stop the timer, to prevent an 'avalanche' of events m_updateStateTimer.stop(); for(int row = 0; row < m_model->rowCount(); ++row) { QString session = m_model->index(row, 0).data().toString(); if(session == i18n("Create New Session")) continue; QString state, tooltip; SessionController::LockSessionState lockState = KDevelop::SessionController::tryLockSession(session); if(!lockState) { tooltip = i18n("Active session.\npid %1, app %2, host %3", lockState.holderPid, lockState.holderApp, lockState.holderHostname); state = i18n("Running"); } if(m_model->item(row, 2)) { m_model->item(row, 1)->setIcon(lockState ? KIcon("") : KIcon("media-playback-start")); m_model->item(row, 1)->setToolTip(tooltip); m_model->item(row, 2)->setText(state); } } m_updateStateTimer.start(); } QString SessionController::showSessionChooserDialog(QString headerText, bool onlyRunning) { // The catalog hasn't been loaded yet KGlobal::locale()->insertCatalog("kdevplatform"); QListView* view = new QListView; QStandardItemModel* model = new QStandardItemModel(view); SessionChooserDialog dialog(view, model); view->setEditTriggers(QAbstractItemView::NoEditTriggers); QVBoxLayout layout(dialog.mainWidget()); if(!headerText.isEmpty()) layout.addWidget(new QLabel(headerText)); model->setColumnCount(3); model->setHeaderData(0, Qt::Horizontal,i18n("Identity")); model->setHeaderData(1, Qt::Horizontal, i18n("Contents")); model->setHeaderData(2, Qt::Horizontal,i18n("State")); view->setModel(model); view->setModelColumn(1); layout.addWidget(view); int row = 0; int defaultRow = 0; QString defaultSession = KGlobal::config()->group( cfgSessionGroup() ).readEntry( cfgActiveSessionEntry(), "default" ); foreach(const KDevelop::SessionInfo& si, KDevelop::SessionController::availableSessionInfo()) { if ( si.name.isEmpty() && si.projects.isEmpty() ) { continue; } bool running = KDevelop::SessionController::tryLockSession(si.uuid.toString()); if(onlyRunning && !running) continue; if(si.uuid.toString() == defaultSession) defaultRow = row; model->setItem(row, 0, new QStandardItem(si.uuid.toString())); model->setItem(row, 1, new QStandardItem(si.description)); model->setItem(row, 2, new QStandardItem); if(defaultRow == row && running) ++defaultRow; ++row; } - + + int cnsRow = row; if(!onlyRunning) { model->setItem(row, 0, new QStandardItem); model->setItem(row, 1, new QStandardItem(KIcon("window-new"), i18n("Create New Session"))); } dialog.updateState(); dialog.mainWidget()->layout()->setContentsMargins(0,0,0,0); view->selectionModel()->setCurrentIndex(model->index(defaultRow, 0), QItemSelectionModel::ClearAndSelect | QItemSelectionModel::Rows); view->setSizePolicy(QSizePolicy::MinimumExpanding, QSizePolicy::MinimumExpanding); ///@todo We need a way to get a proper size-hint from the view, but unfortunately, that only seems possible after the view was shown. dialog.setInitialSize(QSize(900, 600)); if(dialog.exec() != QDialog::Accepted) { return QString(); } QModelIndex selected = view->selectionModel()->currentIndex(); if(selected.isValid()) { - selected = selected.sibling(selected.row(), 0); - QString ret = selected.data().toString(); - if(ret == i18n("Create New Session")) - { + QString ret; + if( selected.row() == cnsRow ) { qsrand(QDateTime::currentDateTime().toTime_t()); ret = QUuid::createUuid().toString(); + } else { + selected = selected.sibling(selected.row(), 0); + ret = selected.data().toString(); } return ret; } return QString(); } void SessionChooserDialog::itemEntered(const QModelIndex& index) { // The last row says "Create new session", we don't want to delete that if(index.row() == m_model->rowCount()-1) { m_deleteButton->hide(); m_deleteButtonTimer.stop(); return; } // align the delete-button to stay on the right border of the item // we need the right most column's index QModelIndex in = m_model->index( index.row(), 1 ); const QRect rect = m_view->visualRect(in); m_deleteButton->resize(rect.height(), rect.height()); QPoint p(rect.right() - m_deleteButton->size().width(), rect.top()+rect.height()/2-m_deleteButton->height()/2); m_deleteButton->move(p); m_deleteCandidateRow = index.row(); m_deleteButtonTimer.start(); } void SessionChooserDialog::showDeleteButton() { m_deleteButton->show(); } bool SessionChooserDialog::eventFilter(QObject* object, QEvent* event) { if(object == m_view && event->type() == QEvent::Leave ) { m_deleteButtonTimer.stop(); m_deleteButton->hide(); } return false; } void SessionChooserDialog::deleteButtonPressed() { if(m_deleteCandidateRow == -1) return; const QString text = i18nc("@info", "The session and all contained settings will be deleted. The projects will stay unaffected. Do you really want to continue?"); const QString caption = i18nc("@title", "Delete Session"); const KGuiItem deleteItem(i18nc("@action:button", "Delete"), KIcon("edit-delete")); const KGuiItem cancelItem(i18nc("@action:button", "Cancel"), KIcon("dialog-cancel")); if(KMessageBox::warningYesNo(this, text, caption, deleteItem, cancelItem) == KMessageBox::Yes) { QModelIndex index = m_model->index(m_deleteCandidateRow, 0); QStandardItem *item = m_model->itemFromIndex(index); const QString uuid = item->text(); //FIXME: What about running sessions? KDevelop::Session session( uuid ); session.deleteFromDisk(); m_model->removeRows( m_deleteCandidateRow, 1 ); m_deleteCandidateRow = -1; } } QString SessionController::sessionName() { if(!activeSession()) return QString(); return activeSession()->description(); } } #include "sessioncontroller.moc" #include "moc_sessioncontroller.cpp"