2010-07-16 11:18:30 +02:00
|
|
|
/**************************************************************************
|
|
|
|
|
**
|
2013-01-28 17:12:19 +01:00
|
|
|
** Copyright (c) 2013 Denis Mingulov
|
2012-10-02 09:12:39 +02:00
|
|
|
** Contact: http://www.qt-project.org/legal
|
2010-07-16 11:18:30 +02:00
|
|
|
**
|
2012-10-02 09:12:39 +02:00
|
|
|
** This file is part of Qt Creator.
|
2010-07-16 11:18:30 +02:00
|
|
|
**
|
2012-10-02 09:12:39 +02: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.
|
2010-07-16 11:18:30 +02:00
|
|
|
**
|
|
|
|
|
** GNU Lesser General Public License Usage
|
2012-10-02 09:12:39 +02: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.
|
|
|
|
|
**
|
|
|
|
|
** 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.
|
|
|
|
|
**
|
2012-10-02 09:12:39 +02:00
|
|
|
****************************************************************************/
|
2010-07-16 11:18:30 +02:00
|
|
|
|
|
|
|
|
#include "classviewnavigationwidget.h"
|
|
|
|
|
#include "ui_classviewnavigationwidget.h"
|
|
|
|
|
#include "classviewtreeitemmodel.h"
|
|
|
|
|
#include "classviewmanager.h"
|
|
|
|
|
#include "classviewsymbollocation.h"
|
|
|
|
|
#include "classviewsymbolinformation.h"
|
|
|
|
|
#include "classviewutils.h"
|
|
|
|
|
#include "classviewconstants.h"
|
|
|
|
|
|
|
|
|
|
#include <utils/qtcassert.h>
|
|
|
|
|
|
2012-02-15 10:42:41 +01:00
|
|
|
#include <QDebug>
|
|
|
|
|
#include <QPointer>
|
2010-07-16 11:18:30 +02:00
|
|
|
|
|
|
|
|
enum { debug = false };
|
|
|
|
|
|
|
|
|
|
namespace ClassView {
|
|
|
|
|
namespace Internal {
|
|
|
|
|
|
|
|
|
|
///////////////////////////////// NavigationWidgetPrivate //////////////////////////////////
|
|
|
|
|
|
|
|
|
|
/*!
|
2013-05-24 17:35:14 +02:00
|
|
|
\class NavigationWidgetPrivate
|
|
|
|
|
|
|
|
|
|
The NavigationWidgetPrivate class provides internal data structures and
|
|
|
|
|
methods for NavigationWidget.
|
2010-07-16 11:18:30 +02:00
|
|
|
*/
|
|
|
|
|
|
2011-07-06 17:40:54 +02:00
|
|
|
class NavigationWidgetPrivate
|
2010-07-16 11:18:30 +02:00
|
|
|
{
|
2011-07-06 17:40:54 +02:00
|
|
|
public:
|
2010-07-16 11:18:30 +02:00
|
|
|
NavigationWidgetPrivate() : ui(0) {}
|
|
|
|
|
|
|
|
|
|
//! Ui generated by Designer
|
|
|
|
|
Ui::NavigationWidget *ui;
|
|
|
|
|
|
|
|
|
|
//! current tree model
|
|
|
|
|
QPointer<TreeItemModel> treeModel;
|
|
|
|
|
|
|
|
|
|
//! full projects mode
|
|
|
|
|
QPointer<QToolButton> fullProjectsModeButton;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
///////////////////////////////// NavigationWidget //////////////////////////////////
|
|
|
|
|
|
2013-05-24 17:35:14 +02:00
|
|
|
/*!
|
|
|
|
|
\class NavigationWidget
|
|
|
|
|
|
|
|
|
|
The NavigationWidget class is a widget for the class view tree.
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/*!
|
|
|
|
|
\fn void NavigationWidget::visibilityChanged(bool visibility)
|
|
|
|
|
|
|
|
|
|
Emits a signal when the widget visibility is changed. \a visibility returns
|
|
|
|
|
true if plugin becames visible, otherwise it returns false.
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
/*!
|
|
|
|
|
\fn void NavigationWidget::requestGotoLocation(const QString &name,
|
|
|
|
|
int line,
|
|
|
|
|
int column)
|
|
|
|
|
|
|
|
|
|
Emits a signal that requests to open the file with \a name at \a line
|
|
|
|
|
and \a column.
|
|
|
|
|
|
|
|
|
|
\sa Manager::gotoLocation
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
/*!
|
|
|
|
|
\fn void NavigationWidget::requestGotoLocations(const QList<QVariant> &locations)
|
|
|
|
|
|
|
|
|
|
Emits a signal to request to go to any of the Symbol \a locations in the
|
|
|
|
|
list.
|
|
|
|
|
|
|
|
|
|
\sa Manager::gotoLocations
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
/*!
|
|
|
|
|
\fn void NavigationWidget::requestTreeDataUpdate()
|
|
|
|
|
|
|
|
|
|
Emits a signal that the widget wants to receive the latest tree info.
|
|
|
|
|
|
|
|
|
|
\sa Manager::onRequestTreeDataUpdate
|
|
|
|
|
*/
|
2010-07-16 11:18:30 +02:00
|
|
|
|
|
|
|
|
NavigationWidget::NavigationWidget(QWidget *parent) :
|
|
|
|
|
QWidget(parent),
|
2011-07-06 17:40:54 +02:00
|
|
|
d(new NavigationWidgetPrivate())
|
2010-07-16 11:18:30 +02:00
|
|
|
{
|
2011-07-06 17:40:54 +02:00
|
|
|
d->ui = new Ui::NavigationWidget;
|
|
|
|
|
d->ui->setupUi(this);
|
2010-07-16 11:18:30 +02:00
|
|
|
|
|
|
|
|
// tree model
|
2011-07-06 17:40:54 +02:00
|
|
|
d->treeModel = new TreeItemModel(this);
|
|
|
|
|
d->ui->treeView->setModel(d->treeModel);
|
2010-07-16 11:18:30 +02:00
|
|
|
|
|
|
|
|
// connect signal/slots
|
|
|
|
|
// selected item
|
2011-07-06 17:40:54 +02:00
|
|
|
connect(d->ui->treeView, SIGNAL(activated(QModelIndex)), SLOT(onItemActivated(QModelIndex)));
|
2010-07-16 11:18:30 +02:00
|
|
|
|
|
|
|
|
// connections to the manager
|
|
|
|
|
Manager *manager = Manager::instance();
|
|
|
|
|
|
|
|
|
|
connect(this, SIGNAL(visibilityChanged(bool)),
|
|
|
|
|
manager, SLOT(onWidgetVisibilityIsChanged(bool)));
|
|
|
|
|
|
|
|
|
|
connect(this, SIGNAL(requestGotoLocation(QString,int,int)),
|
|
|
|
|
manager, SLOT(gotoLocation(QString,int,int)));
|
|
|
|
|
|
|
|
|
|
connect(this, SIGNAL(requestGotoLocations(QList<QVariant>)),
|
|
|
|
|
manager, SLOT(gotoLocations(QList<QVariant>)));
|
|
|
|
|
|
|
|
|
|
connect(manager, SIGNAL(treeDataUpdate(QSharedPointer<QStandardItem>)),
|
|
|
|
|
this, SLOT(onDataUpdate(QSharedPointer<QStandardItem>)));
|
|
|
|
|
|
|
|
|
|
connect(this, SIGNAL(requestTreeDataUpdate()),
|
|
|
|
|
manager, SLOT(onRequestTreeDataUpdate()));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
NavigationWidget::~NavigationWidget()
|
|
|
|
|
{
|
2011-07-06 17:40:54 +02:00
|
|
|
delete d->fullProjectsModeButton;
|
|
|
|
|
delete d->ui;
|
|
|
|
|
delete d;
|
2010-07-16 11:18:30 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void NavigationWidget::hideEvent(QHideEvent *event)
|
|
|
|
|
{
|
|
|
|
|
emit visibilityChanged(false);
|
|
|
|
|
QWidget::hideEvent(event);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void NavigationWidget::showEvent(QShowEvent *event)
|
|
|
|
|
{
|
|
|
|
|
emit visibilityChanged(true);
|
|
|
|
|
|
|
|
|
|
// request to update to the current state - to be sure
|
|
|
|
|
emit requestTreeDataUpdate();
|
|
|
|
|
|
|
|
|
|
QWidget::showEvent(event);
|
|
|
|
|
}
|
|
|
|
|
|
2013-05-24 17:35:14 +02:00
|
|
|
/*!
|
|
|
|
|
Creates QToolbuttons for the Navigation Pane widget.
|
|
|
|
|
|
|
|
|
|
Returns the list of created QToolButtons.
|
|
|
|
|
|
|
|
|
|
\sa NavigationWidgetFactory::createWidget
|
|
|
|
|
*/
|
|
|
|
|
|
2010-07-16 11:18:30 +02:00
|
|
|
QList<QToolButton *> NavigationWidget::createToolButtons()
|
|
|
|
|
{
|
|
|
|
|
QList<QToolButton *> list;
|
|
|
|
|
|
|
|
|
|
// full projects mode
|
2011-07-06 17:40:54 +02:00
|
|
|
if (!d->fullProjectsModeButton) {
|
2010-07-16 11:18:30 +02:00
|
|
|
// create a button
|
2011-07-06 17:40:54 +02:00
|
|
|
d->fullProjectsModeButton = new QToolButton();
|
|
|
|
|
d->fullProjectsModeButton->setIcon(
|
2010-07-16 11:18:30 +02:00
|
|
|
QIcon(QLatin1String(":/classview/images/hierarchicalmode.png")));
|
2011-07-06 17:40:54 +02:00
|
|
|
d->fullProjectsModeButton->setCheckable(true);
|
|
|
|
|
d->fullProjectsModeButton->setToolTip(tr("Show Subprojects"));
|
2010-07-16 11:18:30 +02:00
|
|
|
|
|
|
|
|
// by default - not a flat mode
|
|
|
|
|
setFlatMode(false);
|
|
|
|
|
|
|
|
|
|
// connections
|
2011-07-06 17:40:54 +02:00
|
|
|
connect(d->fullProjectsModeButton, SIGNAL(toggled(bool)),
|
2010-07-16 11:18:30 +02:00
|
|
|
this, SLOT(onFullProjectsModeToggled(bool)));
|
|
|
|
|
}
|
|
|
|
|
|
2011-07-06 17:40:54 +02:00
|
|
|
list << d->fullProjectsModeButton;
|
2010-07-16 11:18:30 +02:00
|
|
|
|
|
|
|
|
return list;
|
|
|
|
|
}
|
|
|
|
|
|
2013-05-24 17:35:14 +02:00
|
|
|
/*!
|
|
|
|
|
Returns flat mode state.
|
|
|
|
|
*/
|
|
|
|
|
|
2010-07-16 11:18:30 +02:00
|
|
|
bool NavigationWidget::flatMode() const
|
|
|
|
|
{
|
2011-07-06 17:40:54 +02:00
|
|
|
QTC_ASSERT(d->fullProjectsModeButton, return false);
|
2010-07-16 11:18:30 +02:00
|
|
|
|
|
|
|
|
// button is 'full projects mode' - so it has to be inverted
|
2011-07-06 17:40:54 +02:00
|
|
|
return !d->fullProjectsModeButton->isChecked();
|
2010-07-16 11:18:30 +02:00
|
|
|
}
|
|
|
|
|
|
2013-05-24 17:35:14 +02:00
|
|
|
/*!
|
|
|
|
|
Sets the flat mode state to \a flatMode.
|
|
|
|
|
*/
|
|
|
|
|
|
2010-07-16 11:18:30 +02:00
|
|
|
void NavigationWidget::setFlatMode(bool flatMode)
|
|
|
|
|
{
|
2011-07-06 17:40:54 +02:00
|
|
|
QTC_ASSERT(d->fullProjectsModeButton, return);
|
2010-07-16 11:18:30 +02:00
|
|
|
|
|
|
|
|
// button is 'full projects mode' - so it has to be inverted
|
2011-07-06 17:40:54 +02:00
|
|
|
d->fullProjectsModeButton->setChecked(!flatMode);
|
2010-07-16 11:18:30 +02:00
|
|
|
}
|
|
|
|
|
|
2013-05-24 17:35:14 +02:00
|
|
|
/*!
|
|
|
|
|
Full projects mode button has been toggled. \a state holds the full
|
|
|
|
|
projects mode.
|
|
|
|
|
*/
|
|
|
|
|
|
2010-07-16 11:18:30 +02:00
|
|
|
void NavigationWidget::onFullProjectsModeToggled(bool state)
|
|
|
|
|
{
|
|
|
|
|
// button is 'full projects mode' - so it has to be inverted
|
|
|
|
|
Manager::instance()->setFlatMode(!state);
|
|
|
|
|
}
|
|
|
|
|
|
2013-05-24 17:35:14 +02:00
|
|
|
/*!
|
|
|
|
|
Activates the item with the \a index in the tree view.
|
|
|
|
|
*/
|
|
|
|
|
|
2010-07-16 11:18:30 +02:00
|
|
|
void NavigationWidget::onItemActivated(const QModelIndex &index)
|
|
|
|
|
{
|
|
|
|
|
if (!index.isValid())
|
|
|
|
|
return;
|
|
|
|
|
|
2011-07-06 17:40:54 +02:00
|
|
|
QList<QVariant> list = d->treeModel->data(index, Constants::SymbolLocationsRole).toList();
|
2010-07-16 11:18:30 +02:00
|
|
|
|
|
|
|
|
emit requestGotoLocations(list);
|
|
|
|
|
}
|
|
|
|
|
|
2013-05-24 17:35:14 +02:00
|
|
|
/*!
|
|
|
|
|
Receives new data for the tree. \a result is a pointer to the Class View
|
|
|
|
|
model root item. The method does nothing if null is passed.
|
|
|
|
|
*/
|
|
|
|
|
|
2010-07-16 11:18:30 +02:00
|
|
|
void NavigationWidget::onDataUpdate(QSharedPointer<QStandardItem> result)
|
|
|
|
|
{
|
|
|
|
|
if (result.isNull())
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
QTime timer;
|
|
|
|
|
if (debug)
|
|
|
|
|
timer.start();
|
|
|
|
|
// update is received. root item must be updated - and received information
|
|
|
|
|
// might be just a root - if a lazy data population is enabled.
|
|
|
|
|
// so expanded items must be parsed and 'fetched'
|
|
|
|
|
|
2011-07-06 17:40:54 +02:00
|
|
|
fetchExpandedItems(result.data(), d->treeModel->invisibleRootItem());
|
2010-07-16 11:18:30 +02:00
|
|
|
|
2011-07-06 17:40:54 +02:00
|
|
|
d->treeModel->moveRootToTarget(result.data());
|
2010-07-16 11:18:30 +02:00
|
|
|
|
|
|
|
|
// expand top level projects
|
|
|
|
|
QModelIndex sessionIndex;
|
|
|
|
|
|
2011-07-06 17:40:54 +02:00
|
|
|
for (int i = 0; i < d->treeModel->rowCount(sessionIndex); ++i)
|
|
|
|
|
d->ui->treeView->expand(d->treeModel->index(i, 0, sessionIndex));
|
2010-07-16 11:18:30 +02:00
|
|
|
|
|
|
|
|
if (debug)
|
|
|
|
|
qDebug() << "Class View:" << QDateTime::currentDateTime().toString()
|
|
|
|
|
<< "TreeView is updated in" << timer.elapsed() << "msecs";
|
|
|
|
|
}
|
|
|
|
|
|
2013-05-24 17:35:14 +02:00
|
|
|
/*!
|
|
|
|
|
Fetches data for expanded items to make sure that the content will exist.
|
|
|
|
|
\a item and \a target do nothing if null is passed.
|
|
|
|
|
*/
|
|
|
|
|
|
2010-07-16 11:18:30 +02:00
|
|
|
void NavigationWidget::fetchExpandedItems(QStandardItem *item, const QStandardItem *target) const
|
|
|
|
|
{
|
|
|
|
|
if (!item || !target)
|
|
|
|
|
return;
|
|
|
|
|
|
2011-07-06 17:40:54 +02:00
|
|
|
const QModelIndex &parent = d->treeModel->indexFromItem(target);
|
|
|
|
|
if (d->ui->treeView->isExpanded(parent))
|
2010-07-16 11:18:30 +02:00
|
|
|
Manager::instance()->fetchMore(item, true);
|
|
|
|
|
|
|
|
|
|
int itemIndex = 0;
|
|
|
|
|
int targetIndex = 0;
|
|
|
|
|
int itemRows = item->rowCount();
|
|
|
|
|
int targetRows = target->rowCount();
|
|
|
|
|
|
|
|
|
|
while (itemIndex < itemRows && targetIndex < targetRows) {
|
|
|
|
|
QStandardItem *itemChild = item->child(itemIndex);
|
|
|
|
|
const QStandardItem *targetChild = target->child(targetIndex);
|
|
|
|
|
|
|
|
|
|
const SymbolInformation &itemInf = Utils::symbolInformationFromItem(itemChild);
|
|
|
|
|
const SymbolInformation &targetInf = Utils::symbolInformationFromItem(targetChild);
|
|
|
|
|
|
|
|
|
|
if (itemInf < targetInf) {
|
|
|
|
|
++itemIndex;
|
|
|
|
|
} else if (itemInf == targetInf) {
|
|
|
|
|
fetchExpandedItems(itemChild, targetChild);
|
|
|
|
|
++itemIndex;
|
|
|
|
|
++targetIndex;
|
|
|
|
|
} else {
|
|
|
|
|
++targetIndex;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
} // namespace Internal
|
|
|
|
|
} // namespace ClassView
|