forked from qt-creator/qt-creator
Change-Id: I9f4f8c0e499eeb1e686d1094f3442c415b845c21 Reviewed-by: <github-actions-qt-creator@cristianadam.eu> Reviewed-by: hjk <hjk@qt.io>
391 lines
13 KiB
C++
391 lines
13 KiB
C++
// Copyright (C) 2016 The Qt Company Ltd.
|
|
// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR GPL-3.0+ OR GPL-3.0 WITH Qt-GPL-exception-1.0
|
|
|
|
#include "qmlprofilerstatisticsview.h"
|
|
#include "qmlprofilertool.h"
|
|
#include "qmlprofilertr.h"
|
|
|
|
#include <coreplugin/minisplitter.h>
|
|
#include <utils/qtcassert.h>
|
|
#include <tracing/timelineformattime.h>
|
|
|
|
#include <QHeaderView>
|
|
#include <QApplication>
|
|
#include <QClipboard>
|
|
#include <QVBoxLayout>
|
|
#include <QMenu>
|
|
#include <QSortFilterProxyModel>
|
|
|
|
#include <functional>
|
|
|
|
namespace QmlProfiler {
|
|
namespace Internal {
|
|
|
|
const int DEFAULT_SORT_COLUMN = MainTimeInPercent;
|
|
|
|
static void setViewDefaults(Utils::TreeView *view)
|
|
{
|
|
view->setFrameStyle(QFrame::NoFrame);
|
|
QHeaderView *header = view->header();
|
|
header->setSectionResizeMode(QHeaderView::Interactive);
|
|
header->setDefaultSectionSize(100);
|
|
header->setMinimumSectionSize(50);
|
|
}
|
|
|
|
static void getSourceLocation(const QModelIndex &index,
|
|
std::function<void (const QString &, int, int)> receiver)
|
|
{
|
|
const int line = index.data(LineRole).toInt();
|
|
const int column = index.data(ColumnRole).toInt();
|
|
const QString fileName = index.data(FilenameRole).toString();
|
|
if (line != -1 && !fileName.isEmpty())
|
|
receiver(fileName, line, column);
|
|
}
|
|
|
|
QmlProfilerStatisticsView::QmlProfilerStatisticsView(QmlProfilerModelManager *profilerModelManager,
|
|
QWidget *parent)
|
|
: QmlProfilerEventsView(parent)
|
|
{
|
|
setObjectName(QLatin1String("QmlProfiler.Statistics.Dock"));
|
|
setWindowTitle(Tr::tr("Statistics"));
|
|
|
|
auto model = new QmlProfilerStatisticsModel(profilerModelManager);
|
|
m_mainView.reset(new QmlProfilerStatisticsMainView(model));
|
|
connect(m_mainView.get(), &QmlProfilerStatisticsMainView::gotoSourceLocation,
|
|
this, &QmlProfilerStatisticsView::gotoSourceLocation);
|
|
|
|
connect(m_mainView.get(), &QmlProfilerStatisticsMainView::typeClicked,
|
|
this, [this, profilerModelManager](int typeIndex) {
|
|
// Statistics view has an extra type for "whole program". Translate that into "invalid" for
|
|
// others.
|
|
emit typeSelected((typeIndex < profilerModelManager->numEventTypes())
|
|
? typeIndex : QmlProfilerStatisticsModel::s_invalidTypeId);
|
|
});
|
|
|
|
m_calleesView.reset(new QmlProfilerStatisticsRelativesView(
|
|
new QmlProfilerStatisticsRelativesModel(profilerModelManager, model,
|
|
QmlProfilerStatisticsCallees)));
|
|
m_callersView.reset(new QmlProfilerStatisticsRelativesView(
|
|
new QmlProfilerStatisticsRelativesModel(profilerModelManager, model,
|
|
QmlProfilerStatisticsCallers)));
|
|
connect(m_calleesView.get(), &QmlProfilerStatisticsRelativesView::typeClicked,
|
|
m_mainView.get(), &QmlProfilerStatisticsMainView::jumpToItem);
|
|
connect(m_callersView.get(), &QmlProfilerStatisticsRelativesView::typeClicked,
|
|
m_mainView.get(), &QmlProfilerStatisticsMainView::jumpToItem);
|
|
connect(m_mainView.get(), &QmlProfilerStatisticsMainView::propagateTypeIndex,
|
|
m_calleesView.get(), &QmlProfilerStatisticsRelativesView::displayType);
|
|
connect(m_mainView.get(), &QmlProfilerStatisticsMainView::propagateTypeIndex,
|
|
m_callersView.get(), &QmlProfilerStatisticsRelativesView::displayType);
|
|
|
|
// widget arrangement
|
|
auto groupLayout = new QVBoxLayout;
|
|
groupLayout->setContentsMargins(0,0,0,0);
|
|
groupLayout->setSpacing(0);
|
|
|
|
auto splitterVertical = new Core::MiniSplitter;
|
|
splitterVertical->addWidget(m_mainView.get());
|
|
auto splitterHorizontal = new Core::MiniSplitter;
|
|
splitterHorizontal->addWidget(m_callersView.get());
|
|
splitterHorizontal->addWidget(m_calleesView.get());
|
|
splitterHorizontal->setOrientation(Qt::Horizontal);
|
|
splitterVertical->addWidget(splitterHorizontal);
|
|
splitterVertical->setOrientation(Qt::Vertical);
|
|
splitterVertical->setStretchFactor(0,5);
|
|
splitterVertical->setStretchFactor(1,2);
|
|
groupLayout->addWidget(splitterVertical);
|
|
setLayout(groupLayout);
|
|
}
|
|
|
|
QString QmlProfilerStatisticsView::summary(const QVector<int> &typeIds) const
|
|
{
|
|
return m_mainView->summary(typeIds);
|
|
}
|
|
|
|
QStringList QmlProfilerStatisticsView::details(int typeId) const
|
|
{
|
|
return m_mainView->details(typeId);
|
|
}
|
|
|
|
void QmlProfilerStatisticsView::contextMenuEvent(QContextMenuEvent *ev)
|
|
{
|
|
QMenu menu;
|
|
QAction *copyRowAction = nullptr;
|
|
QAction *copyTableAction = nullptr;
|
|
QAction *showExtendedStatsAction = nullptr;
|
|
QAction *getGlobalStatsAction = nullptr;
|
|
|
|
QPoint position = ev->globalPos();
|
|
|
|
const QList <QAction *> commonActions = QmlProfilerTool::profilerContextMenuActions();
|
|
for (QAction *act : commonActions)
|
|
menu.addAction(act);
|
|
|
|
if (mouseOnTable(position)) {
|
|
menu.addSeparator();
|
|
if (m_mainView->selectedModelIndex().isValid())
|
|
copyRowAction = menu.addAction(Tr::tr("Copy Row"));
|
|
copyTableAction = menu.addAction(Tr::tr("Copy Table"));
|
|
|
|
showExtendedStatsAction = menu.addAction(Tr::tr("Extended Event Statistics"));
|
|
showExtendedStatsAction->setCheckable(true);
|
|
showExtendedStatsAction->setChecked(m_mainView->showExtendedStatistics());
|
|
}
|
|
|
|
menu.addSeparator();
|
|
getGlobalStatsAction = menu.addAction(Tr::tr("Show Full Range"));
|
|
if (!m_mainView->isRestrictedToRange())
|
|
getGlobalStatsAction->setEnabled(false);
|
|
|
|
QAction *selectedAction = menu.exec(position);
|
|
|
|
if (selectedAction) {
|
|
if (selectedAction == copyRowAction)
|
|
m_mainView->copyRowToClipboard();
|
|
if (selectedAction == copyTableAction)
|
|
m_mainView->copyTableToClipboard();
|
|
if (selectedAction == getGlobalStatsAction)
|
|
emit showFullRange();
|
|
if (selectedAction == showExtendedStatsAction)
|
|
m_mainView->setShowExtendedStatistics(showExtendedStatsAction->isChecked());
|
|
}
|
|
}
|
|
|
|
bool QmlProfilerStatisticsView::mouseOnTable(const QPoint &position) const
|
|
{
|
|
QPoint tableTopLeft = m_mainView->mapToGlobal(QPoint(0,0));
|
|
QPoint tableBottomRight = m_mainView->mapToGlobal(QPoint(m_mainView->width(),
|
|
m_mainView->height()));
|
|
return position.x() >= tableTopLeft.x() && position.x() <= tableBottomRight.x()
|
|
&& position.y() >= tableTopLeft.y() && position.y() <= tableBottomRight.y();
|
|
}
|
|
|
|
void QmlProfilerStatisticsView::selectByTypeId(int typeIndex)
|
|
{
|
|
// Other models cannot discern between "nothing" and "Main Program". So don't propagate invalid
|
|
// typeId, if we already have whole program selected.
|
|
if (typeIndex >= 0
|
|
|| m_mainView->currentIndex().data(TypeIdRole).toInt()
|
|
!= QmlProfilerStatisticsModel::s_mainEntryTypeId) {
|
|
m_mainView->displayTypeIndex(typeIndex);
|
|
}
|
|
}
|
|
|
|
void QmlProfilerStatisticsView::onVisibleFeaturesChanged(quint64 features)
|
|
{
|
|
m_mainView->restrictToFeatures(features);
|
|
}
|
|
|
|
QmlProfilerStatisticsMainView::QmlProfilerStatisticsMainView(QmlProfilerStatisticsModel *model) :
|
|
m_model(model)
|
|
{
|
|
setViewDefaults(this);
|
|
setObjectName(QLatin1String("QmlProfilerEventsTable"));
|
|
|
|
auto sortModel = new QSortFilterProxyModel(this);
|
|
sortModel->setSourceModel(model);
|
|
sortModel->setSortRole(SortRole);
|
|
sortModel->setSortCaseSensitivity(Qt::CaseInsensitive);
|
|
sortModel->setFilterRole(FilterRole);
|
|
sortModel->setFilterKeyColumn(MainType);
|
|
sortModel->setFilterFixedString("+");
|
|
|
|
setModel(sortModel);
|
|
|
|
connect(this, &QAbstractItemView::activated, this, [this](const QModelIndex &index) {
|
|
jumpToItem(index.data(TypeIdRole).toInt());
|
|
});
|
|
|
|
setSortingEnabled(true);
|
|
sortByColumn(DEFAULT_SORT_COLUMN, Qt::DescendingOrder);
|
|
|
|
setShowExtendedStatistics(m_showExtendedStatistics);
|
|
setRootIsDecorated(false);
|
|
|
|
resizeColumnToContents(MainLocation);
|
|
resizeColumnToContents(MainType);
|
|
}
|
|
|
|
QmlProfilerStatisticsMainView::~QmlProfilerStatisticsMainView() = default;
|
|
|
|
void QmlProfilerStatisticsMainView::setShowExtendedStatistics(bool show)
|
|
{
|
|
// Not checking if already set because we don't want the first call to skip
|
|
m_showExtendedStatistics = show;
|
|
if (show) {
|
|
showColumn(MainMedianTime);
|
|
showColumn(MainMaxTime);
|
|
showColumn(MainMinTime);
|
|
} else {
|
|
hideColumn(MainMedianTime);
|
|
hideColumn(MainMaxTime);
|
|
hideColumn(MainMinTime);
|
|
}
|
|
}
|
|
|
|
bool QmlProfilerStatisticsMainView::showExtendedStatistics() const
|
|
{
|
|
return m_showExtendedStatistics;
|
|
}
|
|
|
|
void QmlProfilerStatisticsMainView::restrictToFeatures(quint64 features)
|
|
{
|
|
m_model->restrictToFeatures(features);
|
|
}
|
|
|
|
bool QmlProfilerStatisticsMainView::isRestrictedToRange() const
|
|
{
|
|
return m_model->isRestrictedToRange();
|
|
}
|
|
|
|
QString QmlProfilerStatisticsMainView::summary(const QVector<int> &typeIds) const
|
|
{
|
|
return m_model->summary(typeIds);
|
|
}
|
|
|
|
QStringList QmlProfilerStatisticsMainView::details(int typeId) const
|
|
{
|
|
return m_model->details(typeId);
|
|
}
|
|
|
|
void QmlProfilerStatisticsMainView::jumpToItem(int typeIndex)
|
|
{
|
|
displayTypeIndex(typeIndex);
|
|
|
|
auto sortModel = qobject_cast<const QSortFilterProxyModel *>(model());
|
|
QTC_ASSERT(sortModel, return);
|
|
|
|
QAbstractItemModel *sourceModel = sortModel->sourceModel();
|
|
QTC_ASSERT(sourceModel, return);
|
|
|
|
// show in editor
|
|
getSourceLocation(sourceModel->index(typeIndex, MainLocation),
|
|
[this](const QString &fileName, int line, int column) {
|
|
emit gotoSourceLocation(fileName, line, column);
|
|
});
|
|
|
|
emit typeClicked(typeIndex);
|
|
}
|
|
|
|
void QmlProfilerStatisticsMainView::displayTypeIndex(int typeIndex)
|
|
{
|
|
if (typeIndex < 0) {
|
|
setCurrentIndex(QModelIndex());
|
|
} else {
|
|
auto sortModel = qobject_cast<const QSortFilterProxyModel *>(model());
|
|
QTC_ASSERT(sortModel, return);
|
|
|
|
QAbstractItemModel *sourceModel = sortModel->sourceModel();
|
|
QTC_ASSERT(sourceModel, return);
|
|
|
|
if (typeIndex < sourceModel->rowCount()) {
|
|
QModelIndex sourceIndex = sourceModel->index(typeIndex, MainCallCount);
|
|
QTC_ASSERT(sourceIndex.data(TypeIdRole).toInt() == typeIndex, return);
|
|
setCurrentIndex(sourceIndex.data(SortRole).toInt() > 0
|
|
? sortModel->mapFromSource(sourceIndex)
|
|
: QModelIndex());
|
|
} else {
|
|
setCurrentIndex(QModelIndex());
|
|
}
|
|
}
|
|
|
|
// show in callers/callees subwindow
|
|
emit propagateTypeIndex(typeIndex);
|
|
}
|
|
|
|
QModelIndex QmlProfilerStatisticsMainView::selectedModelIndex() const
|
|
{
|
|
QModelIndexList sel = selectedIndexes();
|
|
if (sel.isEmpty())
|
|
return QModelIndex();
|
|
else
|
|
return sel.first();
|
|
}
|
|
|
|
QString QmlProfilerStatisticsMainView::textForItem(const QModelIndex &index) const
|
|
{
|
|
QString str;
|
|
|
|
// item's data
|
|
const int colCount = model()->columnCount();
|
|
for (int j = 0; j < colCount; ++j) {
|
|
const QModelIndex cellIndex = model()->index(index.row(), j);
|
|
str += cellIndex.data(Qt::DisplayRole).toString();
|
|
if (j < colCount-1) str += QLatin1Char('\t');
|
|
}
|
|
str += QLatin1Char('\n');
|
|
|
|
return str;
|
|
}
|
|
|
|
void QmlProfilerStatisticsMainView::copyTableToClipboard() const
|
|
{
|
|
QString str;
|
|
|
|
const QAbstractItemModel *itemModel = model();
|
|
|
|
// headers
|
|
const int columnCount = itemModel->columnCount();
|
|
for (int i = 0; i < columnCount; ++i) {
|
|
str += itemModel->headerData(i, Qt::Horizontal, Qt::DisplayRole).toString();
|
|
if (i < columnCount - 1)
|
|
str += QLatin1Char('\t');
|
|
else
|
|
str += QLatin1Char('\n');
|
|
}
|
|
|
|
// data
|
|
const int rowCount = itemModel->rowCount();
|
|
for (int i = 0; i != rowCount; ++i)
|
|
str += textForItem(itemModel->index(i, 0));
|
|
|
|
QClipboard *clipboard = QApplication::clipboard();
|
|
if (clipboard->supportsSelection())
|
|
clipboard->setText(str, QClipboard::Selection);
|
|
clipboard->setText(str, QClipboard::Clipboard);
|
|
}
|
|
|
|
void QmlProfilerStatisticsMainView::copyRowToClipboard() const
|
|
{
|
|
QString str = textForItem(selectedModelIndex());
|
|
QClipboard *clipboard = QApplication::clipboard();
|
|
if (clipboard->supportsSelection())
|
|
clipboard->setText(str, QClipboard::Selection);
|
|
clipboard->setText(str, QClipboard::Clipboard);
|
|
}
|
|
|
|
QmlProfilerStatisticsRelativesView::QmlProfilerStatisticsRelativesView(
|
|
QmlProfilerStatisticsRelativesModel *model) :
|
|
m_model(model)
|
|
{
|
|
setViewDefaults(this);
|
|
auto sortModel = new QSortFilterProxyModel(this);
|
|
sortModel->setSourceModel(model);
|
|
sortModel->setSortRole(SortRole);
|
|
sortModel->setSortCaseSensitivity(Qt::CaseInsensitive);
|
|
setModel(sortModel);
|
|
setRootIsDecorated(false);
|
|
|
|
setSortingEnabled(true);
|
|
sortByColumn(DEFAULT_SORT_COLUMN, Qt::DescendingOrder);
|
|
|
|
connect(this, &QAbstractItemView::activated, this, [this](const QModelIndex &index) {
|
|
jumpToItem(index.data(TypeIdRole).toInt());
|
|
});
|
|
}
|
|
|
|
QmlProfilerStatisticsRelativesView::~QmlProfilerStatisticsRelativesView() = default;
|
|
|
|
void QmlProfilerStatisticsRelativesView::displayType(int typeIndex)
|
|
{
|
|
model()->setData(QModelIndex(), typeIndex, TypeIdRole);
|
|
resizeColumnToContents(RelativeLocation);
|
|
}
|
|
|
|
void QmlProfilerStatisticsRelativesView::jumpToItem(int typeIndex)
|
|
{
|
|
emit typeClicked(typeIndex);
|
|
}
|
|
|
|
} // namespace Internal
|
|
} // namespace QmlProfiler
|