2009-02-25 09:15:00 +01:00
|
|
|
/**************************************************************************
|
2008-12-02 12:01:29 +01:00
|
|
|
**
|
|
|
|
** This file is part of Qt Creator
|
|
|
|
**
|
2011-01-11 16:28:15 +01:00
|
|
|
** Copyright (c) 2011 Nokia Corporation and/or its subsidiary(-ies).
|
2008-12-02 12:01:29 +01:00
|
|
|
**
|
2009-06-17 00:01:27 +10:00
|
|
|
** Contact: Nokia Corporation (qt-info@nokia.com)
|
2008-12-02 12:01:29 +01:00
|
|
|
**
|
2010-12-17 16:01:08 +01:00
|
|
|
** No Commercial Usage
|
2008-12-02 14:17:16 +01:00
|
|
|
**
|
2010-12-17 16:01:08 +01:00
|
|
|
** This file contains pre-release code and may not be distributed.
|
|
|
|
** You may use this file in accordance with the terms and conditions
|
|
|
|
** contained in the Technology Preview License Agreement accompanying
|
|
|
|
** this package.
|
2008-12-02 14:17:16 +01:00
|
|
|
**
|
2009-02-25 09:15:00 +01:00
|
|
|
** GNU Lesser General Public License Usage
|
2008-12-02 14:17:16 +01:00
|
|
|
**
|
2009-02-25 09:15:00 +01:00
|
|
|
** Alternatively, this file may be used under the terms of the GNU Lesser
|
|
|
|
** General Public License version 2.1 as published by the Free Software
|
|
|
|
** Foundation and appearing in the file LICENSE.LGPL included in the
|
|
|
|
** packaging of this file. Please review the following information to
|
|
|
|
** ensure the GNU Lesser General Public License version 2.1 requirements
|
|
|
|
** will be met: http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.
|
2008-12-02 14:17:16 +01:00
|
|
|
**
|
2010-12-17 16:01:08 +01:00
|
|
|
** In addition, as a special exception, Nokia gives you certain additional
|
|
|
|
** rights. These rights are described in the Nokia Qt LGPL Exception
|
|
|
|
** version 1.1, included in the file LGPL_EXCEPTION.txt in this package.
|
|
|
|
**
|
|
|
|
** If you have questions regarding the use of this file, please contact
|
|
|
|
** Nokia at qt-info@nokia.com.
|
2008-12-02 12:01:29 +01:00
|
|
|
**
|
2009-02-25 09:15:00 +01:00
|
|
|
**************************************************************************/
|
2008-12-02 14:09:21 +01:00
|
|
|
|
2008-12-02 12:01:29 +01:00
|
|
|
#include "modemanager.h"
|
2008-12-09 15:25:01 +01:00
|
|
|
|
2008-12-02 12:01:29 +01:00
|
|
|
#include "fancytabwidget.h"
|
|
|
|
#include "fancyactionbar.h"
|
2009-01-23 13:03:36 +01:00
|
|
|
#include "icore.h"
|
2008-12-02 12:01:29 +01:00
|
|
|
#include "mainwindow.h"
|
|
|
|
|
|
|
|
#include <aggregation/aggregate.h>
|
2008-12-09 15:25:01 +01:00
|
|
|
|
2009-01-13 13:39:31 +01:00
|
|
|
#include <coreplugin/actionmanager/actionmanager.h>
|
2009-01-14 13:17:53 +01:00
|
|
|
#include <coreplugin/actionmanager/command.h>
|
2008-12-02 12:01:29 +01:00
|
|
|
#include <coreplugin/coreconstants.h>
|
|
|
|
#include <coreplugin/imode.h>
|
|
|
|
#include <coreplugin/uniqueidmanager.h>
|
|
|
|
|
2009-01-19 12:39:20 +01:00
|
|
|
#include <extensionsystem/pluginmanager.h>
|
|
|
|
|
2008-12-09 15:25:01 +01:00
|
|
|
#include <utils/qtcassert.h>
|
|
|
|
|
2008-12-02 12:01:29 +01:00
|
|
|
#include <QtCore/QDebug>
|
2010-03-18 10:59:06 +01:00
|
|
|
#include <QtCore/QList>
|
|
|
|
#include <QtCore/QMap>
|
|
|
|
#include <QtCore/QVector>
|
|
|
|
|
2008-12-02 12:01:29 +01:00
|
|
|
#include <QtCore/QSignalMapper>
|
2009-10-01 16:38:08 +02:00
|
|
|
#include <QtGui/QShortcut>
|
2008-12-02 12:01:29 +01:00
|
|
|
#include <QtGui/QAction>
|
|
|
|
|
2010-03-18 10:59:06 +01:00
|
|
|
namespace Core {
|
|
|
|
|
2010-06-25 12:56:16 +02:00
|
|
|
struct ModeManagerPrivate
|
|
|
|
{
|
2010-03-18 10:59:06 +01:00
|
|
|
explicit ModeManagerPrivate(Internal::MainWindow *mainWindow,
|
|
|
|
Internal::FancyTabWidget *modeStack,
|
|
|
|
ModeManager *q);
|
|
|
|
|
|
|
|
static ModeManager *m_instance;
|
|
|
|
Internal::MainWindow *m_mainWindow;
|
|
|
|
Internal::FancyTabWidget *m_modeStack;
|
|
|
|
Internal::FancyActionBar *m_actionBar;
|
|
|
|
QMap<Command*, int> m_actions;
|
|
|
|
QVector<IMode*> m_modes;
|
|
|
|
QVector<Command*> m_modeShortcuts;
|
|
|
|
QSignalMapper *m_signalMapper;
|
2010-06-25 12:56:16 +02:00
|
|
|
Context m_addedContexts;
|
2010-03-18 10:59:06 +01:00
|
|
|
int m_oldCurrent;
|
|
|
|
};
|
|
|
|
|
|
|
|
ModeManager *ModeManagerPrivate::m_instance = 0;
|
|
|
|
|
|
|
|
ModeManagerPrivate::ModeManagerPrivate(Internal::MainWindow *mainWindow,
|
|
|
|
Internal::FancyTabWidget *modeStack,
|
|
|
|
ModeManager *q) :
|
2008-12-02 12:01:29 +01:00
|
|
|
m_mainWindow(mainWindow),
|
|
|
|
m_modeStack(modeStack),
|
2010-03-18 10:59:06 +01:00
|
|
|
m_signalMapper(new QSignalMapper(q)),
|
2010-09-07 15:25:14 +02:00
|
|
|
m_oldCurrent(-1)
|
2010-03-18 10:59:06 +01:00
|
|
|
{
|
|
|
|
}
|
2010-02-25 14:52:04 +01:00
|
|
|
|
2010-03-18 10:59:06 +01:00
|
|
|
ModeManager::ModeManager(Internal::MainWindow *mainWindow,
|
|
|
|
Internal::FancyTabWidget *modeStack) :
|
|
|
|
d(new ModeManagerPrivate(mainWindow, modeStack, this))
|
2008-12-02 12:01:29 +01:00
|
|
|
{
|
2010-03-18 10:59:06 +01:00
|
|
|
ModeManagerPrivate::m_instance = this;
|
2008-12-02 12:01:29 +01:00
|
|
|
|
2010-03-18 10:59:06 +01:00
|
|
|
d->m_actionBar = new Internal::FancyActionBar(modeStack);
|
|
|
|
d->m_modeStack->addCornerWidget(d->m_actionBar);
|
2008-12-02 12:01:29 +01:00
|
|
|
|
2010-03-18 10:59:06 +01:00
|
|
|
connect(d->m_modeStack, SIGNAL(currentAboutToShow(int)), SLOT(currentTabAboutToChange(int)));
|
|
|
|
connect(d->m_modeStack, SIGNAL(currentChanged(int)), SLOT(currentTabChanged(int)));
|
|
|
|
connect(d->m_signalMapper, SIGNAL(mapped(QString)), this, SLOT(activateMode(QString)));
|
2008-12-02 12:01:29 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void ModeManager::init()
|
|
|
|
{
|
|
|
|
QObject::connect(ExtensionSystem::PluginManager::instance(), SIGNAL(objectAdded(QObject*)),
|
|
|
|
this, SLOT(objectAdded(QObject*)));
|
|
|
|
QObject::connect(ExtensionSystem::PluginManager::instance(), SIGNAL(aboutToRemoveObject(QObject*)),
|
|
|
|
this, SLOT(aboutToRemoveObject(QObject*)));
|
|
|
|
}
|
|
|
|
|
2010-03-18 10:59:06 +01:00
|
|
|
ModeManager::~ModeManager()
|
|
|
|
{
|
|
|
|
delete d;
|
|
|
|
ModeManagerPrivate::m_instance = 0;
|
|
|
|
}
|
|
|
|
|
2008-12-02 12:01:29 +01:00
|
|
|
void ModeManager::addWidget(QWidget *widget)
|
|
|
|
{
|
|
|
|
// We want the actionbar to stay on the bottom
|
2010-03-18 10:59:06 +01:00
|
|
|
// so d->m_modeStack->cornerWidgetCount() -1 inserts it at the position immediately above
|
2008-12-02 12:01:29 +01:00
|
|
|
// the actionbar
|
2010-03-18 10:59:06 +01:00
|
|
|
d->m_modeStack->insertCornerWidget(d->m_modeStack->cornerWidgetCount() -1, widget);
|
2008-12-02 12:01:29 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
IMode *ModeManager::currentMode() const
|
|
|
|
{
|
2010-03-18 10:59:06 +01:00
|
|
|
int currentIndex = d->m_modeStack->currentIndex();
|
2009-12-15 16:42:56 +01:00
|
|
|
if (currentIndex < 0)
|
|
|
|
return 0;
|
2010-03-18 10:59:06 +01:00
|
|
|
return d->m_modes.at(currentIndex);
|
2008-12-02 12:01:29 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
int ModeManager::indexOf(const QString &id) const
|
|
|
|
{
|
2010-03-18 10:59:06 +01:00
|
|
|
for (int i = 0; i < d->m_modes.count(); ++i) {
|
|
|
|
if (d->m_modes.at(i)->id() == id)
|
2008-12-02 12:01:29 +01:00
|
|
|
return i;
|
|
|
|
}
|
|
|
|
qDebug() << "Warning, no such mode:" << id;
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
IMode *ModeManager::mode(const QString &id) const
|
|
|
|
{
|
|
|
|
const int index = indexOf(id);
|
|
|
|
if (index >= 0)
|
2010-03-18 10:59:06 +01:00
|
|
|
return d->m_modes.at(index);
|
2008-12-02 12:01:29 +01:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2010-09-14 15:15:57 +02:00
|
|
|
void ModeManager::activateModeType(const QString &type)
|
|
|
|
{
|
2010-12-01 15:08:43 +01:00
|
|
|
if (currentMode() && currentMode()->type() == type)
|
|
|
|
return;
|
2010-09-14 15:15:57 +02:00
|
|
|
int index = -1;
|
|
|
|
for (int i = 0; i < d->m_modes.count(); ++i) {
|
|
|
|
if (d->m_modes.at(i)->type() == type) {
|
|
|
|
index = i;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (index != -1)
|
|
|
|
d->m_modeStack->setCurrentIndex(index);
|
|
|
|
}
|
|
|
|
|
2008-12-02 12:01:29 +01:00
|
|
|
void ModeManager::activateMode(const QString &id)
|
|
|
|
{
|
|
|
|
const int index = indexOf(id);
|
|
|
|
if (index >= 0)
|
2010-03-18 10:59:06 +01:00
|
|
|
d->m_modeStack->setCurrentIndex(index);
|
2008-12-02 12:01:29 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void ModeManager::objectAdded(QObject *obj)
|
|
|
|
{
|
|
|
|
IMode *mode = Aggregation::query<IMode>(obj);
|
|
|
|
if (!mode)
|
|
|
|
return;
|
|
|
|
|
2010-03-18 10:59:06 +01:00
|
|
|
d->m_mainWindow->addContextObject(mode);
|
2008-12-02 12:01:29 +01:00
|
|
|
|
|
|
|
// Count the number of modes with a higher priority
|
|
|
|
int index = 0;
|
2010-03-18 10:59:06 +01:00
|
|
|
foreach (const IMode *m, d->m_modes)
|
2008-12-02 12:01:29 +01:00
|
|
|
if (m->priority() > mode->priority())
|
|
|
|
++index;
|
|
|
|
|
2010-03-18 10:59:06 +01:00
|
|
|
d->m_modes.insert(index, mode);
|
|
|
|
d->m_modeStack->insertTab(index, mode->widget(), mode->icon(), mode->displayName());
|
|
|
|
d->m_modeStack->setTabEnabled(index, mode->isEnabled());
|
2008-12-02 12:01:29 +01:00
|
|
|
|
|
|
|
// Register mode shortcut
|
2010-03-18 10:59:06 +01:00
|
|
|
ActionManager *am = d->m_mainWindow->actionManager();
|
2010-01-07 18:17:24 +01:00
|
|
|
const QString shortcutId = QLatin1String("QtCreator.Mode.") + mode->id();
|
2010-03-18 10:59:06 +01:00
|
|
|
QShortcut *shortcut = new QShortcut(d->m_mainWindow);
|
2010-05-07 16:32:58 +02:00
|
|
|
shortcut->setWhatsThis(tr("Switch to <b>%1</b> mode").arg(mode->displayName()));
|
2010-06-28 14:30:03 +02:00
|
|
|
Command *cmd = am->registerShortcut(shortcut, shortcutId, Context(Constants::C_GLOBAL));
|
2008-12-02 12:01:29 +01:00
|
|
|
|
2010-03-18 10:59:06 +01:00
|
|
|
d->m_modeShortcuts.insert(index, cmd);
|
2008-12-02 12:01:29 +01:00
|
|
|
connect(cmd, SIGNAL(keySequenceChanged()), this, SLOT(updateModeToolTip()));
|
2010-03-18 10:59:06 +01:00
|
|
|
for (int i = 0; i < d->m_modeShortcuts.size(); ++i) {
|
|
|
|
Command *currentCmd = d->m_modeShortcuts.at(i);
|
2010-10-08 13:18:35 +02:00
|
|
|
// we need this hack with currentlyHasDefaultSequence
|
|
|
|
// because we call setDefaultShortcut multiple times on the same cmd
|
|
|
|
// and still expect the current shortcut to change with it
|
2009-07-23 11:02:37 +02:00
|
|
|
bool currentlyHasDefaultSequence = (currentCmd->keySequence()
|
|
|
|
== currentCmd->defaultKeySequence());
|
2009-06-03 20:45:49 +02:00
|
|
|
#ifdef Q_WS_MAC
|
2009-07-23 11:02:37 +02:00
|
|
|
currentCmd->setDefaultKeySequence(QKeySequence(QString("Meta+%1").arg(i+1)));
|
2008-12-02 12:01:29 +01:00
|
|
|
#else
|
2009-07-23 11:02:37 +02:00
|
|
|
currentCmd->setDefaultKeySequence(QKeySequence(QString("Ctrl+%1").arg(i+1)));
|
2008-12-02 12:01:29 +01:00
|
|
|
#endif
|
2009-07-23 11:02:37 +02:00
|
|
|
if (currentlyHasDefaultSequence)
|
|
|
|
currentCmd->setKeySequence(currentCmd->defaultKeySequence());
|
2008-12-02 12:01:29 +01:00
|
|
|
}
|
|
|
|
|
2010-03-18 10:59:06 +01:00
|
|
|
d->m_signalMapper->setMapping(shortcut, mode->id());
|
|
|
|
connect(shortcut, SIGNAL(activated()), d->m_signalMapper, SLOT(map()));
|
2010-02-16 16:11:51 +01:00
|
|
|
connect(mode, SIGNAL(enabledStateChanged(bool)),
|
|
|
|
this, SLOT(enabledStateChanged()));
|
2008-12-02 12:01:29 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void ModeManager::updateModeToolTip()
|
|
|
|
{
|
2009-01-14 13:17:53 +01:00
|
|
|
Command *cmd = qobject_cast<Command *>(sender());
|
2008-12-02 12:01:29 +01:00
|
|
|
if (cmd) {
|
2010-03-18 10:59:06 +01:00
|
|
|
int index = d->m_modeShortcuts.indexOf(cmd);
|
2008-12-02 12:01:29 +01:00
|
|
|
if (index != -1)
|
2010-03-18 10:59:06 +01:00
|
|
|
d->m_modeStack->setTabToolTip(index, cmd->stringWithAppendedShortcut(cmd->shortcut()->whatsThis()));
|
2008-12-02 12:01:29 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-02-16 16:11:51 +01:00
|
|
|
void ModeManager::enabledStateChanged()
|
|
|
|
{
|
|
|
|
IMode *mode = qobject_cast<IMode *>(sender());
|
|
|
|
QTC_ASSERT(mode, return);
|
2010-03-18 10:59:06 +01:00
|
|
|
int index = d->m_modes.indexOf(mode);
|
2010-02-16 16:11:51 +01:00
|
|
|
QTC_ASSERT(index >= 0, return);
|
2010-03-18 10:59:06 +01:00
|
|
|
d->m_modeStack->setTabEnabled(index, mode->isEnabled());
|
2010-04-12 12:43:17 +02:00
|
|
|
|
|
|
|
// Make sure we leave any disabled mode to prevent possible crashes:
|
|
|
|
if (mode == currentMode() && !mode->isEnabled()) {
|
|
|
|
// This assumes that there is always at least one enabled mode.
|
|
|
|
for (int i = 0; i < d->m_modes.count(); ++i) {
|
|
|
|
if (d->m_modes.at(i) != mode &&
|
|
|
|
d->m_modes.at(i)->isEnabled()) {
|
|
|
|
activateMode(d->m_modes.at(i)->id());
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2010-02-16 16:11:51 +01:00
|
|
|
}
|
|
|
|
|
2008-12-02 12:01:29 +01:00
|
|
|
void ModeManager::aboutToRemoveObject(QObject *obj)
|
|
|
|
{
|
|
|
|
IMode *mode = Aggregation::query<IMode>(obj);
|
|
|
|
if (!mode)
|
|
|
|
return;
|
|
|
|
|
2010-03-18 10:59:06 +01:00
|
|
|
const int index = d->m_modes.indexOf(mode);
|
|
|
|
d->m_modes.remove(index);
|
|
|
|
d->m_modeShortcuts.remove(index);
|
|
|
|
d->m_modeStack->removeTab(index);
|
2008-12-02 12:01:29 +01:00
|
|
|
|
2010-03-18 10:59:06 +01:00
|
|
|
d->m_mainWindow->removeContextObject(mode);
|
2008-12-02 12:01:29 +01:00
|
|
|
}
|
|
|
|
|
2010-01-22 18:04:24 +01:00
|
|
|
void ModeManager::addAction(Command *command, int priority)
|
2008-12-02 12:01:29 +01:00
|
|
|
{
|
2010-03-18 10:59:06 +01:00
|
|
|
d->m_actions.insert(command, priority);
|
2008-12-02 12:01:29 +01:00
|
|
|
|
|
|
|
// Count the number of commands with a higher priority
|
|
|
|
int index = 0;
|
2010-03-18 10:59:06 +01:00
|
|
|
foreach (int p, d->m_actions) {
|
2008-12-02 12:01:29 +01:00
|
|
|
if (p > priority)
|
|
|
|
++index;
|
2009-12-09 15:30:23 +01:00
|
|
|
}
|
2008-12-02 12:01:29 +01:00
|
|
|
|
2010-03-18 10:59:06 +01:00
|
|
|
d->m_actionBar->insertAction(index, command->action());
|
2008-12-02 12:01:29 +01:00
|
|
|
}
|
|
|
|
|
2010-01-21 21:12:40 +01:00
|
|
|
void ModeManager::addProjectSelector(QAction *action)
|
|
|
|
{
|
2010-03-18 10:59:06 +01:00
|
|
|
d->m_actionBar->addProjectSelector(action);
|
2010-11-12 20:42:40 +01:00
|
|
|
d->m_actions.insert(0, INT_MAX);
|
2010-01-21 21:12:40 +01:00
|
|
|
}
|
|
|
|
|
2008-12-02 12:01:29 +01:00
|
|
|
void ModeManager::currentTabAboutToChange(int index)
|
|
|
|
{
|
|
|
|
if (index >= 0) {
|
2010-03-18 10:59:06 +01:00
|
|
|
IMode *mode = d->m_modes.at(index);
|
2008-12-02 12:01:29 +01:00
|
|
|
if (mode)
|
|
|
|
emit currentModeAboutToChange(mode);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void ModeManager::currentTabChanged(int index)
|
|
|
|
{
|
|
|
|
// Tab index changes to -1 when there is no tab left.
|
|
|
|
if (index >= 0) {
|
2010-03-18 10:59:06 +01:00
|
|
|
IMode *mode = d->m_modes.at(index);
|
2008-12-02 12:01:29 +01:00
|
|
|
|
|
|
|
// FIXME: This hardcoded context update is required for the Debug and Edit modes, since
|
|
|
|
// they use the editor widget, which is already a context widget so the main window won't
|
|
|
|
// go further up the parent tree to find the mode context.
|
2010-03-22 18:05:22 +01:00
|
|
|
ICore::instance()->updateAdditionalContexts(d->m_addedContexts, mode->context());
|
2010-03-18 10:59:06 +01:00
|
|
|
d->m_addedContexts = mode->context();
|
2010-03-22 18:05:22 +01:00
|
|
|
|
2010-02-25 14:52:04 +01:00
|
|
|
IMode *oldMode = 0;
|
2010-03-18 10:59:06 +01:00
|
|
|
if (d->m_oldCurrent >= 0)
|
|
|
|
oldMode = d->m_modes.at(d->m_oldCurrent);
|
|
|
|
d->m_oldCurrent = index;
|
2010-02-25 14:52:04 +01:00
|
|
|
emit currentModeChanged(mode, oldMode);
|
2008-12-02 12:01:29 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void ModeManager::setFocusToCurrentMode()
|
|
|
|
{
|
|
|
|
IMode *mode = currentMode();
|
2008-12-09 15:25:01 +01:00
|
|
|
QTC_ASSERT(mode, return);
|
2008-12-02 12:01:29 +01:00
|
|
|
QWidget *widget = mode->widget();
|
|
|
|
if (widget) {
|
|
|
|
QWidget *focusWidget = widget->focusWidget();
|
|
|
|
if (focusWidget)
|
|
|
|
focusWidget->setFocus();
|
|
|
|
else
|
|
|
|
widget->setFocus();
|
|
|
|
}
|
|
|
|
}
|
2010-03-18 10:59:06 +01:00
|
|
|
|
|
|
|
ModeManager *ModeManager::instance()
|
|
|
|
{
|
|
|
|
return ModeManagerPrivate::m_instance;
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace Core
|