0
0
mirror of https://github.com/yse/easy_profiler.git synced 2024-12-28 01:04:41 +08:00
easy_profiler/profiler_gui/blocks_tree_widget.cpp

232 lines
6.6 KiB
C++
Raw Normal View History

2016-06-26 18:46:51 +03:00
#include <QMenu>
#include <QContextMenuEvent>
#include "blocks_tree_widget.h"
//////////////////////////////////////////////////////////////////////////
MyTreeItem::MyTreeItem(const BlocksTree* _block, QTreeWidgetItem* _parent) : QTreeWidgetItem(_parent), m_block(_block)
{
}
const BlocksTree* MyTreeItem::block() const
{
return m_block;
}
void MyTreeItem::setTimeSmart(int _column, const ::profiler::timestamp_t& _time)
{
setData(_column, Qt::UserRole, _time);
setToolTip(_column, QString("%1 ns").arg(_time));
if (_time < 1e3)
{
setText(_column, QString("%1 ns").arg(_time));
}
else if (_time < 1e6)
{
setText(_column, QString("%1 us").arg(double(_time) * 1e-3, 0, 'f', 3));
}
else if (_time < 1e9)
{
setText(_column, QString("%1 ms").arg(double(_time) * 1e-6, 0, 'f', 3));
}
else
{
setText(_column, QString("%1 s").arg(double(_time) * 1e-9, 0, 'f', 3));
}
}
void MyTreeItem::setTimeMs(int _column, const ::profiler::timestamp_t& _time)
{
setData(_column, Qt::UserRole, _time);
setToolTip(_column, QString("%1 ns").arg(_time));
setText(_column, QString::number(double(_time) * 1e-6, 'g', 9));
}
//////////////////////////////////////////////////////////////////////////
MyTreeWidget::MyTreeWidget(const thread_blocks_tree_t& _blocksTree, QWidget* _parent) : QTreeWidget(_parent), m_beginTime(-1)
{
setAlternatingRowColors(true);
setItemsExpandable(true);
setAnimated(true);
setSortingEnabled(false);
setColumnCount(10);
auto header = new QTreeWidgetItem();
header->setText(0, "Name");
header->setText(1, "Duration");
header->setText(2, "Begin, ms");
header->setText(3, "End, ms");
header->setText(4, "Min dur.");
header->setText(5, "Max dur.");
header->setText(6, "Average dur.");
header->setText(7, "N Calls");
setHeaderItem(header);
setTreeInternal(_blocksTree);
setSortingEnabled(true);
sortByColumn(0, Qt::AscendingOrder);
sortByColumn(2, Qt::AscendingOrder);
connect(this, &QTreeWidget::itemExpanded, this, &MyTreeWidget::onItemExpand);
}
void MyTreeWidget::setTree(const thread_blocks_tree_t& _blocksTree)
{
m_itemblocks.clear();
clear();
disconnect(this, &QTreeWidget::itemExpanded, this, &MyTreeWidget::onItemExpand);
setSortingEnabled(false);
setTreeInternal(_blocksTree);
setSortingEnabled(true);
sortByColumn(0, Qt::AscendingOrder);
sortByColumn(2, Qt::AscendingOrder);
connect(this, &QTreeWidget::itemExpanded, this, &MyTreeWidget::onItemExpand);
}
void MyTreeWidget::setTreeInternal(const thread_blocks_tree_t& _blocksTree)
{
for (const auto& threadTree : _blocksTree)
{
const auto beginTime = threadTree.second.children.front().node->block()->getBegin();
if (m_beginTime > beginTime)
{
m_beginTime = beginTime;
}
}
for (const auto& threadTree : _blocksTree)
{
auto item = new MyTreeItem(&threadTree.second);
item->setText(0, QString("Thread %1").arg(threadTree.first));
setTreeInternal(threadTree.second.children, item);
addTopLevelItem(item);
}
}
void MyTreeWidget::setTreeInternal(const BlocksTree::children_t& _children, MyTreeItem* _parent)
{
for (const auto& child : _children)
{
const auto duration = child.node->block()->duration();
const auto beginTime = child.node->block()->getBegin() - m_beginTime;
const auto endTime = child.node->block()->getEnd() - m_beginTime;
auto item = new MyTreeItem(&child, _parent);
item->setText(0, child.node->getBlockName());
item->setTimeSmart(1, child.node->block()->duration());
item->setTimeMs(2, child.node->block()->getBegin() - m_beginTime);
item->setTimeMs(3, child.node->block()->getEnd() - m_beginTime);
if (child.total_statistics)
{
item->setTimeSmart(4, child.total_statistics->min_duration);
item->setTimeSmart(5, child.total_statistics->max_duration);
item->setTimeSmart(6, child.total_statistics->average_duration());
item->setText(7, QString::number(child.total_statistics->calls_number));
}
m_itemblocks[child.node] = item;
if (!child.children.empty())
{
setTreeInternal(child.children, item);
}
}
}
void MyTreeWidget::contextMenuEvent(QContextMenuEvent* _event)
{
const auto col = currentColumn();
auto item = static_cast<MyTreeItem*>(currentItem());
if (item == nullptr || col < 0)
{
return;
}
QMenu menu;
auto action = new QAction("Expand all", nullptr);
connect(action, &QAction::triggered, this, &MyTreeWidget::onExpandAllClicked);
menu.addAction(action);
action = new QAction("Collapse all", nullptr);
connect(action, &QAction::triggered, this, &MyTreeWidget::onCollapseAllClicked);
menu.addAction(action);
switch (col)
{
case 4:
{
menu.addSeparator();
auto itemAction = new ItemAction("Jump to such item", item);
connect(itemAction, &ItemAction::clicked, this, &MyTreeWidget::onJumpToMinItemClicked);
menu.addAction(itemAction);
break;
}
case 5:
{
menu.addSeparator();
auto itemAction = new ItemAction("Jump to such item", item);
connect(itemAction, &ItemAction::clicked, this, &MyTreeWidget::onJumpToMaxItemClicked);
menu.addAction(itemAction);
break;
}
}
menu.exec(QCursor::pos());
_event->accept();
}
void MyTreeWidget::onJumpToMinItemClicked(MyTreeItem* _item)
{
auto it = m_itemblocks.find(_item->block()->total_statistics->min_duration_block);
if (it != m_itemblocks.end())
{
scrollToItem(it->second, QAbstractItemView::PositionAtCenter);
setCurrentItem(it->second);
}
}
void MyTreeWidget::onJumpToMaxItemClicked(MyTreeItem* _item)
{
auto it = m_itemblocks.find(_item->block()->total_statistics->max_duration_block);
if (it != m_itemblocks.end())
{
scrollToItem(it->second, QAbstractItemView::PositionAtCenter);
setCurrentItem(it->second);
}
}
void MyTreeWidget::onCollapseAllClicked(bool)
{
collapseAll();
}
void MyTreeWidget::onExpandAllClicked(bool)
{
disconnect(this, &QTreeWidget::itemExpanded, this, &MyTreeWidget::onItemExpand);
expandAll();
resizeColumnToContents(0);
connect(this, &QTreeWidget::itemExpanded, this, &MyTreeWidget::onItemExpand);
}
void MyTreeWidget::onItemExpand(QTreeWidgetItem*)
{
resizeColumnToContents(0);
}
//////////////////////////////////////////////////////////////////////////