2013-04-05 23:46:56 +03:00
|
|
|
/**
|
|
|
|
******************************************************************************
|
|
|
|
*
|
|
|
|
* @file modemanager.cpp
|
|
|
|
* @author The OpenPilot Team, http://www.openpilot.org Copyright (C) 2010.
|
|
|
|
* Parts by Nokia Corporation (qt-info@nokia.com) Copyright (C) 2009.
|
|
|
|
* @addtogroup GCSPlugins GCS Plugins
|
|
|
|
* @{
|
|
|
|
* @addtogroup CorePlugin Core Plugin
|
|
|
|
* @{
|
|
|
|
* @brief The Core GCS plugin
|
|
|
|
*****************************************************************************/
|
|
|
|
/*
|
2013-05-19 17:37:30 +03:00
|
|
|
* 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
|
2013-04-05 23:46:56 +03:00
|
|
|
* (at your option) any later version.
|
2013-05-19 17:37:30 +03:00
|
|
|
*
|
|
|
|
* 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
|
2013-04-05 23:46:56 +03:00
|
|
|
* for more details.
|
2013-05-19 17:37:30 +03:00
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License along
|
|
|
|
* with this program; if not, write to the Free Software Foundation, Inc.,
|
2013-04-05 23:46:56 +03:00
|
|
|
* 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "modemanager.h"
|
|
|
|
|
|
|
|
#include "fancytabwidget.h"
|
|
|
|
#include "fancyactionbar.h"
|
|
|
|
#include "utils/mytabwidget.h"
|
|
|
|
#include "icore.h"
|
|
|
|
#include "mainwindow.h"
|
|
|
|
|
|
|
|
#include <aggregation/aggregate.h>
|
|
|
|
|
|
|
|
#include <coreplugin/actionmanager/actionmanager.h>
|
|
|
|
#include <coreplugin/actionmanager/command.h>
|
|
|
|
#include <coreplugin/coreconstants.h>
|
|
|
|
#include <coreplugin/imode.h>
|
|
|
|
#include <coreplugin/uniqueidmanager.h>
|
|
|
|
|
|
|
|
#include <extensionsystem/pluginmanager.h>
|
|
|
|
|
|
|
|
#include <utils/qtcassert.h>
|
|
|
|
|
|
|
|
#include <QtCore/QObject>
|
|
|
|
#include <QtCore/QDebug>
|
|
|
|
#include <QtCore/QSignalMapper>
|
2013-09-15 23:06:25 +02:00
|
|
|
#include <QShortcut>
|
2013-04-05 23:46:56 +03:00
|
|
|
|
2013-09-15 23:06:25 +02:00
|
|
|
#include <QAction>
|
|
|
|
#include <QTabWidget>
|
|
|
|
#include <QVBoxLayout>
|
2013-04-05 23:46:56 +03:00
|
|
|
|
|
|
|
using namespace Core;
|
|
|
|
using namespace Core::Internal;
|
|
|
|
|
|
|
|
ModeManager *ModeManager::m_instance = 0;
|
|
|
|
|
|
|
|
ModeManager::ModeManager(Internal::MainWindow *mainWindow, MyTabWidget *modeStack) :
|
|
|
|
m_mainWindow(mainWindow),
|
|
|
|
m_modeStack(modeStack),
|
|
|
|
m_signalMapper(new QSignalMapper(this)),
|
|
|
|
m_isReprioritizing(false)
|
|
|
|
{
|
|
|
|
m_instance = this;
|
|
|
|
|
2013-05-19 17:37:30 +03:00
|
|
|
// connect((m_modeStack), SIGNAL(currentAboutToShow(int)), SLOT(currentTabAboutToChange(int)));
|
2013-04-05 23:46:56 +03:00
|
|
|
connect(m_modeStack, SIGNAL(currentChanged(int)), this, SLOT(currentTabChanged(int)));
|
2013-05-19 17:37:30 +03:00
|
|
|
connect(m_modeStack, SIGNAL(tabMoved(int, int)), this, SLOT(tabMoved(int, int)));
|
2013-04-05 23:46:56 +03:00
|
|
|
connect(m_signalMapper, SIGNAL(mapped(QString)), this, SLOT(activateMode(QString)));
|
|
|
|
}
|
|
|
|
|
|
|
|
void ModeManager::init()
|
|
|
|
{
|
2013-05-19 17:37:30 +03:00
|
|
|
QObject::connect(ExtensionSystem::PluginManager::instance(), SIGNAL(objectAdded(QObject *)),
|
|
|
|
this, SLOT(objectAdded(QObject *)));
|
|
|
|
QObject::connect(ExtensionSystem::PluginManager::instance(), SIGNAL(aboutToRemoveObject(QObject *)),
|
|
|
|
this, SLOT(aboutToRemoveObject(QObject *)));
|
2013-04-05 23:46:56 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
void ModeManager::addWidget(QWidget *widget)
|
|
|
|
{
|
|
|
|
Q_UNUSED(widget);
|
|
|
|
|
|
|
|
// We want the actionbar to stay on the bottom
|
|
|
|
// so m_modeStack->cornerWidgetCount() -1 inserts it at the position immediately above
|
|
|
|
// the actionbar
|
2013-05-19 17:37:30 +03:00
|
|
|
// m_modeStack->insertCornerWidget(m_modeStack->cornerWidgetCount() -1, widget);
|
2013-04-05 23:46:56 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
IMode *ModeManager::currentMode() const
|
|
|
|
{
|
2013-05-19 17:37:30 +03:00
|
|
|
if (m_modes.count() > m_modeStack->currentIndex()) {
|
2013-04-05 23:46:56 +03:00
|
|
|
return m_modes.at(m_modeStack->currentIndex());
|
2013-05-19 17:37:30 +03:00
|
|
|
} else {
|
2013-04-05 23:46:56 +03:00
|
|
|
m_modeStack->setCurrentIndex(0); // Fix illegal Index.
|
2013-05-19 17:37:30 +03:00
|
|
|
}
|
2013-04-05 23:46:56 +03:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int ModeManager::indexOf(const QString &id) const
|
|
|
|
{
|
|
|
|
for (int i = 0; i < m_modes.count(); ++i) {
|
2013-05-19 17:37:30 +03:00
|
|
|
if (m_modes.at(i)->uniqueModeName() == id) {
|
2013-04-05 23:46:56 +03:00
|
|
|
return i;
|
2013-05-19 17:37:30 +03:00
|
|
|
}
|
2013-04-05 23:46:56 +03:00
|
|
|
}
|
2013-05-19 17:37:30 +03:00
|
|
|
// qDebug() << "Warning, no such mode:" << id;
|
2013-04-05 23:46:56 +03:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
IMode *ModeManager::mode(const QString &id) const
|
|
|
|
{
|
|
|
|
const int index = indexOf(id);
|
2013-05-19 17:37:30 +03:00
|
|
|
|
|
|
|
if (index >= 0) {
|
2013-04-05 23:46:56 +03:00
|
|
|
return m_modes.at(index);
|
2013-05-19 17:37:30 +03:00
|
|
|
}
|
2013-04-05 23:46:56 +03:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void ModeManager::activateMode(const QString &id)
|
|
|
|
{
|
|
|
|
const int index = indexOf(id);
|
2013-05-19 17:37:30 +03:00
|
|
|
|
|
|
|
if (index >= 0) {
|
2013-04-05 23:46:56 +03:00
|
|
|
m_modeStack->setCurrentIndex(index);
|
2013-05-19 17:37:30 +03:00
|
|
|
}
|
2013-04-05 23:46:56 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
void ModeManager::activateModeByWorkspaceName(const QString &id)
|
|
|
|
{
|
|
|
|
for (int i = 0; i < m_modes.count(); ++i) {
|
2013-05-19 17:37:30 +03:00
|
|
|
if (m_modes.at(i)->name() == id) {
|
2013-04-05 23:46:56 +03:00
|
|
|
m_modeStack->setCurrentIndex(i);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void ModeManager::objectAdded(QObject *obj)
|
|
|
|
{
|
|
|
|
IMode *mode = Aggregation::query<IMode>(obj);
|
2013-05-19 17:37:30 +03:00
|
|
|
|
|
|
|
if (!mode) {
|
2013-04-05 23:46:56 +03:00
|
|
|
return;
|
2013-05-19 17:37:30 +03:00
|
|
|
}
|
2013-04-05 23:46:56 +03:00
|
|
|
|
|
|
|
m_mainWindow->addContextObject(mode);
|
|
|
|
|
|
|
|
// Count the number of modes with a higher priority
|
|
|
|
int index = 0;
|
2013-05-19 17:37:30 +03:00
|
|
|
foreach(const IMode * m, m_modes)
|
|
|
|
if (m->priority() > mode->priority()) {
|
|
|
|
++index;
|
|
|
|
}
|
2013-04-05 23:46:56 +03:00
|
|
|
|
|
|
|
m_modes.insert(index, mode);
|
|
|
|
m_modeStack->insertTab(index, mode->widget(), mode->icon(), mode->name());
|
|
|
|
|
|
|
|
// Register mode shortcut
|
2013-05-19 17:37:30 +03:00
|
|
|
ActionManager *am = m_mainWindow->actionManager();
|
2013-04-05 23:46:56 +03:00
|
|
|
const QString shortcutId = QLatin1String("GCS.Mode.") + mode->uniqueModeName();
|
|
|
|
QShortcut *shortcut = new QShortcut(m_mainWindow);
|
|
|
|
shortcut->setWhatsThis(tr("Switch to %1 mode").arg(mode->name()));
|
|
|
|
Command *cmd = am->registerShortcut(shortcut, shortcutId, QList<int>() << Constants::C_GLOBAL_ID);
|
|
|
|
|
|
|
|
m_modeShortcuts.insert(index, cmd);
|
|
|
|
connect(cmd, SIGNAL(keySequenceChanged()), this, SLOT(updateModeToolTip()));
|
|
|
|
|
|
|
|
setDefaultKeyshortcuts();
|
|
|
|
|
|
|
|
m_signalMapper->setMapping(shortcut, mode->uniqueModeName());
|
|
|
|
connect(shortcut, SIGNAL(activated()), m_signalMapper, SLOT(map()));
|
|
|
|
}
|
|
|
|
|
2013-05-19 17:37:30 +03:00
|
|
|
void ModeManager::setDefaultKeyshortcuts()
|
|
|
|
{
|
2013-04-05 23:46:56 +03:00
|
|
|
for (int i = 0; i < m_modeShortcuts.size(); ++i) {
|
|
|
|
Command *currentCmd = m_modeShortcuts.at(i);
|
|
|
|
bool currentlyHasDefaultSequence = (currentCmd->keySequence()
|
|
|
|
== currentCmd->defaultKeySequence());
|
|
|
|
#ifdef Q_WS_MAC
|
2013-05-19 17:37:30 +03:00
|
|
|
currentCmd->setDefaultKeySequence(QKeySequence(QString("Meta+%1").arg(i + 1)));
|
2013-04-05 23:46:56 +03:00
|
|
|
#else
|
2013-05-19 17:37:30 +03:00
|
|
|
currentCmd->setDefaultKeySequence(QKeySequence(QString("Ctrl+%1").arg(i + 1)));
|
2013-04-05 23:46:56 +03:00
|
|
|
#endif
|
2013-05-19 17:37:30 +03:00
|
|
|
if (currentlyHasDefaultSequence) {
|
2013-04-05 23:46:56 +03:00
|
|
|
currentCmd->setKeySequence(currentCmd->defaultKeySequence());
|
2013-05-19 17:37:30 +03:00
|
|
|
}
|
2013-04-05 23:46:56 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void ModeManager::updateModeToolTip()
|
|
|
|
{
|
|
|
|
Command *cmd = qobject_cast<Command *>(sender());
|
2013-05-19 17:37:30 +03:00
|
|
|
|
2013-04-05 23:46:56 +03:00
|
|
|
if (cmd) {
|
|
|
|
int index = m_modeShortcuts.indexOf(cmd);
|
2013-05-19 17:37:30 +03:00
|
|
|
if (index != -1) {
|
2013-04-05 23:46:56 +03:00
|
|
|
m_modeStack->setTabToolTip(index, cmd->stringWithAppendedShortcut(cmd->shortcut()->whatsThis()));
|
2013-05-19 17:37:30 +03:00
|
|
|
}
|
2013-04-05 23:46:56 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void ModeManager::updateModeNameIcon(IMode *mode, const QIcon &icon, const QString &label)
|
|
|
|
{
|
|
|
|
int index = indexOf(mode->uniqueModeName());
|
2013-05-19 17:37:30 +03:00
|
|
|
|
|
|
|
if (index < 0) {
|
2013-04-05 23:46:56 +03:00
|
|
|
return;
|
2013-05-19 17:37:30 +03:00
|
|
|
}
|
2013-04-05 23:46:56 +03:00
|
|
|
m_modeStack->setTabIcon(index, icon);
|
|
|
|
m_modeStack->setTabText(index, label);
|
|
|
|
}
|
|
|
|
|
|
|
|
void ModeManager::aboutToRemoveObject(QObject *obj)
|
|
|
|
{
|
|
|
|
IMode *mode = Aggregation::query<IMode>(obj);
|
2013-05-19 17:37:30 +03:00
|
|
|
|
|
|
|
if (!mode) {
|
2013-04-05 23:46:56 +03:00
|
|
|
return;
|
2013-05-19 17:37:30 +03:00
|
|
|
}
|
2013-04-05 23:46:56 +03:00
|
|
|
|
|
|
|
const int index = m_modes.indexOf(mode);
|
|
|
|
m_modes.remove(index);
|
|
|
|
m_modeShortcuts.remove(index);
|
|
|
|
disconnect(m_modeStack, SIGNAL(currentChanged(int)), this, SLOT(currentTabChanged(int)));
|
|
|
|
m_modeStack->removeTab(index);
|
|
|
|
connect(m_modeStack, SIGNAL(currentChanged(int)), this, SLOT(currentTabChanged(int)));
|
|
|
|
|
|
|
|
m_mainWindow->removeContextObject(mode);
|
|
|
|
}
|
|
|
|
|
|
|
|
void ModeManager::addAction(Command *command, int priority, QMenu *menu)
|
|
|
|
{
|
|
|
|
Q_UNUSED(menu);
|
|
|
|
|
|
|
|
m_actions.insert(command, priority);
|
|
|
|
|
|
|
|
// Count the number of commands with a higher priority
|
|
|
|
int index = 0;
|
2013-05-19 17:37:30 +03:00
|
|
|
foreach(int p, m_actions.values())
|
|
|
|
if (p > priority) {
|
|
|
|
++index;
|
|
|
|
}
|
2013-04-05 23:46:56 +03:00
|
|
|
|
2013-05-19 17:37:30 +03:00
|
|
|
// m_actionBar->insertAction(index, command->action(), menu);
|
2013-04-05 23:46:56 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
void ModeManager::currentTabAboutToChange(int index)
|
|
|
|
{
|
|
|
|
if (index >= 0) {
|
|
|
|
IMode *mode = m_modes.at(index);
|
2013-05-19 17:37:30 +03:00
|
|
|
if (mode) {
|
2013-04-05 23:46:56 +03:00
|
|
|
emit currentModeAboutToChange(mode);
|
2013-05-19 17:37:30 +03:00
|
|
|
}
|
2013-04-05 23:46:56 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void ModeManager::currentTabChanged(int index)
|
|
|
|
{
|
2013-05-19 17:37:30 +03:00
|
|
|
// qDebug() << "Current tab changed " << index;
|
2013-04-05 23:46:56 +03:00
|
|
|
// Tab index changes to -1 when there is no tab left.
|
|
|
|
if (index >= 0) {
|
|
|
|
IMode *mode = m_modes.at(index);
|
|
|
|
|
|
|
|
// 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 *core = ICore::instance();
|
2013-05-19 17:37:30 +03:00
|
|
|
foreach(const int context, m_addedContexts)
|
|
|
|
core->removeAdditionalContext(context);
|
2013-04-05 23:46:56 +03:00
|
|
|
|
|
|
|
m_addedContexts = mode->context();
|
2013-05-19 17:37:30 +03:00
|
|
|
foreach(const int context, m_addedContexts)
|
|
|
|
core->addAdditionalContext(context);
|
2013-04-05 23:46:56 +03:00
|
|
|
emit currentModeChanged(mode);
|
|
|
|
core->updateContext();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void ModeManager::tabMoved(int from, int to)
|
|
|
|
{
|
|
|
|
IMode *mode = m_modes.at(from);
|
2013-05-19 17:37:30 +03:00
|
|
|
|
2013-04-05 23:46:56 +03:00
|
|
|
m_modes.remove(from);
|
|
|
|
m_modes.insert(to, mode);
|
|
|
|
Command *cmd = m_modeShortcuts.at(from);
|
|
|
|
m_modeShortcuts.remove(from);
|
|
|
|
m_modeShortcuts.insert(to, cmd);
|
|
|
|
setDefaultKeyshortcuts();
|
|
|
|
// Reprioritize, high priority means show to the left
|
|
|
|
if (!m_isReprioritizing) {
|
|
|
|
for (int i = 0; i < m_modes.count(); ++i) {
|
2013-05-19 17:37:30 +03:00
|
|
|
m_modes.at(i)->setPriority(100 - i);
|
2013-04-05 23:46:56 +03:00
|
|
|
}
|
|
|
|
emit newModeOrder(m_modes);
|
2013-05-19 17:37:30 +03:00
|
|
|
}
|
2013-04-05 23:46:56 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
void ModeManager::reorderModes(QMap<QString, int> priorities)
|
|
|
|
{
|
2013-05-19 17:37:30 +03:00
|
|
|
foreach(IMode * mode, m_modes)
|
|
|
|
mode->setPriority(priorities.value(QString(QLatin1String(mode->uniqueModeName())), mode->priority()));
|
2013-04-05 23:46:56 +03:00
|
|
|
|
|
|
|
m_isReprioritizing = true;
|
|
|
|
IMode *current = currentMode();
|
|
|
|
// Bubble sort
|
2013-05-19 17:37:30 +03:00
|
|
|
bool swapped = false;
|
2013-04-05 23:46:56 +03:00
|
|
|
do {
|
|
|
|
swapped = false;
|
2013-05-19 17:37:30 +03:00
|
|
|
for (int i = 0; i < m_modes.count() - 1; ++i) {
|
2013-04-05 23:46:56 +03:00
|
|
|
IMode *mode1 = m_modes.at(i);
|
2013-05-19 17:37:30 +03:00
|
|
|
IMode *mode2 = m_modes.at(i + 1);
|
|
|
|
// qDebug() << "Comparing " << i << " to " << i+1 << " p1 " << mode1->priority() << " p2 " << mode2->priority();
|
2013-04-05 23:46:56 +03:00
|
|
|
if (mode2->priority() > mode1->priority()) {
|
2013-05-19 17:37:30 +03:00
|
|
|
m_modeStack->moveTab(i, i + 1);
|
|
|
|
// qDebug() << "Tab moved from " << i << " to " << i+1;
|
2013-04-05 23:46:56 +03:00
|
|
|
swapped = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} while (swapped);
|
|
|
|
m_isReprioritizing = false;
|
|
|
|
m_modeStack->setCurrentIndex(0);
|
|
|
|
activateMode(current->uniqueModeName());
|
|
|
|
emit newModeOrder(m_modes);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void ModeManager::setFocusToCurrentMode()
|
|
|
|
{
|
2013-05-19 17:37:30 +03:00
|
|
|
IMode *mode = currentMode();
|
|
|
|
|
|
|
|
QTC_ASSERT(mode, return );
|
2013-04-05 23:46:56 +03:00
|
|
|
QWidget *widget = mode->widget();
|
|
|
|
if (widget) {
|
|
|
|
QWidget *focusWidget = widget->focusWidget();
|
2013-05-19 17:37:30 +03:00
|
|
|
|
|
|
|
if (focusWidget) {
|
2013-04-05 23:46:56 +03:00
|
|
|
focusWidget->setFocus();
|
2013-05-19 17:37:30 +03:00
|
|
|
} else {
|
2013-04-05 23:46:56 +03:00
|
|
|
widget->setFocus();
|
2013-05-19 17:37:30 +03:00
|
|
|
}
|
2013-04-05 23:46:56 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void ModeManager::triggerAction(const QString &actionId)
|
|
|
|
{
|
2013-05-19 17:37:30 +03:00
|
|
|
foreach(Command * command, m_actions.keys()) {
|
|
|
|
if (command->action()->objectName() == actionId) {
|
2013-04-05 23:46:56 +03:00
|
|
|
command->action()->trigger();
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|