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

431 lines
13 KiB
C++
Raw Normal View History

/****************************************************************************
2008-12-02 12:01:29 +01:00
**
** Copyright (C) 2014 Digia Plc and/or its subsidiary(-ies).
** Contact: http://www.qt-project.org/legal
2008-12-02 12:01:29 +01:00
**
** This file is part of Qt Creator.
2008-12-02 12:01:29 +01:00
**
** Commercial License Usage
** Licensees holding valid commercial Qt licenses may use this file in
** accordance with the commercial license agreement provided with the
** Software or, alternatively, in accordance with the terms contained in
** a written agreement between you and Digia. For licensing terms and
** conditions see http://qt.digia.com/licensing. For further information
** use the contact form at http://qt.digia.com/contact-us.
**
** GNU Lesser General Public License Usage
** 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.
**
** In addition, as a special exception, Digia gives you certain additional
** rights. These rights are described in the Digia Qt LGPL Exception
2010-12-17 16:01:08 +01:00
** version 1.1, included in the file LGPL_EXCEPTION.txt in this package.
**
****************************************************************************/
2008-12-02 14:09:21 +01:00
2008-12-02 12:01:29 +01:00
#include "navigationwidget.h"
2010-09-16 12:26:28 +02:00
#include "navigationsubwidget.h"
#include "icontext.h"
#include "icore.h"
#include "coreconstants.h"
#include "inavigationwidgetfactory.h"
#include "modemanager.h"
#include "actionmanager/actionmanager.h"
#include "actionmanager/command.h"
#include "id.h"
#include <QDebug>
#include <QSettings>
2008-12-02 12:01:29 +01:00
#include <QAction>
#include <QHBoxLayout>
#include <QResizeEvent>
#include <QStandardItemModel>
2008-12-02 12:01:29 +01:00
Q_DECLARE_METATYPE(Core::INavigationWidgetFactory *)
2010-09-16 12:26:28 +02:00
namespace Core {
2008-12-02 12:01:29 +01:00
NavigationWidgetPlaceHolder *NavigationWidgetPlaceHolder::m_current = 0;
NavigationWidgetPlaceHolder* NavigationWidgetPlaceHolder::current()
{
return m_current;
}
NavigationWidgetPlaceHolder::NavigationWidgetPlaceHolder(Core::IMode *mode, QWidget *parent)
:QWidget(parent), m_mode(mode)
{
setLayout(new QVBoxLayout);
layout()->setMargin(0);
connect(Core::ModeManager::instance(), SIGNAL(currentModeAboutToChange(Core::IMode*)),
this, SLOT(currentModeAboutToChange(Core::IMode*)));
2008-12-02 12:01:29 +01:00
}
NavigationWidgetPlaceHolder::~NavigationWidgetPlaceHolder()
{
if (m_current == this) {
if (NavigationWidget *nw = NavigationWidget::instance()) {
nw->setParent(0);
nw->hide();
}
2008-12-02 12:01:29 +01:00
}
}
void NavigationWidgetPlaceHolder::applyStoredSize(int width)
{
if (width) {
QSplitter *splitter = qobject_cast<QSplitter *>(parentWidget());
if (splitter) {
// A splitter we need to resize the splitter sizes
QList<int> sizes = splitter->sizes();
int index = splitter->indexOf(this);
int diff = width - sizes.at(index);
2008-12-09 11:07:24 +01:00
int adjust = sizes.count() > 1 ? (diff / (sizes.count() - 1)) : 0;
for (int i = 0; i < sizes.count(); ++i) {
2008-12-02 12:01:29 +01:00
if (i != index)
sizes[i] += adjust;
}
sizes[index]= width;
splitter->setSizes(sizes);
} else {
QSize s = size();
s.setWidth(width);
resize(s);
}
}
}
// This function does work even though the order in which
// the placeHolder get the signal is undefined.
// It does ensure that after all PlaceHolders got the signal
// m_current points to the current PlaceHolder, or zero if there
// is no PlaceHolder in this mode
// And that the parent of the NavigationWidget gets the correct parent
void NavigationWidgetPlaceHolder::currentModeAboutToChange(Core::IMode *mode)
{
NavigationWidget *navigationWidget = NavigationWidget::instance();
if (m_current == this) {
m_current = 0;
navigationWidget->setParent(0);
navigationWidget->hide();
navigationWidget->placeHolderChanged(m_current);
}
if (m_mode == mode) {
m_current = this;
int width = navigationWidget->storedWidth();
layout()->addWidget(navigationWidget);
navigationWidget->show();
applyStoredSize(width);
setVisible(navigationWidget->isShown());
navigationWidget->placeHolderChanged(m_current);
}
}
struct NavigationWidgetPrivate
{
2010-09-16 12:26:28 +02:00
explicit NavigationWidgetPrivate(QAction *toggleSideBarAction);
~NavigationWidgetPrivate() { delete m_factoryModel; }
2008-12-02 12:01:29 +01:00
2010-09-16 12:26:28 +02:00
QList<Internal::NavigationSubWidget *> m_subWidgets;
QHash<QAction *, Core::Id> m_actionMap;
QHash<Core::Id, Core::Command *> m_commandMap;
2010-09-16 12:26:28 +02:00
QStandardItemModel *m_factoryModel;
bool m_shown;
bool m_suppressed;
int m_width;
static NavigationWidget* m_instance;
QAction *m_toggleSideBarAction; // does not take ownership
2010-09-16 12:26:28 +02:00
};
NavigationWidgetPrivate::NavigationWidgetPrivate(QAction *toggleSideBarAction) :
m_factoryModel(new QStandardItemModel),
m_shown(true),
m_suppressed(false),
m_width(0),
m_toggleSideBarAction(toggleSideBarAction)
2008-12-02 12:01:29 +01:00
{
2010-09-16 12:26:28 +02:00
}
NavigationWidget *NavigationWidgetPrivate::m_instance = 0;
NavigationWidget::NavigationWidget(QAction *toggleSideBarAction) :
d(new NavigationWidgetPrivate(toggleSideBarAction))
{
d->m_factoryModel->setSortRole(FactoryPriorityRole);
2008-12-02 12:01:29 +01:00
setOrientation(Qt::Vertical);
2010-09-16 12:26:28 +02:00
d->m_instance = this;
2008-12-02 12:01:29 +01:00
}
NavigationWidget::~NavigationWidget()
{
2010-09-16 12:26:28 +02:00
NavigationWidgetPrivate::m_instance = 0;
delete d;
2008-12-02 12:01:29 +01:00
}
NavigationWidget *NavigationWidget::instance()
{
2010-09-16 12:26:28 +02:00
return NavigationWidgetPrivate::m_instance;
2008-12-02 12:01:29 +01:00
}
void NavigationWidget::setFactories(const QList<INavigationWidgetFactory *> &factories)
{
Context navicontext(Core::Constants::C_NAVIGATION_PANE);
foreach (INavigationWidgetFactory *factory, factories) {
const Id id = factory->id();
QAction *action= new QAction(tr("Activate %1 Pane").arg(factory->displayName()), this);
connect(action, SIGNAL(triggered()), this, SLOT(activateSubWidget()));
d->m_actionMap.insert(action, id);
Command *cmd = ActionManager::registerAction(action,
id.withPrefix("QtCreator.Sidebar."), navicontext);
cmd->setDefaultKeySequence(factory->activationSequence());
2010-09-16 12:26:28 +02:00
d->m_commandMap.insert(id, cmd);
QStandardItem *newRow = new QStandardItem(factory->displayName());
newRow->setData(qVariantFromValue(factory), FactoryObjectRole);
newRow->setData(QVariant::fromValue(factory->id()), FactoryIdRole);
newRow->setData(factory->priority(), FactoryPriorityRole);
2010-09-16 12:26:28 +02:00
d->m_factoryModel->appendRow(newRow);
}
2010-09-16 12:26:28 +02:00
d->m_factoryModel->sort(0);
updateToggleText();
}
2008-12-02 12:01:29 +01:00
int NavigationWidget::storedWidth()
{
2010-09-16 12:26:28 +02:00
return d->m_width;
2008-12-02 12:01:29 +01:00
}
QAbstractItemModel *NavigationWidget::factoryModel() const
{
2010-09-16 12:26:28 +02:00
return d->m_factoryModel;
}
void NavigationWidget::updateToggleText()
{
bool haveData = d->m_factoryModel->rowCount();
d->m_toggleSideBarAction->setVisible(haveData);
d->m_toggleSideBarAction->setEnabled(haveData && NavigationWidgetPlaceHolder::m_current);
if (isShown())
d->m_toggleSideBarAction->setToolTip(tr("Hide Sidebar"));
else
d->m_toggleSideBarAction->setToolTip(tr("Show Sidebar"));
}
2008-12-02 12:01:29 +01:00
void NavigationWidget::placeHolderChanged(NavigationWidgetPlaceHolder *holder)
{
2010-09-16 12:26:28 +02:00
d->m_toggleSideBarAction->setChecked(holder && isShown());
updateToggleText();
2008-12-02 12:01:29 +01:00
}
void NavigationWidget::resizeEvent(QResizeEvent *re)
{
2010-09-16 12:26:28 +02:00
if (d->m_width && re->size().width())
d->m_width = re->size().width();
2008-12-02 12:01:29 +01:00
MiniSplitter::resizeEvent(re);
}
2010-09-16 12:26:28 +02:00
Internal::NavigationSubWidget *NavigationWidget::insertSubItem(int position,int index)
2008-12-02 12:01:29 +01:00
{
2010-09-16 12:26:28 +02:00
for (int pos = position + 1; pos < d->m_subWidgets.size(); ++pos) {
d->m_subWidgets.at(pos)->setPosition(pos + 1);
}
2010-09-16 12:26:28 +02:00
Internal::NavigationSubWidget *nsw = new Internal::NavigationSubWidget(this, position, index);
connect(nsw, SIGNAL(splitMe()), this, SLOT(splitSubWidget()));
connect(nsw, SIGNAL(closeMe()), this, SLOT(closeSubWidget()));
2008-12-02 12:01:29 +01:00
insertWidget(position, nsw);
2010-09-16 12:26:28 +02:00
d->m_subWidgets.insert(position, nsw);
2008-12-02 12:01:29 +01:00
return nsw;
}
void NavigationWidget::activateSubWidget()
{
QAction *original = qobject_cast<QAction *>(sender());
Id id = d->m_actionMap[original];
activateSubWidget(id);
}
2008-12-02 12:01:29 +01:00
void NavigationWidget::activateSubWidget(const Id &factoryId)
{
setShown(true);
2010-09-16 12:26:28 +02:00
foreach (Internal::NavigationSubWidget *subWidget, d->m_subWidgets) {
if (subWidget->factory()->id() == factoryId) {
2008-12-02 12:01:29 +01:00
subWidget->setFocusWidget();
ICore::raiseWindow(this);
2008-12-02 12:01:29 +01:00
return;
}
}
2008-12-02 12:01:29 +01:00
int index = factoryIndex(factoryId);
if (index >= 0) {
2010-09-16 12:26:28 +02:00
d->m_subWidgets.first()->setFactoryIndex(index);
d->m_subWidgets.first()->setFocusWidget();
ICore::raiseWindow(this);
}
2008-12-02 12:01:29 +01:00
}
void NavigationWidget::splitSubWidget()
2008-12-02 12:01:29 +01:00
{
2010-09-16 12:26:28 +02:00
Internal::NavigationSubWidget *original = qobject_cast<Internal::NavigationSubWidget *>(sender());
2008-12-02 12:01:29 +01:00
int pos = indexOf(original) + 1;
insertSubItem(pos, original->factoryIndex());
2008-12-02 12:01:29 +01:00
}
void NavigationWidget::closeSubWidget()
2008-12-02 12:01:29 +01:00
{
2010-09-16 12:26:28 +02:00
if (d->m_subWidgets.count() != 1) {
Internal::NavigationSubWidget *subWidget = qobject_cast<Internal::NavigationSubWidget *>(sender());
subWidget->saveSettings();
2010-09-16 12:26:28 +02:00
d->m_subWidgets.removeOne(subWidget);
2008-12-02 12:01:29 +01:00
subWidget->hide();
subWidget->deleteLater();
} else {
setShown(false);
}
}
void NavigationWidget::saveSettings(QSettings *settings)
{
QStringList viewIds;
2010-09-16 12:26:28 +02:00
for (int i=0; i<d->m_subWidgets.count(); ++i) {
d->m_subWidgets.at(i)->saveSettings();
viewIds.append(d->m_subWidgets.at(i)->factory()->id().toString());
2008-12-02 12:01:29 +01:00
}
2010-10-21 11:56:25 +02:00
settings->setValue(QLatin1String("Navigation/Views"), viewIds);
settings->setValue(QLatin1String("Navigation/Visible"), isShown());
settings->setValue(QLatin1String("Navigation/VerticalPosition"), saveState());
settings->setValue(QLatin1String("Navigation/Width"), d->m_width);
2008-12-02 12:01:29 +01:00
}
void NavigationWidget::restoreSettings(QSettings *settings)
2008-12-02 12:01:29 +01:00
{
if (!d->m_factoryModel->rowCount()) {
// We have no widgets to show!
setShown(false);
return;
}
2010-10-21 11:56:25 +02:00
int version = settings->value(QLatin1String("Navigation/Version"), 1).toInt();
QStringList viewIds = settings->value(QLatin1String("Navigation/Views"),
QStringList(QLatin1String("Projects"))).toStringList();
bool restoreSplitterState = true;
if (version == 1) {
2010-10-21 11:56:25 +02:00
if (!viewIds.contains(QLatin1String("Open Documents"))) {
viewIds += QLatin1String("Open Documents");
restoreSplitterState = false;
2008-12-02 12:01:29 +01:00
}
2010-10-21 11:56:25 +02:00
settings->setValue(QLatin1String("Navigation/Version"), 2);
}
int position = 0;
foreach (const QString &id, viewIds) {
int index = factoryIndex(Id::fromString(id));
if (index >= 0) {
// Only add if the id was actually found!
insertSubItem(position, index);
++position;
} else {
restoreSplitterState = false;
}
2008-12-02 12:01:29 +01:00
}
if (d->m_subWidgets.isEmpty())
// Make sure we have at least the projects widget
insertSubItem(0, qMax(0, factoryIndex("Projects")));
2010-10-21 11:56:25 +02:00
setShown(settings->value(QLatin1String("Navigation/Visible"), true).toBool());
2008-12-02 12:01:29 +01:00
2010-10-21 11:56:25 +02:00
if (restoreSplitterState && settings->contains(QLatin1String("Navigation/VerticalPosition"))) {
restoreState(settings->value(QLatin1String("Navigation/VerticalPosition")).toByteArray());
} else {
QList<int> sizes;
sizes += 256;
for (int i = viewIds.size()-1; i > 0; --i)
sizes.prepend(512);
setSizes(sizes);
}
2008-12-02 12:01:29 +01:00
2010-10-21 11:56:25 +02:00
d->m_width = settings->value(QLatin1String("Navigation/Width"), 240).toInt();
if (d->m_width < 40)
d->m_width = 40;
2010-10-21 11:56:25 +02:00
2008-12-02 12:01:29 +01:00
// Apply
2010-10-21 11:56:25 +02:00
if (NavigationWidgetPlaceHolder::m_current)
2010-09-16 12:26:28 +02:00
NavigationWidgetPlaceHolder::m_current->applyStoredSize(d->m_width);
2008-12-02 12:01:29 +01:00
}
void NavigationWidget::closeSubWidgets()
{
2010-09-16 12:26:28 +02:00
foreach (Internal::NavigationSubWidget *subWidget, d->m_subWidgets) {
subWidget->saveSettings();
delete subWidget;
}
2010-09-16 12:26:28 +02:00
d->m_subWidgets.clear();
}
2008-12-02 12:01:29 +01:00
void NavigationWidget::setShown(bool b)
{
2010-09-16 12:26:28 +02:00
if (d->m_shown == b)
2008-12-02 12:01:29 +01:00
return;
bool haveData = d->m_factoryModel->rowCount();
2010-09-16 12:26:28 +02:00
d->m_shown = b;
if (NavigationWidgetPlaceHolder::m_current) {
NavigationWidgetPlaceHolder::m_current->setVisible(d->m_shown && !d->m_suppressed && haveData);
d->m_toggleSideBarAction->setChecked(d->m_shown && !d->m_suppressed && haveData);
} else {
2010-09-16 12:26:28 +02:00
d->m_toggleSideBarAction->setChecked(false);
}
updateToggleText();
2008-12-02 12:01:29 +01:00
}
bool NavigationWidget::isShown() const
{
2010-09-16 12:26:28 +02:00
return d->m_shown;
2008-12-02 12:01:29 +01:00
}
bool NavigationWidget::isSuppressed() const
{
2010-09-16 12:26:28 +02:00
return d->m_suppressed;
2008-12-02 12:01:29 +01:00
}
void NavigationWidget::setSuppressed(bool b)
{
2010-09-16 12:26:28 +02:00
if (d->m_suppressed == b)
2008-12-02 12:01:29 +01:00
return;
2010-09-16 12:26:28 +02:00
d->m_suppressed = b;
2008-12-02 12:01:29 +01:00
if (NavigationWidgetPlaceHolder::m_current)
2010-09-16 12:26:28 +02:00
NavigationWidgetPlaceHolder::m_current->setVisible(d->m_shown && !d->m_suppressed);
2008-12-02 12:01:29 +01:00
}
int NavigationWidget::factoryIndex(const Id &id)
2008-12-02 12:01:29 +01:00
{
2010-09-16 12:26:28 +02:00
for (int row = 0; row < d->m_factoryModel->rowCount(); ++row) {
if (d->m_factoryModel->data(d->m_factoryModel->index(row, 0), FactoryIdRole).value<Core::Id>() == id)
return row;
}
return -1;
2008-12-02 12:01:29 +01:00
}
QHash<Id, Command *> NavigationWidget::commandMap() const
2008-12-02 12:01:29 +01:00
{
2010-09-16 12:26:28 +02:00
return d->m_commandMap;
2008-12-02 12:01:29 +01:00
}
2010-09-16 12:26:28 +02:00
} // namespace Core