Files
qt-creator/src/plugins/coreplugin/modemanager.cpp

340 lines
10 KiB
C++
Raw Normal View History

/**************************************************************************
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
**
2011-04-13 08:42:33 +02:00
** Contact: Nokia Corporation (info@qt.nokia.com)
2008-12-02 12:01:29 +01:00
**
**
** GNU Lesser General Public License Usage
**
2011-04-13 08:42:33 +02:00
** 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.
**
2010-12-17 16:01:08 +01:00
** In addition, as a special exception, Nokia gives you certain additional
2011-04-13 08:42:33 +02:00
** rights. These rights are described in the Nokia Qt LGPL Exception
2010-12-17 16:01:08 +01:00
** version 1.1, included in the file LGPL_EXCEPTION.txt in this package.
**
2011-04-13 08:42:33 +02:00
** Other Usage
**
** Alternatively, this file may be used in accordance with the terms and
** conditions contained in a signed written agreement between you and Nokia.
**
2010-12-17 16:01:08 +01:00
** If you have questions regarding the use of this file, please contact
** Nokia at info@qt.nokia.com.
2008-12-02 12:01:29 +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"
#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
#include <coreplugin/actionmanager/actionmanager.h>
#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>
#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>
#include <QtCore/QList>
#include <QtCore/QMap>
#include <QtCore/QVector>
2008-12-02 12:01:29 +01:00
#include <QtCore/QSignalMapper>
#include <QtGui/QShortcut>
2008-12-02 12:01:29 +01:00
#include <QtGui/QAction>
namespace Core {
struct ModeManagerPrivate
{
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<QAction*, int> m_actions;
QVector<IMode*> m_modes;
QVector<Command*> m_modeShortcuts;
QSignalMapper *m_signalMapper;
Context m_addedContexts;
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),
m_signalMapper(new QSignalMapper(q)),
m_oldCurrent(-1)
{
}
ModeManager::ModeManager(Internal::MainWindow *mainWindow,
Internal::FancyTabWidget *modeStack) :
d(new ModeManagerPrivate(mainWindow, modeStack, this))
2008-12-02 12:01:29 +01:00
{
ModeManagerPrivate::m_instance = this;
2008-12-02 12:01:29 +01:00
d->m_actionBar = new Internal::FancyActionBar(modeStack);
d->m_modeStack->addCornerWidget(d->m_actionBar);
2008-12-02 12:01:29 +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*)));
}
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
// so d->m_modeStack->cornerWidgetCount() -1 inserts it at the position immediately above
2008-12-02 12:01:29 +01:00
// the actionbar
d->m_modeStack->insertCornerWidget(d->m_modeStack->cornerWidgetCount() -1, widget);
2008-12-02 12:01:29 +01:00
}
IMode *ModeManager::currentMode() const
{
int currentIndex = d->m_modeStack->currentIndex();
if (currentIndex < 0)
return 0;
return d->m_modes.at(currentIndex);
2008-12-02 12:01:29 +01:00
}
int ModeManager::indexOf(const QString &id) const
{
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)
return d->m_modes.at(index);
2008-12-02 12:01:29 +01:00
return 0;
}
void ModeManager::activateModeType(const QString &type)
{
if (currentMode() && currentMode()->type() == type)
return;
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)
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;
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;
foreach (const IMode *m, d->m_modes)
2008-12-02 12:01:29 +01:00
if (m->priority() > mode->priority())
++index;
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
ActionManager *am = d->m_mainWindow->actionManager();
const QString shortcutId = QLatin1String("QtCreator.Mode.") + mode->id();
QShortcut *shortcut = new QShortcut(d->m_mainWindow);
shortcut->setWhatsThis(tr("Switch to <b>%1</b> mode").arg(mode->displayName()));
Command *cmd = am->registerShortcut(shortcut, shortcutId, Context(Constants::C_GLOBAL));
2008-12-02 12:01:29 +01:00
d->m_modeShortcuts.insert(index, cmd);
2008-12-02 12:01:29 +01:00
connect(cmd, SIGNAL(keySequenceChanged()), this, SLOT(updateModeToolTip()));
for (int i = 0; i < d->m_modeShortcuts.size(); ++i) {
Command *currentCmd = d->m_modeShortcuts.at(i);
// 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
bool currentlyHasDefaultSequence = (currentCmd->keySequence()
== currentCmd->defaultKeySequence());
#ifdef Q_WS_MAC
currentCmd->setDefaultKeySequence(QKeySequence(QString("Meta+%1").arg(i+1)));
2008-12-02 12:01:29 +01:00
#else
currentCmd->setDefaultKeySequence(QKeySequence(QString("Ctrl+%1").arg(i+1)));
2008-12-02 12:01:29 +01:00
#endif
if (currentlyHasDefaultSequence)
currentCmd->setKeySequence(currentCmd->defaultKeySequence());
2008-12-02 12:01:29 +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()
{
Command *cmd = qobject_cast<Command *>(sender());
2008-12-02 12:01:29 +01:00
if (cmd) {
int index = d->m_modeShortcuts.indexOf(cmd);
2008-12-02 12:01:29 +01:00
if (index != -1)
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);
int index = d->m_modes.indexOf(mode);
2010-02-16 16:11:51 +01:00
QTC_ASSERT(index >= 0, return);
d->m_modeStack->setTabEnabled(index, mode->isEnabled());
// 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;
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
d->m_mainWindow->removeContextObject(mode);
2008-12-02 12:01:29 +01:00
}
void ModeManager::addAction(QAction *action, int priority)
2008-12-02 12:01:29 +01:00
{
d->m_actions.insert(action, priority);
2008-12-02 12:01:29 +01:00
// Count the number of commands with a higher priority
int index = 0;
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
d->m_actionBar->insertAction(index, action);
2008-12-02 12:01:29 +01:00
}
void ModeManager::addProjectSelector(QAction *action)
{
d->m_actionBar->addProjectSelector(action);
d->m_actions.insert(0, INT_MAX);
}
2008-12-02 12:01:29 +01:00
void ModeManager::currentTabAboutToChange(int index)
{
if (index >= 0) {
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) {
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.
ICore::instance()->updateAdditionalContexts(d->m_addedContexts, mode->context());
d->m_addedContexts = mode->context();
IMode *oldMode = 0;
if (d->m_oldCurrent >= 0)
oldMode = d->m_modes.at(d->m_oldCurrent);
d->m_oldCurrent = index;
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();
}
}
ModeManager *ModeManager::instance()
{
return ModeManagerPrivate::m_instance;
}
} // namespace Core