Files
qt-creator/src/plugins/cppeditor/cppoutline.cpp

215 lines
6.5 KiB
C++
Raw Normal View History

2011-02-18 10:36:52 +01:00
/**************************************************************************
**
** This file is part of Qt Creator
**
** Copyright (c) 2011 Nokia Corporation and/or its subsidiary(-ies).
**
** Contact: Nokia Corporation (qt-info@nokia.com)
**
** No Commercial Usage
**
** This file contains pre-release code and may not be distributed.
** You may use this file in accordance with the terms and conditions
** contained in the Technology Preview License Agreement accompanying
** this package.
**
** 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, Nokia gives you certain additional
** rights. These rights are described in the Nokia Qt LGPL Exception
** version 1.1, included in the file LGPL_EXCEPTION.txt in this package.
**
** If you have questions regarding the use of this file, please contact
** Nokia at qt-info@nokia.com.
**
**************************************************************************/
2010-07-08 11:26:33 +02:00
#include "cppoutline.h"
#include <TranslationUnit.h>
#include <Symbol.h>
#include <coreplugin/ifile.h>
#include <coreplugin/editormanager/editormanager.h>
2010-07-08 11:26:33 +02:00
#include <cplusplus/OverviewModel.h>
#include <QtCore/QDebug>
#include <QtCore/QTimer>
#include <QtGui/QVBoxLayout>
#include <QtGui/QMenu>
2010-07-08 11:26:33 +02:00
using namespace CppEditor::Internal;
enum {
debug = false
};
CppOutlineTreeView::CppOutlineTreeView(QWidget *parent) :
Utils::NavigationTreeView(parent)
2010-07-08 11:26:33 +02:00
{
// see also QmlJSOutlineTreeView
setFocusPolicy(Qt::NoFocus);
setExpandsOnDoubleClick(false);
}
void CppOutlineTreeView::contextMenuEvent(QContextMenuEvent *event)
{
if (!event)
return;
QMenu contextMenu;
contextMenu.addAction(tr("Expand All"), this, SLOT(expandAll()));
contextMenu.addAction(tr("Collapse All"), this, SLOT(collapseAll()));
contextMenu.exec(event->globalPos());
event->accept();
}
CppOutlineFilterModel::CppOutlineFilterModel(CPlusPlus::OverviewModel *sourceModel, QObject *parent) :
QSortFilterProxyModel(parent),
m_sourceModel(sourceModel)
2010-07-08 11:26:33 +02:00
{
setSourceModel(m_sourceModel);
2010-07-08 11:26:33 +02:00
}
bool CppOutlineFilterModel::filterAcceptsRow(int sourceRow,
const QModelIndex &sourceParent) const
{
// ignore artifical "<Select Symbol>" entry
if (!sourceParent.isValid() && sourceRow == 0) {
return false;
}
// ignore generated symbols, e.g. by macro expansion (Q_OBJECT)
const QModelIndex sourceIndex = m_sourceModel->index(sourceRow, 0, sourceParent);
CPlusPlus::Symbol *symbol = m_sourceModel->symbolFromIndex(sourceIndex);
if (symbol && symbol->isGenerated())
return false;
2010-07-08 11:26:33 +02:00
return QSortFilterProxyModel::filterAcceptsRow(sourceRow, sourceParent);
}
CppOutlineWidget::CppOutlineWidget(CPPEditor *editor) :
TextEditor::IOutlineWidget(),
m_editor(editor),
m_treeView(new CppOutlineTreeView(this)),
m_model(m_editor->outlineModel()),
m_proxyModel(new CppOutlineFilterModel(m_model, this)),
2010-07-08 11:26:33 +02:00
m_enableCursorSync(true),
m_blockCursorSync(false)
{
QVBoxLayout *layout = new QVBoxLayout;
layout->setMargin(0);
layout->setSpacing(0);
layout->addWidget(m_treeView);
setLayout(layout);
m_treeView->setModel(m_proxyModel);
connect(m_model, SIGNAL(modelReset()), this, SLOT(modelUpdated()));
modelUpdated();
2010-07-08 11:26:33 +02:00
connect(m_editor, SIGNAL(outlineModelIndexChanged(QModelIndex)),
this, SLOT(updateSelectionInTree(QModelIndex)));
2010-07-08 11:26:33 +02:00
connect(m_treeView->selectionModel(), SIGNAL(selectionChanged(QItemSelection,QItemSelection)),
this, SLOT(updateSelectionInText(QItemSelection)));
connect(m_treeView, SIGNAL(doubleClicked(QModelIndex)),
this, SLOT(updateTextCursor(QModelIndex)));
2010-07-08 11:26:33 +02:00
}
QList<QAction*> CppOutlineWidget::filterMenuActions() const
{
return QList<QAction*>();
}
2010-07-08 11:26:33 +02:00
void CppOutlineWidget::setCursorSynchronization(bool syncWithCursor)
{
m_enableCursorSync = syncWithCursor;
if (m_enableCursorSync)
updateSelectionInTree(m_editor->outlineModelIndex());
2010-07-08 11:26:33 +02:00
}
void CppOutlineWidget::modelUpdated()
2010-07-08 11:26:33 +02:00
{
m_treeView->expandAll();
2010-07-08 11:26:33 +02:00
}
void CppOutlineWidget::updateSelectionInTree(const QModelIndex &index)
2010-07-08 11:26:33 +02:00
{
if (!syncCursor())
return;
QModelIndex proxyIndex = m_proxyModel->mapFromSource(index);
m_blockCursorSync = true;
if (debug)
qDebug() << "CppOutline - updating selection due to cursor move";
m_treeView->selectionModel()->select(proxyIndex, QItemSelectionModel::ClearAndSelect);
m_treeView->scrollTo(proxyIndex);
2010-07-08 11:26:33 +02:00
m_blockCursorSync = false;
}
void CppOutlineWidget::updateSelectionInText(const QItemSelection &selection)
{
if (!syncCursor())
return;
if (!selection.indexes().isEmpty()) {
QModelIndex proxyIndex = selection.indexes().first();
updateTextCursor(proxyIndex);
}
}
void CppOutlineWidget::updateTextCursor(const QModelIndex &proxyIndex)
{
QModelIndex index = m_proxyModel->mapToSource(proxyIndex);
CPlusPlus::Symbol *symbol = m_model->symbolFromIndex(index);
if (symbol) {
m_blockCursorSync = true;
if (debug)
qDebug() << "CppOutline - moving cursor to" << symbol->line() << symbol->column() - 1;
Core::EditorManager *editorManager = Core::EditorManager::instance();
editorManager->cutForwardNavigationHistory();
editorManager->addCurrentPositionToNavigationHistory();
// line has to be 1 based, column 0 based!
m_editor->gotoLine(symbol->line(), symbol->column() - 1);
m_blockCursorSync = false;
2010-07-08 11:26:33 +02:00
}
}
bool CppOutlineWidget::syncCursor()
{
return m_enableCursorSync && !m_blockCursorSync;
}
bool CppOutlineWidgetFactory::supportsEditor(Core::IEditor *editor) const
{
if (qobject_cast<CPPEditorEditable*>(editor))
return true;
return false;
}
TextEditor::IOutlineWidget *CppOutlineWidgetFactory::createWidget(Core::IEditor *editor)
{
CPPEditorEditable *cppEditable = qobject_cast<CPPEditorEditable*>(editor);
CPPEditor *cppEditor = qobject_cast<CPPEditor*>(cppEditable->widget());
Q_ASSERT(cppEditor);
CppOutlineWidget *widget = new CppOutlineWidget(cppEditor);
return widget;
}