2010-07-16 11:18:30 +02:00
|
|
|
/**************************************************************************
|
|
|
|
|
**
|
2014-01-07 13:27:11 +01:00
|
|
|
** Copyright (c) 2014 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
|
2014-10-01 13:21:18 +02:00
|
|
|
** conditions see http://www.qt.io/licensing. For further information
|
|
|
|
|
** use the contact form at http://www.qt.io/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
|
2014-10-01 13:21:18 +02:00
|
|
|
** General Public License version 2.1 or version 3 as published by the Free
|
|
|
|
|
** Software Foundation and appearing in the file LICENSE.LGPLv21 and
|
|
|
|
|
** LICENSE.LGPLv3 included in the packaging of this file. Please review the
|
|
|
|
|
** following information to ensure the GNU Lesser General Public License
|
|
|
|
|
** requirements will be met: https://www.gnu.org/licenses/lgpl.html and
|
|
|
|
|
** http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.
|
2012-10-02 09:12:39 +02:00
|
|
|
**
|
|
|
|
|
** 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 "classviewmanager.h"
|
|
|
|
|
#include "classviewsymbollocation.h"
|
|
|
|
|
#include "classviewsymbolinformation.h"
|
|
|
|
|
#include "classviewutils.h"
|
|
|
|
|
#include "classviewconstants.h"
|
|
|
|
|
|
2014-07-31 18:15:19 +02:00
|
|
|
#include <coreplugin/find/itemviewfind.h>
|
2014-06-12 22:25:38 +03:00
|
|
|
#include <utils/navigationtreeview.h>
|
2010-07-16 11:18:30 +02:00
|
|
|
#include <utils/qtcassert.h>
|
|
|
|
|
|
2012-02-15 10:42:41 +01:00
|
|
|
#include <QDebug>
|
2014-06-12 22:25:38 +03:00
|
|
|
#include <QVBoxLayout>
|
2010-07-16 11:18:30 +02:00
|
|
|
|
|
|
|
|
enum { debug = false };
|
|
|
|
|
|
|
|
|
|
namespace ClassView {
|
|
|
|
|
namespace Internal {
|
|
|
|
|
|
|
|
|
|
///////////////////////////////// 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) :
|
2014-06-11 16:56:43 +02:00
|
|
|
QWidget(parent)
|
2010-07-16 11:18:30 +02:00
|
|
|
{
|
2014-06-11 17:50:45 +02:00
|
|
|
QVBoxLayout *verticalLayout = new QVBoxLayout(this);
|
|
|
|
|
verticalLayout->setSpacing(0);
|
|
|
|
|
verticalLayout->setContentsMargins(0, 0, 0, 0);
|
|
|
|
|
treeView = new ::Utils::NavigationTreeView(this);
|
|
|
|
|
treeView->setEditTriggers(QAbstractItemView::NoEditTriggers);
|
2014-10-17 10:22:26 +02:00
|
|
|
treeView->setDragEnabled(true);
|
|
|
|
|
treeView->setDragDropMode(QAbstractItemView::DragOnly);
|
|
|
|
|
treeView->setDefaultDropAction(Qt::MoveAction);
|
2014-07-31 18:15:19 +02:00
|
|
|
verticalLayout->addWidget(Core::ItemViewFind::createSearchableWrapper(
|
2014-08-11 17:31:27 +02:00
|
|
|
treeView, Core::ItemViewFind::DarkColored,
|
|
|
|
|
Core::ItemViewFind::FetchMoreWhileSearching));
|
2010-07-16 11:18:30 +02:00
|
|
|
|
|
|
|
|
// tree model
|
2014-06-11 16:56:43 +02:00
|
|
|
treeModel = new TreeItemModel(this);
|
2014-06-11 17:50:45 +02:00
|
|
|
treeView->setModel(treeModel);
|
2010-07-16 11:18:30 +02:00
|
|
|
|
|
|
|
|
// connect signal/slots
|
|
|
|
|
// selected item
|
2014-06-11 17:50:45 +02:00
|
|
|
connect(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()
|
|
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
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
|
2014-06-11 16:56:43 +02:00
|
|
|
if (!fullProjectsModeButton) {
|
2010-07-16 11:18:30 +02:00
|
|
|
// create a button
|
2014-06-11 16:56:43 +02:00
|
|
|
fullProjectsModeButton = new QToolButton();
|
|
|
|
|
fullProjectsModeButton->setIcon(
|
2010-07-16 11:18:30 +02:00
|
|
|
QIcon(QLatin1String(":/classview/images/hierarchicalmode.png")));
|
2014-06-11 16:56:43 +02:00
|
|
|
fullProjectsModeButton->setCheckable(true);
|
|
|
|
|
fullProjectsModeButton->setToolTip(tr("Show Subprojects"));
|
2010-07-16 11:18:30 +02:00
|
|
|
|
|
|
|
|
// by default - not a flat mode
|
|
|
|
|
setFlatMode(false);
|
|
|
|
|
|
|
|
|
|
// connections
|
2014-06-11 16:56:43 +02:00
|
|
|
connect(fullProjectsModeButton, SIGNAL(toggled(bool)),
|
2010-07-16 11:18:30 +02:00
|
|
|
this, SLOT(onFullProjectsModeToggled(bool)));
|
|
|
|
|
}
|
|
|
|
|
|
2014-06-11 16:56:43 +02:00
|
|
|
list << 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
|
|
|
|
|
{
|
2014-06-11 16:56:43 +02:00
|
|
|
QTC_ASSERT(fullProjectsModeButton, return false);
|
2010-07-16 11:18:30 +02:00
|
|
|
|
|
|
|
|
// button is 'full projects mode' - so it has to be inverted
|
2014-06-11 16:56:43 +02:00
|
|
|
return !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)
|
|
|
|
|
{
|
2014-06-11 16:56:43 +02:00
|
|
|
QTC_ASSERT(fullProjectsModeButton, return);
|
2010-07-16 11:18:30 +02:00
|
|
|
|
|
|
|
|
// button is 'full projects mode' - so it has to be inverted
|
2014-06-11 16:56:43 +02:00
|
|
|
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;
|
|
|
|
|
|
2014-06-11 16:56:43 +02:00
|
|
|
QList<QVariant> list = 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
|
2013-10-07 13:34:40 +02:00
|
|
|
model root item. The function does nothing if null is passed.
|
2013-05-24 17:35:14 +02:00
|
|
|
*/
|
|
|
|
|
|
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'
|
|
|
|
|
|
2014-06-11 16:56:43 +02:00
|
|
|
fetchExpandedItems(result.data(), treeModel->invisibleRootItem());
|
2010-07-16 11:18:30 +02:00
|
|
|
|
2014-06-11 16:56:43 +02:00
|
|
|
treeModel->moveRootToTarget(result.data());
|
2010-07-16 11:18:30 +02:00
|
|
|
|
|
|
|
|
// expand top level projects
|
|
|
|
|
QModelIndex sessionIndex;
|
|
|
|
|
|
2014-06-11 16:56:43 +02:00
|
|
|
for (int i = 0; i < treeModel->rowCount(sessionIndex); ++i)
|
2014-06-11 17:50:45 +02:00
|
|
|
treeView->expand(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;
|
|
|
|
|
|
2014-06-11 16:56:43 +02:00
|
|
|
const QModelIndex &parent = treeModel->indexFromItem(target);
|
2014-06-11 17:50:45 +02:00
|
|
|
if (treeView->isExpanded(parent) && Manager::instance()->canFetchMore(item, true))
|
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
|