2016-08-08 22:45:57 +03:00
|
|
|
/************************************************************************
|
|
|
|
* file name : blocks_graphics_view.cpp
|
|
|
|
* ----------------- :
|
|
|
|
* creation time : 2016/06/26
|
|
|
|
* author : Victor Zarubkin
|
|
|
|
* email : v.s.zarubkin@gmail.com
|
|
|
|
* ----------------- :
|
|
|
|
* description : The file contains implementation of GraphicsScene and GraphicsView and
|
|
|
|
* : it's auxiliary classes for displyaing easy_profiler blocks tree.
|
|
|
|
* ----------------- :
|
|
|
|
* change log : * 2016/06/26 Victor Zarubkin: Moved sources from graphics_view.h
|
|
|
|
* : and renamed classes from My* to Prof*.
|
|
|
|
* :
|
|
|
|
* : * 2016/06/27 Victor Zarubkin: Added text shifting relatively to it's parent item.
|
|
|
|
* : Disabled border lines painting because of vertical lines painting bug.
|
|
|
|
* : Changed height of blocks. Variable thread-block height.
|
|
|
|
* :
|
|
|
|
* : * 2016/06/29 Victor Zarubkin: Highly optimized painting performance and memory consumption.
|
|
|
|
* :
|
|
|
|
* : * 2016/06/30 Victor Zarubkin: Replaced doubles with floats (in ProfBlockItem) for less memory consumption.
|
|
|
|
* :
|
|
|
|
* : *
|
|
|
|
* ----------------- :
|
2016-09-06 21:49:32 +03:00
|
|
|
* license : Lightweight profiler library for c++
|
|
|
|
* : Copyright(C) 2016 Sergey Yagovtsev, Victor Zarubkin
|
|
|
|
* :
|
|
|
|
* : This program is free software : you can redistribute it and / or modify
|
|
|
|
* : it under the terms of the GNU General Public License as published by
|
|
|
|
* : the Free Software Foundation, either version 3 of the License, or
|
|
|
|
* : (at your option) any later version.
|
|
|
|
* :
|
|
|
|
* : This program is distributed in the hope that it will be useful,
|
|
|
|
* : but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* : MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.See the
|
|
|
|
* : GNU General Public License for more details.
|
|
|
|
* :
|
|
|
|
* : You should have received a copy of the GNU General Public License
|
|
|
|
* : along with this program.If not, see <http://www.gnu.org/licenses/>.
|
2016-06-26 18:56:40 +03:00
|
|
|
************************************************************************/
|
2016-06-26 18:46:51 +03:00
|
|
|
|
2016-08-18 23:26:41 +03:00
|
|
|
#include <QGraphicsScene>
|
2016-06-26 18:46:51 +03:00
|
|
|
#include <QWheelEvent>
|
2016-06-26 20:54:16 +03:00
|
|
|
#include <QMouseEvent>
|
2016-09-09 00:07:27 +03:00
|
|
|
#include <QKeyEvent>
|
2016-06-26 20:54:16 +03:00
|
|
|
#include <QScrollBar>
|
2016-08-10 22:08:27 +03:00
|
|
|
#include <QGridLayout>
|
2016-08-18 23:26:41 +03:00
|
|
|
#include <QFont>
|
2016-07-31 18:48:41 +03:00
|
|
|
#include <QFontMetrics>
|
2016-08-11 23:43:34 +03:00
|
|
|
#include <QDebug>
|
2016-08-19 01:40:14 +03:00
|
|
|
#include <QSignalBlocker>
|
2016-06-30 02:57:57 +03:00
|
|
|
#include <math.h>
|
2016-07-10 01:24:31 +03:00
|
|
|
#include <algorithm>
|
2016-06-26 18:46:51 +03:00
|
|
|
#include "blocks_graphics_view.h"
|
|
|
|
|
2016-08-08 22:45:57 +03:00
|
|
|
#include "globals.h"
|
|
|
|
|
2016-08-06 14:50:31 +03:00
|
|
|
using namespace profiler_gui;
|
|
|
|
|
2016-07-31 18:48:41 +03:00
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
2016-06-26 18:46:51 +03:00
|
|
|
|
2016-08-23 22:42:20 +03:00
|
|
|
enum BlockItemState
|
|
|
|
{
|
|
|
|
BLOCK_ITEM_DO_NOT_PAINT = -1,
|
|
|
|
BLOCK_ITEM_UNCHANGED,
|
|
|
|
BLOCK_ITEM_DO_PAINT
|
|
|
|
};
|
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
|
2016-09-07 21:48:50 +03:00
|
|
|
const qreal MIN_SCALE = pow(::profiler_gui::SCALING_COEFFICIENT_INV, 70); // Up to 1000 sec scale
|
|
|
|
const qreal MAX_SCALE = pow(::profiler_gui::SCALING_COEFFICIENT, 45); // ~23000 --- Up to 10 ns scale
|
2016-08-21 22:44:03 +03:00
|
|
|
const qreal BASE_SCALE = pow(::profiler_gui::SCALING_COEFFICIENT_INV, 25); // ~0.003
|
2016-07-31 13:13:48 +03:00
|
|
|
|
2016-09-09 00:07:27 +03:00
|
|
|
const uint16_t GRAPHICS_ROW_SIZE = 18;
|
|
|
|
const uint16_t GRAPHICS_ROW_SPACING = 2;
|
|
|
|
const uint16_t GRAPHICS_ROW_SIZE_FULL = GRAPHICS_ROW_SIZE + GRAPHICS_ROW_SPACING;
|
|
|
|
const uint16_t THREADS_ROW_SPACING = 8;
|
|
|
|
const uint16_t TIMELINE_ROW_SIZE = 20;
|
2016-07-10 01:24:31 +03:00
|
|
|
|
2016-09-08 22:42:35 +03:00
|
|
|
const QRgb BORDERS_COLOR = ::profiler::colors::Grey700 & 0x00ffffff;// 0x00686868;
|
|
|
|
const QRgb BACKGROUND_1 = ::profiler::colors::Grey300;
|
|
|
|
const QRgb BACKGROUND_2 = ::profiler::colors::White;
|
|
|
|
const QRgb TIMELINE_BACKGROUND = 0x20000000 | (::profiler::colors::Grey800 & 0x00ffffff);// 0x20303030;
|
|
|
|
//const QRgb SELECTED_ITEM_COLOR = ::profiler::colors::Dark;// 0x000050a0;
|
2016-09-09 00:07:27 +03:00
|
|
|
const QColor CHRONOMETER_COLOR2 = QColor::fromRgba(0x40000000 | (::profiler::colors::Dark & 0x00ffffff));// 0x40408040);
|
2016-09-08 22:42:35 +03:00
|
|
|
|
|
|
|
inline QRgb selectedItemBorderColor(::profiler::color_t _color)
|
|
|
|
{
|
2016-09-09 00:07:27 +03:00
|
|
|
return ::profiler_gui::isLightColor(_color, 192) ? ::profiler::colors::Black : ::profiler::colors::RichRed;
|
|
|
|
//return ::profiler::colors::Black;
|
2016-09-08 22:42:35 +03:00
|
|
|
}
|
2016-08-01 22:21:59 +03:00
|
|
|
|
2016-09-07 21:37:13 +03:00
|
|
|
//const unsigned int TEST_PROGRESSION_BASE = 4;
|
2016-07-31 13:13:48 +03:00
|
|
|
|
2016-08-06 14:50:31 +03:00
|
|
|
const int FLICKER_INTERVAL = 16; // 60Hz
|
|
|
|
|
2016-09-11 16:53:34 +03:00
|
|
|
QFont EFont(const char* _family, int _size, int _weight = -1)
|
|
|
|
{
|
|
|
|
QFont f;
|
|
|
|
f.setStyleHint(QFont::Helvetica, QFont::PreferMatch);
|
|
|
|
f.setFamily(_family);
|
|
|
|
f.setPointSize(_size);
|
|
|
|
f.setWeight(_weight);
|
|
|
|
return f;
|
|
|
|
}
|
|
|
|
|
|
|
|
const auto BG_FONT = EFont("Helvetica", 10, QFont::Bold);
|
|
|
|
const auto CHRONOMETER_FONT = EFont("Helvetica", 16, QFont::Bold);
|
|
|
|
const auto ITEMS_FONT = EFont("Helvetica", 10, QFont::Medium);
|
|
|
|
const auto SELECTED_ITEM_FONT = EFont("Helvetica", 10, QFont::Bold);
|
|
|
|
|
|
|
|
#ifdef _WIN32
|
|
|
|
const qreal FONT_METRICS_FACTOR = 1.05;
|
|
|
|
#else
|
|
|
|
const qreal FONT_METRICS_FACTOR = 1.;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef max
|
|
|
|
#undef max
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef min
|
|
|
|
#undef min
|
|
|
|
#endif
|
2016-08-18 23:26:41 +03:00
|
|
|
|
2016-07-31 13:13:48 +03:00
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
auto const sign = [](int _value) { return _value < 0 ? -1 : 1; };
|
|
|
|
auto const absmin = [](int _a, int _b) { return abs(_a) < abs(_b) ? _a : _b; };
|
|
|
|
auto const clamp = [](qreal _minValue, qreal _value, qreal _maxValue) { return _value < _minValue ? _minValue : (_value > _maxValue ? _maxValue : _value); };
|
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
|
2016-08-01 22:21:59 +03:00
|
|
|
template <int N, class T>
|
|
|
|
inline T logn(T _value)
|
|
|
|
{
|
|
|
|
static const double div = 1.0 / log2((double)N);
|
|
|
|
return log2(_value) * div;
|
|
|
|
}
|
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
|
2016-09-08 22:42:35 +03:00
|
|
|
EasyGraphicsItem::EasyGraphicsItem(uint8_t _index, const::profiler::BlocksTreeRoot& _root)
|
2016-08-18 23:26:41 +03:00
|
|
|
: QGraphicsItem(nullptr)
|
2016-09-08 22:42:35 +03:00
|
|
|
, m_threadName(*_root.thread_name != 0 ? QString("%1 Thread %2").arg(_root.thread_name).arg(_root.thread_id) : QString("Thread %1").arg(_root.thread_id))
|
|
|
|
, m_pRoot(&_root)
|
2016-08-18 23:26:41 +03:00
|
|
|
, m_index(_index)
|
2016-06-30 02:57:57 +03:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2016-08-18 23:26:41 +03:00
|
|
|
EasyGraphicsItem::~EasyGraphicsItem()
|
2016-06-30 02:57:57 +03:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2016-08-18 23:26:41 +03:00
|
|
|
const EasyGraphicsView* EasyGraphicsItem::view() const
|
2016-08-08 22:45:57 +03:00
|
|
|
{
|
2016-08-18 23:26:41 +03:00
|
|
|
return static_cast<const EasyGraphicsView*>(scene()->parent());
|
2016-07-31 13:13:48 +03:00
|
|
|
}
|
|
|
|
|
2016-07-31 18:48:41 +03:00
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
|
2016-08-18 23:26:41 +03:00
|
|
|
QRectF EasyGraphicsItem::boundingRect() const
|
2016-06-30 02:57:57 +03:00
|
|
|
{
|
2016-07-31 13:13:48 +03:00
|
|
|
//const auto sceneView = view();
|
2016-07-31 18:48:41 +03:00
|
|
|
//return QRectF(m_boundingRect.left() - sceneView->offset() / sceneView->scale(), m_boundingRect.top(), m_boundingRect.width() * sceneView->scale(), m_boundingRect.height());
|
|
|
|
return m_boundingRect;
|
2016-06-30 02:57:57 +03:00
|
|
|
}
|
|
|
|
|
2016-07-31 18:48:41 +03:00
|
|
|
//////////////////////////////////////////////////////////////////////////
|
2016-07-31 13:13:48 +03:00
|
|
|
|
2016-09-09 00:07:27 +03:00
|
|
|
void EasyGraphicsItem::paint(QPainter* _painter, const QStyleOptionGraphicsItem*, QWidget*)
|
2016-07-10 01:24:31 +03:00
|
|
|
{
|
|
|
|
if (m_levels.empty() || m_levels.front().empty())
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2016-07-31 13:13:48 +03:00
|
|
|
const auto sceneView = view();
|
|
|
|
const auto visibleSceneRect = sceneView->visibleSceneRect(); // Current visible scene rect
|
2016-07-31 18:48:41 +03:00
|
|
|
const auto currentScale = sceneView->scale(); // Current GraphicsView scale
|
2016-07-31 13:13:48 +03:00
|
|
|
const auto offset = sceneView->offset();
|
|
|
|
const auto sceneLeft = offset, sceneRight = offset + visibleSceneRect.width() / currentScale;
|
|
|
|
|
|
|
|
//printf("VISIBLE = {%lf, %lf}\n", sceneLeft, sceneRight);
|
2016-07-10 01:24:31 +03:00
|
|
|
|
|
|
|
QRectF rect;
|
|
|
|
QBrush brush;
|
2016-09-07 21:48:50 +03:00
|
|
|
QRgb previousColor = 0, inverseColor = 0xffffffff, textColor = 0;
|
2016-08-04 23:35:38 +03:00
|
|
|
Qt::PenStyle previousPenStyle = Qt::NoPen;
|
2016-07-10 01:24:31 +03:00
|
|
|
brush.setStyle(Qt::SolidPattern);
|
|
|
|
|
|
|
|
_painter->save();
|
2016-08-24 01:00:24 +03:00
|
|
|
_painter->setFont(ITEMS_FONT);
|
2016-08-21 14:26:04 +03:00
|
|
|
|
2016-07-27 21:50:11 +03:00
|
|
|
// Reset indices of first visible item for each layer
|
|
|
|
const auto levelsNumber = levels();
|
2016-09-04 14:48:35 +03:00
|
|
|
for (uint8_t i = 1; i < levelsNumber; ++i) ::profiler_gui::set_max(m_levelsIndexes[i]);
|
2016-07-10 01:24:31 +03:00
|
|
|
|
2016-07-27 21:50:11 +03:00
|
|
|
|
2016-08-01 22:21:59 +03:00
|
|
|
// Search for first visible top-level item
|
2016-08-08 22:17:56 +03:00
|
|
|
auto& level0 = m_levels.front();
|
2016-08-30 22:51:18 +03:00
|
|
|
auto first = ::std::lower_bound(level0.begin(), level0.end(), sceneLeft, [](const ::profiler_gui::EasyBlockItem& _item, qreal _value)
|
2016-07-10 01:24:31 +03:00
|
|
|
{
|
|
|
|
return _item.left() < _value;
|
|
|
|
});
|
|
|
|
|
|
|
|
if (first != level0.end())
|
|
|
|
{
|
|
|
|
m_levelsIndexes[0] = first - level0.begin();
|
|
|
|
if (m_levelsIndexes[0] > 0)
|
|
|
|
m_levelsIndexes[0] -= 1;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2016-08-04 22:38:45 +03:00
|
|
|
m_levelsIndexes[0] = static_cast<unsigned int>(level0.size() - 1);
|
2016-07-10 01:24:31 +03:00
|
|
|
}
|
|
|
|
|
2016-07-27 21:50:11 +03:00
|
|
|
|
2016-09-04 14:48:35 +03:00
|
|
|
auto firstSync = ::std::lower_bound(m_pRoot->sync.begin(), m_pRoot->sync.end(), sceneLeft, [&sceneView](::profiler::block_index_t _index, qreal _value)
|
|
|
|
{
|
|
|
|
return sceneView->time2position(easyBlock(_index).tree.node->begin()) < _value;
|
|
|
|
});
|
|
|
|
|
|
|
|
if (firstSync != m_pRoot->sync.end())
|
|
|
|
{
|
|
|
|
if (firstSync != m_pRoot->sync.begin())
|
|
|
|
--firstSync;
|
|
|
|
}
|
|
|
|
else if (!m_pRoot->sync.empty())
|
|
|
|
{
|
|
|
|
firstSync = m_pRoot->sync.begin() + m_pRoot->sync.size() - 1;
|
|
|
|
}
|
|
|
|
firstSync = m_pRoot->sync.begin();
|
|
|
|
|
2016-07-31 13:13:48 +03:00
|
|
|
|
2016-07-27 21:50:11 +03:00
|
|
|
// This is to make _painter->drawText() work properly
|
|
|
|
// (it seems there is a bug in Qt5.6 when drawText called for big coordinates,
|
|
|
|
// drawRect at the same time called for actually same coordinates
|
2016-07-31 13:13:48 +03:00
|
|
|
// works fine without using this additional shifting)
|
2016-09-07 21:48:50 +03:00
|
|
|
//const auto dx = level0[m_levelsIndexes[0]].left() * currentScale;
|
|
|
|
const auto dx = offset * currentScale;
|
2016-07-27 21:50:11 +03:00
|
|
|
|
2016-08-10 22:08:27 +03:00
|
|
|
|
|
|
|
|
2016-07-31 13:13:48 +03:00
|
|
|
// Shifting coordinates to current screen offset
|
2016-09-07 21:48:50 +03:00
|
|
|
//_painter->setTransform(QTransform::fromTranslate(dx - offset * currentScale, -y()), true);
|
|
|
|
_painter->setTransform(QTransform::fromTranslate(0, -y()), true);
|
2016-08-10 22:08:27 +03:00
|
|
|
|
|
|
|
|
2016-08-30 22:51:18 +03:00
|
|
|
if (EASY_GLOBALS.draw_graphics_items_borders)
|
2016-08-04 23:35:38 +03:00
|
|
|
{
|
|
|
|
previousPenStyle = Qt::SolidLine;
|
|
|
|
_painter->setPen(BORDERS_COLOR);
|
|
|
|
}
|
2016-08-18 23:26:41 +03:00
|
|
|
else
|
|
|
|
{
|
|
|
|
_painter->setPen(Qt::NoPen);
|
|
|
|
}
|
2016-07-27 21:50:11 +03:00
|
|
|
|
2016-08-10 22:08:27 +03:00
|
|
|
|
2016-08-30 22:51:18 +03:00
|
|
|
static const auto MAX_CHILD_INDEX = ::profiler_gui::numeric_max<decltype(::profiler_gui::EasyBlockItem::children_begin)>();
|
|
|
|
auto const skip_children = [this, &levelsNumber](short next_level, decltype(::profiler_gui::EasyBlockItem::children_begin) children_begin)
|
2016-08-23 22:42:20 +03:00
|
|
|
{
|
|
|
|
// Mark that we would not paint children of current item
|
|
|
|
if (next_level < levelsNumber && children_begin != MAX_CHILD_INDEX)
|
|
|
|
m_levels[next_level][children_begin].state = BLOCK_ITEM_DO_NOT_PAINT;
|
|
|
|
};
|
|
|
|
|
2016-08-10 22:08:27 +03:00
|
|
|
|
2016-07-27 21:50:11 +03:00
|
|
|
// Iterate through layers and draw visible items
|
2016-08-08 23:03:11 +03:00
|
|
|
bool selectedItemsWasPainted = false;
|
2016-08-21 18:04:38 +03:00
|
|
|
const auto visibleBottom = visibleSceneRect.bottom() - 1;
|
2016-09-04 14:48:35 +03:00
|
|
|
for (uint8_t l = 0; l < levelsNumber; ++l)
|
2016-07-10 01:24:31 +03:00
|
|
|
{
|
|
|
|
auto& level = m_levels[l];
|
2016-08-11 23:43:34 +03:00
|
|
|
const short next_level = l + 1;
|
2016-08-23 22:42:20 +03:00
|
|
|
char state = BLOCK_ITEM_DO_PAINT;
|
2016-07-10 01:24:31 +03:00
|
|
|
|
2016-08-11 23:43:34 +03:00
|
|
|
const auto top = levelY(l);
|
2016-08-23 22:42:20 +03:00
|
|
|
qreal prevRight = -1e100;
|
2016-08-04 22:38:45 +03:00
|
|
|
for (unsigned int i = m_levelsIndexes[l], end = static_cast<unsigned int>(level.size()); i < end; ++i)
|
2016-07-10 01:24:31 +03:00
|
|
|
{
|
|
|
|
auto& item = level[i];
|
2016-07-31 13:13:48 +03:00
|
|
|
|
2016-08-23 22:42:20 +03:00
|
|
|
if (item.left() > sceneRight)
|
|
|
|
break; // This is first totally invisible item. No need to check other items.
|
|
|
|
|
|
|
|
if (item.state != BLOCK_ITEM_UNCHANGED)
|
2016-07-31 13:13:48 +03:00
|
|
|
{
|
|
|
|
state = item.state;
|
|
|
|
}
|
2016-07-10 01:24:31 +03:00
|
|
|
|
2016-08-23 22:42:20 +03:00
|
|
|
if (item.right() < sceneLeft || state == BLOCK_ITEM_DO_NOT_PAINT || top > visibleBottom || (top + item.totalHeight) < visibleSceneRect.top())
|
2016-07-10 01:24:31 +03:00
|
|
|
{
|
2016-08-01 22:21:59 +03:00
|
|
|
// This item is not visible
|
2016-08-23 22:42:20 +03:00
|
|
|
skip_children(next_level, item.children_begin);
|
|
|
|
continue;
|
|
|
|
}
|
2016-08-21 14:26:04 +03:00
|
|
|
|
2016-08-23 22:42:20 +03:00
|
|
|
const auto x = item.left() * currentScale - dx;
|
|
|
|
auto w = item.width() * currentScale;
|
|
|
|
if (x + w <= prevRight)
|
|
|
|
{
|
|
|
|
// This item is not visible
|
|
|
|
if (w < 20)
|
|
|
|
skip_children(next_level, item.children_begin);
|
2016-07-10 01:24:31 +03:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2016-08-30 22:51:18 +03:00
|
|
|
const auto& itemBlock = easyBlock(item.block);
|
2016-08-18 23:26:41 +03:00
|
|
|
int h = 0, flags = 0;
|
2016-08-30 22:51:18 +03:00
|
|
|
if (w < 20 || !itemBlock.expanded)
|
2016-07-10 01:24:31 +03:00
|
|
|
{
|
2016-07-27 21:50:11 +03:00
|
|
|
// Items which width is less than 20 will be painted as big rectangles which are hiding it's children
|
|
|
|
|
2016-08-23 22:42:20 +03:00
|
|
|
//x = item.left() * currentScale - dx;
|
2016-08-18 23:26:41 +03:00
|
|
|
h = item.totalHeight;
|
2016-08-21 18:04:38 +03:00
|
|
|
const auto dh = top + h - visibleBottom;
|
|
|
|
if (dh > 0)
|
|
|
|
h -= dh;
|
2016-07-27 21:50:11 +03:00
|
|
|
|
2016-08-30 22:51:18 +03:00
|
|
|
if (item.block == EASY_GLOBALS.selected_block)
|
2016-08-08 23:03:11 +03:00
|
|
|
selectedItemsWasPainted = true;
|
2016-08-08 22:35:21 +03:00
|
|
|
|
2016-09-08 22:42:35 +03:00
|
|
|
const bool colorChange = (previousColor != item.color);
|
|
|
|
if (colorChange)
|
|
|
|
{
|
|
|
|
// Set background color brush for rectangle
|
|
|
|
previousColor = item.color;
|
2016-09-07 21:48:50 +03:00
|
|
|
inverseColor = 0xffffffff - previousColor;
|
|
|
|
textColor = ::profiler_gui::textColorForRgb(previousColor);
|
2016-08-08 22:54:20 +03:00
|
|
|
brush.setColor(previousColor);
|
2016-08-08 22:35:21 +03:00
|
|
|
_painter->setBrush(brush);
|
2016-08-08 22:17:56 +03:00
|
|
|
}
|
2016-08-08 22:35:21 +03:00
|
|
|
|
2016-09-08 22:42:35 +03:00
|
|
|
if (EASY_GLOBALS.draw_graphics_items_borders && (previousPenStyle != Qt::SolidLine || colorChange))
|
|
|
|
{
|
|
|
|
// Restore pen for item which is wide enough to paint borders
|
|
|
|
previousPenStyle = Qt::SolidLine;
|
|
|
|
_painter->setPen(BORDERS_COLOR & inverseColor);// BORDERS_COLOR);
|
2016-07-31 13:13:48 +03:00
|
|
|
}
|
2016-07-27 21:50:11 +03:00
|
|
|
|
2016-08-23 22:42:20 +03:00
|
|
|
if (w < 2) w = 2;
|
2016-07-27 21:50:11 +03:00
|
|
|
|
2016-09-08 22:42:35 +03:00
|
|
|
// Draw rectangle
|
|
|
|
rect.setRect(x, top, w, h);
|
|
|
|
_painter->drawRect(rect);
|
2016-08-08 22:17:56 +03:00
|
|
|
|
2016-08-23 22:42:20 +03:00
|
|
|
prevRight = rect.right();
|
|
|
|
skip_children(next_level, item.children_begin);
|
2016-08-18 23:26:41 +03:00
|
|
|
if (w < 20)
|
|
|
|
continue;
|
2016-07-10 01:24:31 +03:00
|
|
|
|
2016-08-21 22:44:03 +03:00
|
|
|
if (item.totalHeight > GRAPHICS_ROW_SIZE)
|
|
|
|
flags = Qt::AlignCenter;
|
|
|
|
else if (!(item.width() < 1))
|
|
|
|
flags = Qt::AlignHCenter;
|
2016-08-18 23:26:41 +03:00
|
|
|
}
|
|
|
|
else
|
2016-07-27 21:50:11 +03:00
|
|
|
{
|
2016-08-18 23:26:41 +03:00
|
|
|
if (next_level < levelsNumber && item.children_begin != MAX_CHILD_INDEX)
|
2016-07-31 13:13:48 +03:00
|
|
|
{
|
2016-08-18 23:26:41 +03:00
|
|
|
if (m_levelsIndexes[next_level] == MAX_CHILD_INDEX)
|
|
|
|
{
|
|
|
|
// Mark first potentially visible child item on next sublevel
|
|
|
|
m_levelsIndexes[next_level] = item.children_begin;
|
|
|
|
}
|
2016-07-31 13:13:48 +03:00
|
|
|
|
2016-08-18 23:26:41 +03:00
|
|
|
// Mark children items that we want to draw them
|
2016-08-23 22:42:20 +03:00
|
|
|
m_levels[next_level][item.children_begin].state = BLOCK_ITEM_DO_PAINT;
|
2016-08-18 23:26:41 +03:00
|
|
|
}
|
2016-08-08 22:35:21 +03:00
|
|
|
|
2016-08-30 22:51:18 +03:00
|
|
|
if (item.block == EASY_GLOBALS.selected_block)
|
2016-08-18 23:26:41 +03:00
|
|
|
selectedItemsWasPainted = true;
|
|
|
|
|
2016-09-08 22:42:35 +03:00
|
|
|
const bool colorChange = (previousColor != item.color);
|
|
|
|
if (colorChange)
|
|
|
|
{
|
|
|
|
// Set background color brush for rectangle
|
|
|
|
previousColor = item.color;
|
2016-09-07 21:48:50 +03:00
|
|
|
inverseColor = 0xffffffff - previousColor;
|
|
|
|
textColor = ::profiler_gui::textColorForRgb(previousColor);
|
2016-08-08 22:54:20 +03:00
|
|
|
brush.setColor(previousColor);
|
|
|
|
_painter->setBrush(brush);
|
2016-08-08 22:35:21 +03:00
|
|
|
}
|
2016-08-18 23:26:41 +03:00
|
|
|
|
2016-09-08 22:42:35 +03:00
|
|
|
if (EASY_GLOBALS.draw_graphics_items_borders && (previousPenStyle != Qt::SolidLine || colorChange))
|
|
|
|
{
|
|
|
|
// Restore pen for item which is wide enough to paint borders
|
|
|
|
previousPenStyle = Qt::SolidLine;
|
|
|
|
_painter->setPen(BORDERS_COLOR & inverseColor);// BORDERS_COLOR);
|
2016-08-08 22:54:20 +03:00
|
|
|
}
|
2016-08-08 22:45:57 +03:00
|
|
|
|
2016-08-18 23:26:41 +03:00
|
|
|
// Draw rectangle
|
2016-08-23 22:42:20 +03:00
|
|
|
//x = item.left() * currentScale - dx;
|
2016-08-18 23:26:41 +03:00
|
|
|
h = GRAPHICS_ROW_SIZE;
|
2016-08-21 18:04:38 +03:00
|
|
|
const auto dh = top + h - visibleBottom;
|
|
|
|
if (dh > 0)
|
|
|
|
h -= dh;
|
|
|
|
|
|
|
|
rect.setRect(x, top, w, h);
|
2016-08-18 23:26:41 +03:00
|
|
|
_painter->drawRect(rect);
|
|
|
|
|
2016-08-21 22:44:03 +03:00
|
|
|
if (!(item.width() < 1))
|
|
|
|
flags = Qt::AlignHCenter;
|
2016-08-23 22:42:20 +03:00
|
|
|
|
|
|
|
prevRight = rect.right();
|
2016-08-18 23:26:41 +03:00
|
|
|
}
|
2016-08-08 22:45:57 +03:00
|
|
|
|
|
|
|
// Draw text-----------------------------------
|
|
|
|
// calculating text coordinates
|
2016-07-31 13:13:48 +03:00
|
|
|
auto xtext = x;
|
2016-07-27 21:50:11 +03:00
|
|
|
if (item.left() < sceneLeft)
|
|
|
|
{
|
2016-08-01 22:21:59 +03:00
|
|
|
// if item left border is out of screen then attach text to the left border of the screen
|
|
|
|
// to ensure text is always visible for items presenting on the screen.
|
2016-07-27 21:50:11 +03:00
|
|
|
w += (item.left() - sceneLeft) * currentScale;
|
2016-07-31 13:13:48 +03:00
|
|
|
xtext = sceneLeft * currentScale - dx;
|
2016-07-27 21:50:11 +03:00
|
|
|
}
|
2016-07-10 01:24:31 +03:00
|
|
|
|
2016-08-18 23:26:41 +03:00
|
|
|
if (item.right() > sceneRight)
|
|
|
|
{
|
|
|
|
w -= (item.right() - sceneRight) * currentScale;
|
|
|
|
}
|
|
|
|
|
|
|
|
rect.setRect(xtext + 1, top, w - 1, h);
|
2016-08-08 22:45:57 +03:00
|
|
|
|
|
|
|
// text will be painted with inverse color
|
2016-09-07 21:48:50 +03:00
|
|
|
//auto textColor = inverseColor < 0x00808080 ? profiler::colors::Black : profiler::colors::White;
|
|
|
|
//if (textColor == previousColor) textColor = 0;
|
|
|
|
_painter->setPen(QColor::fromRgb(textColor));
|
2016-08-08 22:45:57 +03:00
|
|
|
|
2016-09-08 22:42:35 +03:00
|
|
|
if (item.block == EASY_GLOBALS.selected_block)
|
|
|
|
_painter->setFont(SELECTED_ITEM_FONT);
|
|
|
|
|
2016-08-08 22:45:57 +03:00
|
|
|
// drawing text
|
2016-08-30 22:51:18 +03:00
|
|
|
auto name = *itemBlock.tree.node->name() != 0 ? itemBlock.tree.node->name() : easyDescriptor(itemBlock.tree.node->id()).name();
|
2016-08-28 18:19:12 +03:00
|
|
|
_painter->drawText(rect, flags, ::profiler_gui::toUnicode(name));
|
2016-08-08 22:45:57 +03:00
|
|
|
|
|
|
|
// restore previous pen color
|
|
|
|
if (previousPenStyle == Qt::NoPen)
|
|
|
|
_painter->setPen(Qt::NoPen);
|
|
|
|
else
|
2016-08-21 18:04:38 +03:00
|
|
|
_painter->setPen(BORDERS_COLOR & inverseColor);// BORDERS_COLOR); // restore pen for rectangle painting
|
2016-09-08 22:42:35 +03:00
|
|
|
|
|
|
|
// restore font
|
|
|
|
if (item.block == EASY_GLOBALS.selected_block)
|
|
|
|
_painter->setFont(ITEMS_FONT);
|
2016-08-08 22:45:57 +03:00
|
|
|
// END Draw text~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
2016-07-10 01:24:31 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-09-08 22:42:35 +03:00
|
|
|
if (EASY_GLOBALS.selected_block < EASY_GLOBALS.gui_blocks.size())
|
2016-08-08 23:03:11 +03:00
|
|
|
{
|
2016-08-30 22:51:18 +03:00
|
|
|
const auto& guiblock = EASY_GLOBALS.gui_blocks[EASY_GLOBALS.selected_block];
|
2016-08-11 23:43:34 +03:00
|
|
|
if (guiblock.graphics_item == m_index)
|
2016-08-08 23:03:11 +03:00
|
|
|
{
|
|
|
|
const auto& item = m_levels[guiblock.graphics_item_level][guiblock.graphics_item_index];
|
|
|
|
if (item.left() < sceneRight && item.right() > sceneLeft)
|
|
|
|
{
|
2016-08-18 23:26:41 +03:00
|
|
|
auto top = levelY(guiblock.graphics_item_level);
|
2016-09-08 22:42:35 +03:00
|
|
|
auto w = ::std::max(item.width() * currentScale, 1.0);
|
|
|
|
decltype(top) h = (selectedItemsWasPainted && easyBlock(item.block).expanded && w > 20) ? GRAPHICS_ROW_SIZE : item.totalHeight;
|
2016-08-18 23:26:41 +03:00
|
|
|
|
2016-09-07 21:48:50 +03:00
|
|
|
auto dh = top + h - visibleBottom;
|
|
|
|
if (dh < h)
|
2016-08-18 23:26:41 +03:00
|
|
|
{
|
2016-09-07 21:48:50 +03:00
|
|
|
if (dh > 0)
|
|
|
|
h -= dh;
|
|
|
|
|
|
|
|
QPen pen(Qt::SolidLine);
|
2016-09-08 22:42:35 +03:00
|
|
|
pen.setJoinStyle(Qt::MiterJoin);
|
|
|
|
pen.setColor(selectedItemBorderColor(item.color));//Qt::red);
|
|
|
|
pen.setWidth(3);
|
2016-09-07 21:48:50 +03:00
|
|
|
_painter->setPen(pen);
|
2016-08-18 23:26:41 +03:00
|
|
|
|
2016-09-08 22:42:35 +03:00
|
|
|
if (!selectedItemsWasPainted)
|
|
|
|
{
|
|
|
|
brush.setColor(item.color);// SELECTED_ITEM_COLOR);
|
|
|
|
_painter->setBrush(brush);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
_painter->setBrush(Qt::NoBrush);
|
|
|
|
}
|
2016-09-07 21:48:50 +03:00
|
|
|
|
|
|
|
auto x = item.left() * currentScale - dx;
|
|
|
|
rect.setRect(x, top, w, h);
|
|
|
|
_painter->drawRect(rect);
|
|
|
|
|
2016-09-08 22:42:35 +03:00
|
|
|
if (!selectedItemsWasPainted && w > 20)
|
2016-08-18 23:26:41 +03:00
|
|
|
{
|
2016-09-07 21:48:50 +03:00
|
|
|
// Draw text-----------------------------------
|
|
|
|
// calculating text coordinates
|
|
|
|
auto xtext = x;
|
|
|
|
if (item.left() < sceneLeft)
|
|
|
|
{
|
|
|
|
// if item left border is out of screen then attach text to the left border of the screen
|
|
|
|
// to ensure text is always visible for items presenting on the screen.
|
|
|
|
w += (item.left() - sceneLeft) * currentScale;
|
|
|
|
xtext = sceneLeft * currentScale - dx;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (item.right() > sceneRight)
|
|
|
|
{
|
|
|
|
w -= (item.right() - sceneRight) * currentScale;
|
|
|
|
}
|
2016-08-18 23:26:41 +03:00
|
|
|
|
2016-09-07 21:48:50 +03:00
|
|
|
rect.setRect(xtext + 1, top, w - 1, h);
|
2016-08-18 23:26:41 +03:00
|
|
|
|
2016-09-07 21:48:50 +03:00
|
|
|
// text will be painted with inverse color
|
|
|
|
//auto textColor = 0x00ffffff - previousColor;
|
|
|
|
//if (textColor == previousColor) textColor = 0;
|
2016-09-08 22:42:35 +03:00
|
|
|
textColor = ::profiler_gui::textColorForRgb(item.color);// SELECTED_ITEM_COLOR);
|
2016-09-07 21:48:50 +03:00
|
|
|
_painter->setPen(textColor);
|
2016-08-18 23:26:41 +03:00
|
|
|
|
2016-09-08 22:42:35 +03:00
|
|
|
_painter->setFont(SELECTED_ITEM_FONT);
|
|
|
|
|
2016-09-07 21:48:50 +03:00
|
|
|
// drawing text
|
|
|
|
const auto& itemBlock = easyBlock(item.block);
|
|
|
|
auto name = *itemBlock.tree.node->name() != 0 ? itemBlock.tree.node->name() : easyDescriptor(itemBlock.tree.node->id()).name();
|
|
|
|
_painter->drawText(rect, Qt::AlignCenter, ::profiler_gui::toUnicode(name));
|
|
|
|
// END Draw text~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
|
|
|
}
|
2016-08-18 23:26:41 +03:00
|
|
|
}
|
2016-08-08 23:03:11 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-09-04 14:48:35 +03:00
|
|
|
|
|
|
|
if (!m_pRoot->sync.empty())
|
|
|
|
{
|
2016-09-08 22:42:35 +03:00
|
|
|
_painter->setBrush(QColor::fromRgb(::profiler::colors::Coral));// 0xfffe6030));
|
|
|
|
_painter->setPen(QColor::fromRgb(::profiler::colors::Grey800));// 0x00505050));
|
2016-09-04 14:48:35 +03:00
|
|
|
|
2016-09-06 00:23:46 +03:00
|
|
|
qreal prevRight = -1e100, top = y() - 4, h = 3;
|
|
|
|
if (top + h < visibleBottom)
|
2016-09-04 14:48:35 +03:00
|
|
|
{
|
2016-09-06 00:23:46 +03:00
|
|
|
for (auto it = firstSync, end = m_pRoot->sync.end(); it != end; ++it)
|
|
|
|
{
|
|
|
|
const auto& item = easyBlock(*it).tree;
|
|
|
|
auto begin = sceneView->time2position(item.node->begin());
|
2016-09-04 14:48:35 +03:00
|
|
|
|
2016-09-06 00:23:46 +03:00
|
|
|
if (begin > sceneRight)
|
|
|
|
break; // This is first totally invisible item. No need to check other items.
|
2016-09-04 14:48:35 +03:00
|
|
|
|
2016-09-06 00:23:46 +03:00
|
|
|
decltype(begin) width = sceneView->time2position(item.node->end()) - begin;
|
|
|
|
auto r = begin + width;
|
|
|
|
if (r < sceneLeft) // This item is not visible
|
|
|
|
continue;
|
|
|
|
|
|
|
|
begin *= currentScale;
|
|
|
|
begin -= dx;
|
|
|
|
width *= currentScale;
|
|
|
|
r = begin + width;
|
|
|
|
if (r <= prevRight) // This item is not visible
|
|
|
|
continue;
|
2016-09-04 14:48:35 +03:00
|
|
|
|
2016-09-06 00:23:46 +03:00
|
|
|
if (begin < prevRight)
|
|
|
|
{
|
|
|
|
width -= prevRight - begin;
|
|
|
|
begin = prevRight;
|
|
|
|
}
|
2016-09-04 14:48:35 +03:00
|
|
|
|
2016-09-06 00:23:46 +03:00
|
|
|
if (width < 2)
|
|
|
|
width = 2;
|
2016-09-04 14:48:35 +03:00
|
|
|
|
2016-09-06 00:23:46 +03:00
|
|
|
//_painter->drawLine(QLineF(::std::max(begin, prevRight), top, begin + width, top));
|
|
|
|
rect.setRect(begin, top, width, h);
|
|
|
|
_painter->drawRect(rect);
|
|
|
|
prevRight = begin + width;
|
|
|
|
}
|
2016-09-04 14:48:35 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-07-10 01:24:31 +03:00
|
|
|
_painter->restore();
|
|
|
|
}
|
2016-07-31 13:13:48 +03:00
|
|
|
|
2016-08-18 23:26:41 +03:00
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
|
2016-09-08 22:42:35 +03:00
|
|
|
const ::profiler::BlocksTreeRoot* EasyGraphicsItem::root() const
|
|
|
|
{
|
|
|
|
return m_pRoot;
|
|
|
|
}
|
|
|
|
|
|
|
|
const QString& EasyGraphicsItem::threadName() const
|
|
|
|
{
|
|
|
|
return m_threadName;
|
|
|
|
}
|
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
|
2016-08-18 23:26:41 +03:00
|
|
|
QRect EasyGraphicsItem::getRect() const
|
2016-08-09 00:45:45 +03:00
|
|
|
{
|
|
|
|
return view()->mapFromScene(m_boundingRect).boundingRect();
|
|
|
|
}
|
|
|
|
|
2016-07-31 18:48:41 +03:00
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
|
2016-08-18 23:26:41 +03:00
|
|
|
void EasyGraphicsItem::getBlocks(qreal _left, qreal _right, ::profiler_gui::TreeBlocks& _blocks) const
|
2016-07-31 18:48:41 +03:00
|
|
|
{
|
2016-08-08 22:45:57 +03:00
|
|
|
//if (m_bTest)
|
|
|
|
//{
|
|
|
|
// return;
|
|
|
|
//}
|
2016-07-31 18:48:41 +03:00
|
|
|
|
2016-08-01 22:21:59 +03:00
|
|
|
// Search for first visible top-level item
|
2016-08-08 22:17:56 +03:00
|
|
|
auto& level0 = m_levels.front();
|
2016-08-30 22:51:18 +03:00
|
|
|
auto first = ::std::lower_bound(level0.begin(), level0.end(), _left, [](const ::profiler_gui::EasyBlockItem& _item, qreal _value)
|
2016-07-31 18:48:41 +03:00
|
|
|
{
|
|
|
|
return _item.left() < _value;
|
|
|
|
});
|
|
|
|
|
|
|
|
size_t itemIndex = 0;
|
|
|
|
if (first != level0.end())
|
|
|
|
{
|
|
|
|
itemIndex = first - level0.begin();
|
|
|
|
if (itemIndex > 0)
|
|
|
|
itemIndex -= 1;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
itemIndex = level0.size() - 1;
|
|
|
|
}
|
|
|
|
|
2016-08-01 22:21:59 +03:00
|
|
|
// Add all visible top-level items into array of visible blocks
|
2016-07-31 18:48:41 +03:00
|
|
|
for (size_t i = itemIndex, end = level0.size(); i < end; ++i)
|
|
|
|
{
|
|
|
|
const auto& item = level0[i];
|
|
|
|
|
|
|
|
if (item.left() > _right)
|
|
|
|
{
|
2016-08-01 22:21:59 +03:00
|
|
|
// First invisible item. No need to check other items.
|
2016-07-31 18:48:41 +03:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (item.right() < _left)
|
|
|
|
{
|
2016-08-01 22:21:59 +03:00
|
|
|
// This item is not visible yet
|
|
|
|
// This is just to be sure
|
2016-07-31 18:48:41 +03:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2016-08-03 23:00:04 +03:00
|
|
|
_blocks.emplace_back(m_pRoot, item.block);
|
2016-07-31 18:48:41 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
|
2016-08-30 22:51:18 +03:00
|
|
|
const ::profiler_gui::EasyBlockItem* EasyGraphicsItem::intersect(const QPointF& _pos) const
|
2016-08-08 22:17:56 +03:00
|
|
|
{
|
|
|
|
if (m_levels.empty() || m_levels.front().empty())
|
|
|
|
{
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
const auto& level0 = m_levels.front();
|
2016-08-11 23:43:34 +03:00
|
|
|
const auto top = y();
|
2016-08-08 22:17:56 +03:00
|
|
|
|
|
|
|
if (top > _pos.y())
|
|
|
|
{
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
const auto bottom = top + m_levels.size() * GRAPHICS_ROW_SIZE_FULL;
|
|
|
|
if (bottom < _pos.y())
|
|
|
|
{
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
const unsigned int levelIndex = static_cast<unsigned int>(_pos.y() - top) / GRAPHICS_ROW_SIZE_FULL;
|
|
|
|
if (levelIndex >= m_levels.size())
|
|
|
|
{
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
const auto currentScale = view()->scale();
|
|
|
|
unsigned int i = 0;
|
2016-08-11 23:43:34 +03:00
|
|
|
size_t itemIndex = ::std::numeric_limits<size_t>::max();
|
2016-08-08 22:17:56 +03:00
|
|
|
size_t firstItem = 0, lastItem = static_cast<unsigned int>(level0.size());
|
|
|
|
while (i <= levelIndex)
|
|
|
|
{
|
|
|
|
const auto& level = m_levels[i];
|
|
|
|
|
|
|
|
// Search for first visible item
|
2016-08-30 22:51:18 +03:00
|
|
|
auto first = ::std::lower_bound(level.begin() + firstItem, level.begin() + lastItem, _pos.x(), [](const ::profiler_gui::EasyBlockItem& _item, qreal _value)
|
2016-08-08 22:17:56 +03:00
|
|
|
{
|
|
|
|
return _item.left() < _value;
|
|
|
|
});
|
|
|
|
|
|
|
|
if (first != level.end())
|
|
|
|
{
|
|
|
|
itemIndex = first - level.begin();
|
|
|
|
if (itemIndex != 0)
|
|
|
|
--itemIndex;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
itemIndex = level.size() - 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (auto size = level.size(); itemIndex < size; ++itemIndex)
|
|
|
|
{
|
|
|
|
const auto& item = level[itemIndex];
|
2016-08-11 23:43:34 +03:00
|
|
|
static const auto MAX_CHILD_INDEX = ::profiler_gui::numeric_max(item.children_begin);
|
2016-08-08 22:17:56 +03:00
|
|
|
|
|
|
|
if (item.left() > _pos.x())
|
|
|
|
{
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (item.right() < _pos.x())
|
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
const auto w = item.width() * currentScale;
|
2016-08-30 22:51:18 +03:00
|
|
|
if (i == levelIndex || w < 20 || !easyBlock(item.block).expanded)
|
2016-08-08 22:17:56 +03:00
|
|
|
{
|
|
|
|
return &item;
|
|
|
|
}
|
|
|
|
|
2016-08-11 23:43:34 +03:00
|
|
|
if (item.children_begin == MAX_CHILD_INDEX)
|
2016-08-08 22:17:56 +03:00
|
|
|
{
|
|
|
|
if (itemIndex != 0)
|
|
|
|
{
|
|
|
|
auto j = itemIndex;
|
|
|
|
firstItem = 0;
|
|
|
|
do {
|
|
|
|
|
|
|
|
--j;
|
|
|
|
const auto& item2 = level[j];
|
2016-08-11 23:43:34 +03:00
|
|
|
if (item2.children_begin != MAX_CHILD_INDEX)
|
2016-08-08 22:17:56 +03:00
|
|
|
{
|
|
|
|
firstItem = item2.children_begin;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
} while (j != 0);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
firstItem = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
firstItem = item.children_begin;
|
|
|
|
}
|
|
|
|
|
|
|
|
lastItem = m_levels[i + 1].size();
|
|
|
|
for (auto j = itemIndex + 1; j < size; ++j)
|
|
|
|
{
|
|
|
|
const auto& item2 = level[j];
|
2016-08-11 23:43:34 +03:00
|
|
|
if (item2.children_begin != MAX_CHILD_INDEX)
|
2016-08-08 22:17:56 +03:00
|
|
|
{
|
|
|
|
lastItem = item2.children_begin;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
++i;
|
|
|
|
}
|
|
|
|
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
|
2016-08-18 23:26:41 +03:00
|
|
|
void EasyGraphicsItem::setBoundingRect(qreal x, qreal y, qreal w, qreal h)
|
2016-06-30 02:57:57 +03:00
|
|
|
{
|
2016-07-31 18:48:41 +03:00
|
|
|
m_boundingRect.setRect(x, y, w, h);
|
2016-06-30 02:57:57 +03:00
|
|
|
}
|
2016-06-26 20:54:16 +03:00
|
|
|
|
2016-08-18 23:26:41 +03:00
|
|
|
void EasyGraphicsItem::setBoundingRect(const QRectF& _rect)
|
2016-06-30 02:57:57 +03:00
|
|
|
{
|
2016-07-31 18:48:41 +03:00
|
|
|
m_boundingRect = _rect;
|
2016-06-30 02:57:57 +03:00
|
|
|
}
|
2016-06-26 18:46:51 +03:00
|
|
|
|
2016-07-31 18:48:41 +03:00
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
|
2016-08-18 23:26:41 +03:00
|
|
|
::profiler::thread_id_t EasyGraphicsItem::threadId() const
|
2016-08-03 23:00:04 +03:00
|
|
|
{
|
|
|
|
return m_pRoot->thread_id;
|
|
|
|
}
|
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
|
2016-09-04 14:48:35 +03:00
|
|
|
uint8_t EasyGraphicsItem::levels() const
|
2016-07-27 21:50:11 +03:00
|
|
|
{
|
2016-09-04 14:48:35 +03:00
|
|
|
return static_cast<uint8_t>(m_levels.size());
|
2016-07-27 21:50:11 +03:00
|
|
|
}
|
|
|
|
|
2016-09-04 14:48:35 +03:00
|
|
|
float EasyGraphicsItem::levelY(uint8_t _level) const
|
2016-08-08 22:45:57 +03:00
|
|
|
{
|
2016-08-11 23:43:34 +03:00
|
|
|
return y() + static_cast<int>(_level) * static_cast<int>(GRAPHICS_ROW_SIZE_FULL);
|
|
|
|
}
|
|
|
|
|
2016-09-04 14:48:35 +03:00
|
|
|
void EasyGraphicsItem::setLevels(uint8_t _levels)
|
2016-08-11 23:43:34 +03:00
|
|
|
{
|
|
|
|
typedef decltype(m_levelsIndexes) IndexesT;
|
|
|
|
static const auto MAX_CHILD_INDEX = ::profiler_gui::numeric_max<IndexesT::value_type>();
|
|
|
|
|
2016-08-08 22:45:57 +03:00
|
|
|
m_levels.resize(_levels);
|
2016-08-11 23:43:34 +03:00
|
|
|
m_levelsIndexes.resize(_levels, MAX_CHILD_INDEX);
|
2016-08-08 22:45:57 +03:00
|
|
|
}
|
|
|
|
|
2016-09-04 14:48:35 +03:00
|
|
|
void EasyGraphicsItem::reserve(uint8_t _level, unsigned int _items)
|
2016-08-08 22:45:57 +03:00
|
|
|
{
|
|
|
|
m_levels[_level].reserve(_items);
|
|
|
|
}
|
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
|
2016-09-04 14:48:35 +03:00
|
|
|
const EasyGraphicsItem::Children& EasyGraphicsItem::items(uint8_t _level) const
|
2016-08-08 22:45:57 +03:00
|
|
|
{
|
|
|
|
return m_levels[_level];
|
|
|
|
}
|
|
|
|
|
2016-09-04 14:48:35 +03:00
|
|
|
const ::profiler_gui::EasyBlockItem& EasyGraphicsItem::getItem(uint8_t _level, unsigned int _index) const
|
2016-08-08 22:45:57 +03:00
|
|
|
{
|
|
|
|
return m_levels[_level][_index];
|
|
|
|
}
|
|
|
|
|
2016-09-04 14:48:35 +03:00
|
|
|
::profiler_gui::EasyBlockItem& EasyGraphicsItem::getItem(uint8_t _level, unsigned int _index)
|
2016-08-08 22:45:57 +03:00
|
|
|
{
|
|
|
|
return m_levels[_level][_index];
|
|
|
|
}
|
|
|
|
|
2016-09-04 14:48:35 +03:00
|
|
|
unsigned int EasyGraphicsItem::addItem(uint8_t _level)
|
2016-08-08 22:45:57 +03:00
|
|
|
{
|
|
|
|
m_levels[_level].emplace_back();
|
|
|
|
return static_cast<unsigned int>(m_levels[_level].size() - 1);
|
|
|
|
}
|
|
|
|
|
2016-07-31 18:48:41 +03:00
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
|
2016-08-18 23:26:41 +03:00
|
|
|
EasyChronometerItem::EasyChronometerItem(bool _main)
|
|
|
|
: QGraphicsItem()
|
|
|
|
, m_color(CHRONOMETER_COLOR)
|
|
|
|
, m_left(0)
|
|
|
|
, m_right(0)
|
|
|
|
, m_bMain(_main)
|
|
|
|
, m_bReverse(false)
|
|
|
|
, m_bHoverIndicator(false)
|
2016-07-31 18:48:41 +03:00
|
|
|
{
|
2016-08-10 22:08:27 +03:00
|
|
|
//setZValue(_main ? 10 : 9);
|
2016-08-06 14:50:31 +03:00
|
|
|
m_indicator.reserve(3);
|
2016-07-31 18:48:41 +03:00
|
|
|
}
|
|
|
|
|
2016-08-18 23:26:41 +03:00
|
|
|
EasyChronometerItem::~EasyChronometerItem()
|
2016-07-31 18:48:41 +03:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2016-08-18 23:26:41 +03:00
|
|
|
QRectF EasyChronometerItem::boundingRect() const
|
2016-07-31 18:48:41 +03:00
|
|
|
{
|
|
|
|
return m_boundingRect;
|
|
|
|
}
|
|
|
|
|
2016-09-09 00:07:27 +03:00
|
|
|
void EasyChronometerItem::paint(QPainter* _painter, const QStyleOptionGraphicsItem*, QWidget*)
|
2016-07-31 18:48:41 +03:00
|
|
|
{
|
2016-09-11 16:53:34 +03:00
|
|
|
auto const sceneView = view();
|
2016-07-31 18:48:41 +03:00
|
|
|
const auto currentScale = sceneView->scale();
|
|
|
|
const auto offset = sceneView->offset();
|
|
|
|
const auto visibleSceneRect = sceneView->visibleSceneRect();
|
2016-08-02 22:24:22 +03:00
|
|
|
auto sceneLeft = offset, sceneRight = offset + visibleSceneRect.width() / currentScale;
|
2016-07-31 18:48:41 +03:00
|
|
|
|
2016-08-08 22:17:56 +03:00
|
|
|
if (m_bMain)
|
|
|
|
m_indicator.clear();
|
|
|
|
|
2016-07-31 18:48:41 +03:00
|
|
|
if (m_left > sceneRight || m_right < sceneLeft)
|
|
|
|
{
|
2016-08-01 22:21:59 +03:00
|
|
|
// This item is out of screen
|
2016-08-06 14:50:31 +03:00
|
|
|
|
|
|
|
if (m_bMain)
|
|
|
|
{
|
2016-08-18 23:26:41 +03:00
|
|
|
const int size = m_bHoverIndicator ? 12 : 10;
|
2016-08-06 14:50:31 +03:00
|
|
|
auto vcenter = visibleSceneRect.top() + visibleSceneRect.height() * 0.5;
|
2016-08-08 22:17:56 +03:00
|
|
|
auto color = QColor::fromRgb(m_color.rgb());
|
|
|
|
auto pen = _painter->pen();
|
|
|
|
pen.setColor(color);
|
2016-08-06 14:50:31 +03:00
|
|
|
|
|
|
|
m_indicator.clear();
|
|
|
|
if (m_left > sceneRight)
|
|
|
|
{
|
2016-08-07 20:16:15 +03:00
|
|
|
sceneRight = (sceneRight - offset) * currentScale;
|
2016-08-08 22:17:56 +03:00
|
|
|
m_indicator.push_back(QPointF(sceneRight - size, vcenter - size));
|
2016-08-06 14:50:31 +03:00
|
|
|
m_indicator.push_back(QPointF(sceneRight, vcenter));
|
2016-08-08 22:17:56 +03:00
|
|
|
m_indicator.push_back(QPointF(sceneRight - size, vcenter + size));
|
2016-08-06 14:50:31 +03:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
sceneLeft = (sceneLeft - offset) * currentScale;
|
2016-08-08 22:17:56 +03:00
|
|
|
m_indicator.push_back(QPointF(sceneLeft + size, vcenter - size));
|
2016-08-06 14:50:31 +03:00
|
|
|
m_indicator.push_back(QPointF(sceneLeft, vcenter));
|
2016-08-08 22:17:56 +03:00
|
|
|
m_indicator.push_back(QPointF(sceneLeft + size, vcenter + size));
|
2016-08-06 14:50:31 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
_painter->save();
|
|
|
|
_painter->setTransform(QTransform::fromTranslate(-x(), -y()), true);
|
2016-08-18 23:26:41 +03:00
|
|
|
_painter->setBrush(m_bHoverIndicator ? QColor::fromRgb(0xffff0000) : color);
|
2016-08-08 22:17:56 +03:00
|
|
|
_painter->setPen(pen);
|
2016-08-06 14:50:31 +03:00
|
|
|
_painter->drawPolygon(m_indicator);
|
|
|
|
_painter->restore();
|
|
|
|
}
|
|
|
|
|
2016-07-31 18:48:41 +03:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2016-08-10 22:08:27 +03:00
|
|
|
auto selectedInterval = width();
|
2016-08-01 22:21:59 +03:00
|
|
|
QRectF rect((m_left - offset) * currentScale, visibleSceneRect.top(), ::std::max(selectedInterval * currentScale, 1.0), visibleSceneRect.height());
|
2016-08-10 22:08:27 +03:00
|
|
|
selectedInterval = units2microseconds(selectedInterval);
|
2016-07-31 18:48:41 +03:00
|
|
|
|
2016-08-10 22:08:27 +03:00
|
|
|
const QString text = ::profiler_gui::timeStringReal(selectedInterval); // Displayed text
|
2016-09-11 16:53:34 +03:00
|
|
|
const auto textRect = QFontMetricsF(CHRONOMETER_FONT, sceneView).boundingRect(text); // Calculate displayed text boundingRect
|
2016-08-10 22:08:27 +03:00
|
|
|
const auto rgb = m_color.rgb() & 0x00ffffff;
|
|
|
|
|
|
|
|
|
2016-07-31 18:48:41 +03:00
|
|
|
|
2016-08-01 22:21:59 +03:00
|
|
|
// Paint!--------------------------
|
2016-07-31 18:48:41 +03:00
|
|
|
_painter->save();
|
2016-06-30 03:45:11 +03:00
|
|
|
|
2016-08-01 22:21:59 +03:00
|
|
|
// instead of scrollbar we're using manual offset
|
|
|
|
_painter->setTransform(QTransform::fromTranslate(-x(), -y()), true);
|
|
|
|
|
2016-08-28 23:40:51 +03:00
|
|
|
if (m_left < sceneLeft)
|
|
|
|
rect.setLeft(0);
|
|
|
|
|
|
|
|
if (m_right > sceneRight)
|
|
|
|
rect.setWidth((sceneRight - offset) * currentScale - rect.left());
|
|
|
|
|
2016-08-01 22:21:59 +03:00
|
|
|
// draw transparent rectangle
|
2016-08-21 14:26:04 +03:00
|
|
|
auto vcenter = rect.top() + rect.height() * 0.5;
|
|
|
|
QLinearGradient g(rect.left(), vcenter, rect.right(), vcenter);
|
|
|
|
g.setColorAt(0, m_color);
|
2016-09-09 00:07:27 +03:00
|
|
|
g.setColorAt(0.2, QColor::fromRgba(0x14000000 | rgb));
|
|
|
|
g.setColorAt(0.8, QColor::fromRgba(0x14000000 | rgb));
|
2016-08-21 14:26:04 +03:00
|
|
|
g.setColorAt(1, m_color);
|
|
|
|
_painter->setBrush(g);
|
2016-07-31 18:48:41 +03:00
|
|
|
_painter->setPen(Qt::NoPen);
|
|
|
|
_painter->drawRect(rect);
|
|
|
|
|
2016-08-10 22:08:27 +03:00
|
|
|
// draw left and right borders
|
|
|
|
_painter->setBrush(Qt::NoBrush);
|
2016-08-21 14:26:04 +03:00
|
|
|
_painter->setPen(QColor::fromRgba(0xd0000000 | rgb));
|
2016-09-09 00:07:27 +03:00
|
|
|
|
2016-08-10 22:08:27 +03:00
|
|
|
if (m_left > sceneLeft)
|
|
|
|
_painter->drawLine(QPointF(rect.left(), rect.top()), QPointF(rect.left(), rect.bottom()));
|
|
|
|
if (m_right < sceneRight)
|
|
|
|
_painter->drawLine(QPointF(rect.right(), rect.top()), QPointF(rect.right(), rect.bottom()));
|
|
|
|
|
2016-08-01 22:21:59 +03:00
|
|
|
// draw text
|
2016-08-07 19:38:31 +03:00
|
|
|
_painter->setCompositionMode(QPainter::CompositionMode_Difference); // This lets the text to be visible on every background
|
2016-09-07 21:48:50 +03:00
|
|
|
_painter->setRenderHint(QPainter::TextAntialiasing);
|
|
|
|
_painter->setPen(0x00ffffff - rgb);
|
2016-08-18 23:26:41 +03:00
|
|
|
_painter->setFont(CHRONOMETER_FONT);
|
2016-06-30 02:57:57 +03:00
|
|
|
|
2016-08-24 01:00:24 +03:00
|
|
|
int textFlags = 0;
|
2016-08-30 22:51:18 +03:00
|
|
|
switch (EASY_GLOBALS.chrono_text_position)
|
2016-08-06 14:50:31 +03:00
|
|
|
{
|
2016-08-24 01:00:24 +03:00
|
|
|
case ::profiler_gui::ChronoTextPosition_Top:
|
|
|
|
textFlags = Qt::AlignTop | Qt::AlignHCenter;
|
|
|
|
if (!m_bMain) rect.setTop(rect.top() + textRect.height() * 0.75);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case ::profiler_gui::ChronoTextPosition_Center:
|
|
|
|
textFlags = Qt::AlignCenter;
|
|
|
|
if (!m_bMain) rect.setTop(rect.top() + textRect.height() * 1.5);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case ::profiler_gui::ChronoTextPosition_Bottom:
|
|
|
|
textFlags = Qt::AlignBottom | Qt::AlignHCenter;
|
|
|
|
if (!m_bMain) rect.setHeight(rect.height() - textRect.height() * 0.75);
|
|
|
|
break;
|
2016-08-06 14:50:31 +03:00
|
|
|
}
|
|
|
|
|
2016-09-11 16:53:34 +03:00
|
|
|
const auto textRect_width = textRect.width() * FONT_METRICS_FACTOR;
|
|
|
|
if (textRect_width < rect.width())
|
2016-08-02 22:24:22 +03:00
|
|
|
{
|
|
|
|
// Text will be drawed inside rectangle
|
2016-08-24 01:00:24 +03:00
|
|
|
_painter->drawText(rect, textFlags, text);
|
2016-08-02 22:24:22 +03:00
|
|
|
_painter->restore();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2016-09-11 16:53:34 +03:00
|
|
|
const auto w = textRect_width / currentScale;
|
2016-08-30 20:47:49 +03:00
|
|
|
if (m_right + w < sceneRight)
|
2016-08-02 22:24:22 +03:00
|
|
|
{
|
|
|
|
// Text will be drawed to the right of rectangle
|
2016-08-24 01:00:24 +03:00
|
|
|
rect.translate(rect.width(), 0);
|
|
|
|
textFlags &= ~Qt::AlignHCenter;
|
|
|
|
textFlags |= Qt::AlignLeft;
|
2016-08-02 22:24:22 +03:00
|
|
|
}
|
2016-08-30 20:47:49 +03:00
|
|
|
else if (m_left - w > sceneLeft)
|
2016-08-02 22:24:22 +03:00
|
|
|
{
|
|
|
|
// Text will be drawed to the left of rectangle
|
2016-08-24 01:00:24 +03:00
|
|
|
rect.translate(-rect.width(), 0);
|
|
|
|
textFlags &= ~Qt::AlignHCenter;
|
|
|
|
textFlags |= Qt::AlignRight;
|
2016-07-31 18:48:41 +03:00
|
|
|
}
|
2016-08-24 01:00:24 +03:00
|
|
|
//else // Text will be drawed inside rectangle
|
|
|
|
|
|
|
|
_painter->drawText(rect, textFlags | Qt::TextDontClip, text);
|
2016-07-31 18:48:41 +03:00
|
|
|
|
|
|
|
_painter->restore();
|
2016-08-01 22:21:59 +03:00
|
|
|
// END Paint!~~~~~~~~~~~~~~~~~~~~~~
|
2016-07-31 18:48:41 +03:00
|
|
|
}
|
|
|
|
|
2016-08-18 23:26:41 +03:00
|
|
|
bool EasyChronometerItem::contains(const QPointF& _pos) const
|
2016-08-08 22:17:56 +03:00
|
|
|
{
|
|
|
|
const auto sceneView = view();
|
|
|
|
const auto clickX = (_pos.x() - sceneView->offset()) * sceneView->scale() - x();
|
|
|
|
if (!m_indicator.empty() && m_indicator.containsPoint(QPointF(clickX, _pos.y()), Qt::OddEvenFill))
|
|
|
|
return true;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2016-08-18 23:26:41 +03:00
|
|
|
void EasyChronometerItem::setColor(const QColor& _color)
|
2016-08-06 14:50:31 +03:00
|
|
|
{
|
|
|
|
m_color = _color;
|
|
|
|
}
|
|
|
|
|
2016-08-18 23:26:41 +03:00
|
|
|
void EasyChronometerItem::setBoundingRect(qreal x, qreal y, qreal w, qreal h)
|
2016-07-31 18:48:41 +03:00
|
|
|
{
|
|
|
|
m_boundingRect.setRect(x, y, w, h);
|
|
|
|
}
|
|
|
|
|
2016-08-18 23:26:41 +03:00
|
|
|
void EasyChronometerItem::setBoundingRect(const QRectF& _rect)
|
2016-07-31 18:48:41 +03:00
|
|
|
{
|
|
|
|
m_boundingRect = _rect;
|
|
|
|
}
|
|
|
|
|
2016-08-18 23:26:41 +03:00
|
|
|
void EasyChronometerItem::setLeftRight(qreal _left, qreal _right)
|
2016-06-26 18:46:51 +03:00
|
|
|
{
|
2016-07-31 18:48:41 +03:00
|
|
|
if (_left < _right)
|
|
|
|
{
|
|
|
|
m_left = _left;
|
|
|
|
m_right = _right;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
m_left = _right;
|
|
|
|
m_right = _left;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-08-18 23:26:41 +03:00
|
|
|
void EasyChronometerItem::setReverse(bool _reverse)
|
2016-08-06 14:50:31 +03:00
|
|
|
{
|
|
|
|
m_bReverse = _reverse;
|
|
|
|
}
|
|
|
|
|
2016-08-18 23:26:41 +03:00
|
|
|
void EasyChronometerItem::setHover(bool _hover)
|
2016-08-08 22:17:56 +03:00
|
|
|
{
|
2016-08-18 23:26:41 +03:00
|
|
|
m_bHoverIndicator = _hover;
|
2016-08-08 22:17:56 +03:00
|
|
|
}
|
|
|
|
|
2016-08-18 23:26:41 +03:00
|
|
|
const EasyGraphicsView* EasyChronometerItem::view() const
|
2016-07-31 18:48:41 +03:00
|
|
|
{
|
2016-08-18 23:26:41 +03:00
|
|
|
return static_cast<const EasyGraphicsView*>(scene()->parent());
|
2016-07-31 18:48:41 +03:00
|
|
|
}
|
|
|
|
|
2016-09-11 16:53:34 +03:00
|
|
|
EasyGraphicsView* EasyChronometerItem::view()
|
|
|
|
{
|
|
|
|
return static_cast<EasyGraphicsView*>(scene()->parent());
|
|
|
|
}
|
|
|
|
|
2016-07-31 18:48:41 +03:00
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
|
2016-09-09 00:07:27 +03:00
|
|
|
void EasyBackgroundItem::paint(QPainter* _painter, const QStyleOptionGraphicsItem*, QWidget*)
|
2016-07-31 18:48:41 +03:00
|
|
|
{
|
2016-09-11 16:53:34 +03:00
|
|
|
auto const sceneView = static_cast<EasyGraphicsView*>(scene()->parent());
|
2016-08-10 22:08:27 +03:00
|
|
|
const auto visibleSceneRect = sceneView->visibleSceneRect();
|
|
|
|
const auto currentScale = sceneView->scale();
|
|
|
|
const auto offset = sceneView->offset();
|
|
|
|
const auto left = offset * currentScale;
|
2016-08-21 14:26:04 +03:00
|
|
|
const auto h = visibleSceneRect.height();
|
2016-08-21 18:04:38 +03:00
|
|
|
const auto visibleBottom = h - 1;
|
2016-08-10 22:08:27 +03:00
|
|
|
|
|
|
|
QRectF rect;
|
|
|
|
|
|
|
|
_painter->save();
|
|
|
|
_painter->setTransform(QTransform::fromTranslate(-x(), -y()));
|
|
|
|
|
|
|
|
const auto& items = sceneView->getItems();
|
|
|
|
if (!items.empty())
|
|
|
|
{
|
2016-09-09 00:07:27 +03:00
|
|
|
static const uint16_t OVERLAP = THREADS_ROW_SPACING >> 1;
|
2016-08-10 22:08:27 +03:00
|
|
|
static const QBrush brushes[2] = {QColor::fromRgb(BACKGROUND_1), QColor::fromRgb(BACKGROUND_2)};
|
|
|
|
int i = -1;
|
|
|
|
|
|
|
|
// Draw background
|
|
|
|
_painter->setPen(Qt::NoPen);
|
|
|
|
for (auto item : items)
|
|
|
|
{
|
|
|
|
++i;
|
|
|
|
|
|
|
|
auto br = item->boundingRect();
|
|
|
|
auto top = item->y() + br.top() - visibleSceneRect.top();
|
|
|
|
auto bottom = top + br.height();
|
|
|
|
|
2016-08-21 14:26:04 +03:00
|
|
|
if (top > h || bottom < 0)
|
2016-08-10 22:08:27 +03:00
|
|
|
continue;
|
|
|
|
|
2016-08-30 22:51:18 +03:00
|
|
|
if (item->threadId() == EASY_GLOBALS.selected_thread)
|
2016-08-10 22:08:27 +03:00
|
|
|
_painter->setBrush(QBrush(QColor::fromRgb(::profiler_gui::SELECTED_THREAD_BACKGROUND)));
|
|
|
|
else
|
|
|
|
_painter->setBrush(brushes[i & 1]);
|
|
|
|
|
2016-08-21 14:26:04 +03:00
|
|
|
rect.setRect(0, top - OVERLAP, visibleSceneRect.width(), br.height() + THREADS_ROW_SPACING);
|
2016-08-21 18:04:38 +03:00
|
|
|
const auto dh = rect.bottom() - visibleBottom;
|
|
|
|
if (dh > 0)
|
|
|
|
rect.setHeight(rect.height() - dh);
|
|
|
|
|
2016-08-10 22:08:27 +03:00
|
|
|
_painter->drawRect(rect);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Draw timeline scale marks ----------------
|
|
|
|
_painter->setBrush(QColor::fromRgba(TIMELINE_BACKGROUND));
|
2016-07-31 18:48:41 +03:00
|
|
|
|
2016-08-21 14:26:04 +03:00
|
|
|
const auto sceneStep = sceneView->timelineStep();
|
|
|
|
const auto factor = ::profiler_gui::timeFactor(sceneStep);
|
|
|
|
const auto step = sceneStep * currentScale;
|
|
|
|
auto first = static_cast<quint64>(offset / sceneStep);
|
|
|
|
const int odd = first & 1;
|
|
|
|
const auto nsteps = (1 + odd) * 2 + static_cast<int>(visibleSceneRect.width() / step);
|
|
|
|
first -= odd;
|
|
|
|
|
|
|
|
QPen pen(Qt::gray);
|
|
|
|
pen.setWidth(2);
|
|
|
|
_painter->setPen(pen);
|
|
|
|
_painter->drawLine(QPointF(0, h), QPointF(visibleSceneRect.width(), h));
|
|
|
|
_painter->setPen(Qt::gray);
|
|
|
|
|
|
|
|
QLineF marks[20];
|
|
|
|
qreal first_x = first * sceneStep;
|
2016-09-11 16:53:34 +03:00
|
|
|
const auto textWidth = QFontMetricsF(_painter->font(), sceneView).width(QString::number(static_cast<quint64>(0.5 + first_x * factor))) * FONT_METRICS_FACTOR + 10;
|
2016-08-21 14:26:04 +03:00
|
|
|
const int n = 1 + static_cast<int>(textWidth / step);
|
|
|
|
int next = first % n;
|
|
|
|
if (next)
|
|
|
|
next = n - next;
|
|
|
|
|
|
|
|
first_x *= currentScale;
|
|
|
|
for (int i = 0; i < nsteps; ++i, --next)
|
2016-06-26 20:54:16 +03:00
|
|
|
{
|
2016-08-21 14:26:04 +03:00
|
|
|
auto current = first_x - left + step * i;
|
|
|
|
|
|
|
|
if ((i & 1) == 0)
|
|
|
|
{
|
|
|
|
rect.setRect(current, 0, step, h);
|
|
|
|
_painter->drawRect(rect);
|
2016-08-10 22:08:27 +03:00
|
|
|
|
2016-08-21 14:26:04 +03:00
|
|
|
for (int j = 0; j < 20; ++j)
|
|
|
|
{
|
|
|
|
auto xmark = current + j * step * 0.1;
|
2016-08-21 22:44:03 +03:00
|
|
|
marks[j].setLine(xmark, h, xmark, h + ((j % 5) ? 4 : 8));
|
2016-08-21 14:26:04 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
_painter->drawLines(marks, 20);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (next <= 0)
|
|
|
|
{
|
|
|
|
next = n;
|
|
|
|
_painter->setPen(Qt::black);
|
|
|
|
_painter->drawText(QPointF(current + 1, h + 17), QString::number(static_cast<quint64>(0.5 + (current + left) * factor / currentScale)));
|
|
|
|
_painter->setPen(Qt::gray);
|
|
|
|
}
|
|
|
|
|
|
|
|
// TEST
|
|
|
|
// this is for testing (order of lines will be painted):
|
|
|
|
//_painter->setPen(Qt::black);
|
|
|
|
//_painter->drawText(QPointF(current + step * 0.4, h - 20), QString::number(i));
|
|
|
|
//_painter->setPen(Qt::gray);
|
|
|
|
// TEST
|
2016-06-26 20:54:16 +03:00
|
|
|
}
|
2016-08-10 22:08:27 +03:00
|
|
|
// END Draw timeline scale marks ~~~~~~~~~~~~
|
2016-07-31 18:48:41 +03:00
|
|
|
|
2016-08-10 22:08:27 +03:00
|
|
|
_painter->restore();
|
2016-06-26 18:46:51 +03:00
|
|
|
}
|
|
|
|
|
2016-08-10 22:08:27 +03:00
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
|
2016-09-09 00:07:27 +03:00
|
|
|
void EasyTimelineIndicatorItem::paint(QPainter* _painter, const QStyleOptionGraphicsItem*, QWidget*)
|
2016-08-10 22:08:27 +03:00
|
|
|
{
|
2016-08-18 23:26:41 +03:00
|
|
|
const auto sceneView = static_cast<const EasyGraphicsView*>(scene()->parent());
|
2016-08-10 22:08:27 +03:00
|
|
|
const auto visibleSceneRect = sceneView->visibleSceneRect();
|
|
|
|
const auto step = sceneView->timelineStep() * sceneView->scale();
|
|
|
|
const QString text = ::profiler_gui::timeStringInt(units2microseconds(sceneView->timelineStep())); // Displayed text
|
|
|
|
|
|
|
|
// Draw scale indicator
|
|
|
|
_painter->save();
|
|
|
|
_painter->setTransform(QTransform::fromTranslate(-x(), -y()));
|
2016-09-07 21:48:50 +03:00
|
|
|
//_painter->setCompositionMode(QPainter::CompositionMode_Difference);
|
2016-08-21 22:44:03 +03:00
|
|
|
_painter->setBrush(Qt::NoBrush);
|
2016-08-10 22:08:27 +03:00
|
|
|
|
2016-09-07 21:48:50 +03:00
|
|
|
QPen pen(Qt::black);
|
2016-08-21 22:44:03 +03:00
|
|
|
pen.setWidth(2);
|
|
|
|
pen.setJoinStyle(Qt::MiterJoin);
|
|
|
|
_painter->setPen(pen);
|
2016-08-10 22:08:27 +03:00
|
|
|
|
2016-08-21 22:44:03 +03:00
|
|
|
QRectF rect(visibleSceneRect.width() - 10 - step, visibleSceneRect.height() - 20, step, 10);
|
|
|
|
const auto rect_right = rect.right();
|
|
|
|
const QPointF points[] = {{rect.left(), rect.bottom()}, {rect.left(), rect.top()}, {rect_right, rect.top()}, {rect_right, rect.top() + 5}};
|
|
|
|
_painter->drawPolyline(points, sizeof(points) / sizeof(QPointF));
|
|
|
|
|
|
|
|
rect.translate(0, 3);
|
2016-08-10 22:08:27 +03:00
|
|
|
_painter->drawText(rect, Qt::AlignRight | Qt::TextDontClip, text);
|
|
|
|
|
|
|
|
_painter->restore();
|
|
|
|
}
|
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
|
2016-08-18 23:26:41 +03:00
|
|
|
EasyGraphicsView::EasyGraphicsView(QWidget* _parent)
|
2016-09-11 16:53:34 +03:00
|
|
|
: Parent(_parent)
|
2016-08-11 23:43:34 +03:00
|
|
|
, m_beginTime(::std::numeric_limits<decltype(m_beginTime)>::max())
|
2016-07-31 18:48:41 +03:00
|
|
|
, m_scale(1)
|
|
|
|
, m_offset(0)
|
2016-08-21 22:44:03 +03:00
|
|
|
, m_timelineStep(0)
|
2016-07-31 18:48:41 +03:00
|
|
|
, m_mouseButtons(Qt::NoButton)
|
|
|
|
, m_pScrollbar(nullptr)
|
|
|
|
, m_chronometerItem(nullptr)
|
2016-08-06 14:50:31 +03:00
|
|
|
, m_chronometerItemAux(nullptr)
|
|
|
|
, m_flickerSpeedX(0)
|
|
|
|
, m_flickerSpeedY(0)
|
|
|
|
, m_bDoubleClick(false)
|
2016-07-31 18:48:41 +03:00
|
|
|
, m_bUpdatingRect(false)
|
|
|
|
, m_bEmpty(true)
|
2016-06-26 18:46:51 +03:00
|
|
|
{
|
2016-07-31 18:48:41 +03:00
|
|
|
initMode();
|
|
|
|
setScene(new QGraphicsScene(this));
|
|
|
|
updateVisibleSceneRect();
|
2016-06-26 18:46:51 +03:00
|
|
|
}
|
|
|
|
|
2016-08-18 23:26:41 +03:00
|
|
|
EasyGraphicsView::~EasyGraphicsView()
|
2016-06-26 18:46:51 +03:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2016-07-31 13:13:48 +03:00
|
|
|
//////////////////////////////////////////////////////////////////////////
|
2016-07-10 01:24:31 +03:00
|
|
|
|
2016-08-18 23:26:41 +03:00
|
|
|
EasyChronometerItem* EasyGraphicsView::createChronometer(bool _main)
|
2016-08-06 14:50:31 +03:00
|
|
|
{
|
2016-08-18 23:26:41 +03:00
|
|
|
auto chronoItem = new EasyChronometerItem(_main);
|
2016-08-06 14:50:31 +03:00
|
|
|
chronoItem->setColor(_main ? CHRONOMETER_COLOR : CHRONOMETER_COLOR2);
|
|
|
|
chronoItem->setBoundingRect(scene()->sceneRect());
|
|
|
|
chronoItem->hide();
|
|
|
|
scene()->addItem(chronoItem);
|
|
|
|
|
|
|
|
return chronoItem;
|
|
|
|
}
|
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
|
2016-08-18 23:26:41 +03:00
|
|
|
void EasyGraphicsView::clearSilent()
|
2016-06-26 20:54:16 +03:00
|
|
|
{
|
2016-07-31 18:48:41 +03:00
|
|
|
const QSignalBlocker blocker(this), sceneBlocker(scene()); // block all scene signals (otherwise clear() would be extremely slow!)
|
2016-08-01 22:21:59 +03:00
|
|
|
|
|
|
|
// Stop flicking
|
2016-07-31 18:48:41 +03:00
|
|
|
m_flickerTimer.stop();
|
2016-08-06 14:50:31 +03:00
|
|
|
m_flickerSpeedX = 0;
|
|
|
|
m_flickerSpeedY = 0;
|
2016-08-01 22:21:59 +03:00
|
|
|
|
|
|
|
// Clear all items
|
2016-07-31 18:48:41 +03:00
|
|
|
scene()->clear();
|
|
|
|
m_items.clear();
|
|
|
|
m_selectedBlocks.clear();
|
2016-08-01 22:21:59 +03:00
|
|
|
|
2016-08-11 23:43:34 +03:00
|
|
|
m_beginTime = ::std::numeric_limits<decltype(m_beginTime)>::max(); // reset begin time
|
2016-07-31 18:48:41 +03:00
|
|
|
m_scale = 1; // scale back to initial 100% scale
|
2016-08-10 22:08:27 +03:00
|
|
|
m_timelineStep = 1;
|
2016-08-01 22:21:59 +03:00
|
|
|
m_offset = 0; // scroll back to the beginning of the scene
|
|
|
|
|
|
|
|
// Reset necessary flags
|
2016-08-18 23:26:41 +03:00
|
|
|
//m_bTest = false;
|
2016-07-31 18:48:41 +03:00
|
|
|
m_bEmpty = true;
|
2016-08-01 22:21:59 +03:00
|
|
|
|
|
|
|
// notify ProfTreeWidget that selection was reset
|
|
|
|
emit intervalChanged(m_selectedBlocks, m_beginTime, 0, 0, false);
|
2016-06-26 20:54:16 +03:00
|
|
|
}
|
|
|
|
|
2016-08-18 23:26:41 +03:00
|
|
|
void EasyGraphicsView::setTree(const ::profiler::thread_blocks_tree_t& _blocksTree)
|
2016-06-26 18:46:51 +03:00
|
|
|
{
|
2016-07-31 18:48:41 +03:00
|
|
|
// clear scene
|
2016-08-08 22:45:57 +03:00
|
|
|
clearSilent();
|
|
|
|
|
|
|
|
if (_blocksTree.empty())
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2016-08-18 23:26:41 +03:00
|
|
|
auto bgItem = new EasyBackgroundItem();
|
2016-08-10 22:08:27 +03:00
|
|
|
scene()->addItem(bgItem);
|
|
|
|
|
2016-07-31 18:48:41 +03:00
|
|
|
// set new blocks tree
|
2016-06-30 02:57:57 +03:00
|
|
|
// calculate scene size and fill it with items
|
2016-07-31 18:48:41 +03:00
|
|
|
|
|
|
|
// Calculating start and end time
|
2016-06-27 23:22:12 +03:00
|
|
|
::profiler::timestamp_t finish = 0;
|
2016-09-08 22:42:35 +03:00
|
|
|
::profiler::thread_id_t longestTree = 0, mainTree = 0;
|
2016-06-26 18:46:51 +03:00
|
|
|
for (const auto& threadTree : _blocksTree)
|
|
|
|
{
|
2016-09-08 22:42:35 +03:00
|
|
|
const auto& t = threadTree.second;
|
2016-09-04 19:35:58 +03:00
|
|
|
|
2016-09-09 00:07:27 +03:00
|
|
|
auto timestart = blocksTree(t.children.front()).node->begin();
|
2016-09-08 22:42:35 +03:00
|
|
|
if (!t.sync.empty())
|
|
|
|
timestart = ::std::min(timestart, blocksTree(t.sync.front()).node->begin());
|
2016-09-04 19:35:58 +03:00
|
|
|
|
2016-09-08 22:42:35 +03:00
|
|
|
const auto timefinish = blocksTree(t.children.back()).node->end();
|
2016-06-27 23:22:12 +03:00
|
|
|
|
2016-07-31 18:48:41 +03:00
|
|
|
if (m_beginTime > timestart)
|
|
|
|
{
|
|
|
|
m_beginTime = timestart;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (finish < timefinish)
|
|
|
|
{
|
|
|
|
finish = timefinish;
|
2016-08-30 22:51:18 +03:00
|
|
|
longestTree = threadTree.first;
|
2016-07-31 18:48:41 +03:00
|
|
|
}
|
2016-09-08 22:42:35 +03:00
|
|
|
|
|
|
|
if (mainTree == 0 && !strcmp(t.thread_name, "Main"))
|
|
|
|
{
|
|
|
|
mainTree = threadTree.first;
|
|
|
|
}
|
2016-07-31 18:48:41 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
// Filling scene with items
|
|
|
|
m_items.reserve(_blocksTree.size());
|
2016-08-21 14:26:04 +03:00
|
|
|
qreal y = TIMELINE_ROW_SIZE;
|
2016-09-08 22:42:35 +03:00
|
|
|
const EasyGraphicsItem *longestItem = nullptr, *mainThreadItem = nullptr;
|
2016-07-31 18:48:41 +03:00
|
|
|
for (const auto& threadTree : _blocksTree)
|
|
|
|
{
|
2016-08-23 22:42:20 +03:00
|
|
|
if (m_items.size() == 0xff)
|
|
|
|
{
|
|
|
|
qWarning() << "Warning: Maximum threads number (255 threads) exceeded! See EasyGraphicsView::setTree() : " << __LINE__ << " in file " << __FILE__;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2016-09-08 22:42:35 +03:00
|
|
|
const auto& t = threadTree.second;
|
|
|
|
|
2016-06-30 02:57:57 +03:00
|
|
|
// fill scene with new items
|
2016-09-08 22:42:35 +03:00
|
|
|
const auto& tree = t.children;
|
2016-08-30 22:51:18 +03:00
|
|
|
qreal h = 0, x = time2position(blocksTree(tree.front()).node->begin());
|
2016-09-08 22:42:35 +03:00
|
|
|
auto item = new EasyGraphicsItem(static_cast<uint8_t>(m_items.size()), t);
|
|
|
|
item->setLevels(t.depth);
|
2016-07-31 13:13:48 +03:00
|
|
|
item->setPos(0, y);
|
2016-07-27 21:50:11 +03:00
|
|
|
|
2016-08-30 22:51:18 +03:00
|
|
|
const auto children_duration = setTree(item, tree, h, y, 0);
|
2016-07-27 21:50:11 +03:00
|
|
|
|
2016-07-31 13:13:48 +03:00
|
|
|
item->setBoundingRect(0, 0, children_duration + x, h);
|
2016-07-31 18:48:41 +03:00
|
|
|
m_items.push_back(item);
|
|
|
|
scene()->addItem(item);
|
2016-07-31 13:13:48 +03:00
|
|
|
|
2016-08-11 23:43:34 +03:00
|
|
|
y += h + THREADS_ROW_SPACING;
|
2016-08-03 00:06:36 +03:00
|
|
|
|
2016-08-30 22:51:18 +03:00
|
|
|
if (longestTree == threadTree.first)
|
2016-08-03 00:06:36 +03:00
|
|
|
longestItem = item;
|
2016-09-07 21:48:50 +03:00
|
|
|
|
2016-09-08 22:42:35 +03:00
|
|
|
if (mainTree == threadTree.first)
|
2016-09-07 21:48:50 +03:00
|
|
|
mainThreadItem = item;
|
2016-06-26 18:46:51 +03:00
|
|
|
}
|
|
|
|
|
2016-07-31 18:48:41 +03:00
|
|
|
// Calculating scene rect
|
2016-08-07 19:38:31 +03:00
|
|
|
const qreal endX = time2position(finish) + 1500.0;
|
2016-08-21 14:26:04 +03:00
|
|
|
scene()->setSceneRect(0, 0, endX, y + TIMELINE_ROW_SIZE);
|
2016-08-10 22:08:27 +03:00
|
|
|
//for (auto item : m_items)
|
|
|
|
// item->setBoundingRect(0, 0, endX, item->boundingRect().height());
|
2016-07-31 18:48:41 +03:00
|
|
|
|
|
|
|
// Center view on the beginning of the scene
|
2016-08-08 22:45:57 +03:00
|
|
|
updateVisibleSceneRect();
|
2016-07-31 18:48:41 +03:00
|
|
|
setScrollbar(m_pScrollbar);
|
2016-08-08 22:17:56 +03:00
|
|
|
|
2016-08-01 22:21:59 +03:00
|
|
|
// Create new chronometer item (previous item was destroyed by scene on scene()->clear()).
|
|
|
|
// It will be shown on mouse right button click.
|
2016-08-06 14:50:31 +03:00
|
|
|
m_chronometerItemAux = createChronometer(false);
|
2016-08-10 22:08:27 +03:00
|
|
|
m_chronometerItem = createChronometer(true);
|
|
|
|
|
|
|
|
bgItem->setBoundingRect(0, 0, endX, y);
|
2016-08-18 23:26:41 +03:00
|
|
|
auto indicator = new EasyTimelineIndicatorItem();
|
2016-08-10 22:08:27 +03:00
|
|
|
indicator->setBoundingRect(0, 0, endX, y);
|
|
|
|
scene()->addItem(indicator);
|
2016-07-31 18:48:41 +03:00
|
|
|
|
|
|
|
// Setting flags
|
2016-08-18 23:26:41 +03:00
|
|
|
//m_bTest = false;
|
2016-07-31 18:48:41 +03:00
|
|
|
m_bEmpty = false;
|
2016-08-01 22:21:59 +03:00
|
|
|
|
|
|
|
scaleTo(BASE_SCALE);
|
2016-09-08 22:42:35 +03:00
|
|
|
|
|
|
|
|
|
|
|
emit treeChanged();
|
|
|
|
|
|
|
|
if (mainThreadItem != nullptr)
|
|
|
|
longestItem = mainThreadItem;
|
|
|
|
|
|
|
|
if (longestItem != nullptr)
|
|
|
|
{
|
|
|
|
m_pScrollbar->setMinimapFrom(longestItem->threadId(), longestItem->items(0));
|
|
|
|
EASY_GLOBALS.selected_thread = longestItem->threadId();
|
|
|
|
emit EASY_GLOBALS.events.selectedThreadChanged(longestItem->threadId());
|
|
|
|
}
|
2016-06-27 23:22:12 +03:00
|
|
|
}
|
2016-06-26 18:46:51 +03:00
|
|
|
|
2016-08-18 23:26:41 +03:00
|
|
|
const EasyGraphicsView::Items &EasyGraphicsView::getItems() const
|
2016-08-09 00:45:45 +03:00
|
|
|
{
|
|
|
|
return m_items;
|
|
|
|
}
|
|
|
|
|
2016-08-18 23:26:41 +03:00
|
|
|
qreal EasyGraphicsView::setTree(EasyGraphicsItem* _item, const ::profiler::BlocksTree::children_t& _children, qreal& _height, qreal _y, short _level)
|
2016-06-30 02:57:57 +03:00
|
|
|
{
|
2016-08-01 22:21:59 +03:00
|
|
|
static const qreal MIN_DURATION = 0.25;
|
|
|
|
|
2016-06-30 02:57:57 +03:00
|
|
|
if (_children.empty())
|
|
|
|
{
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2016-09-04 14:48:35 +03:00
|
|
|
const auto level = static_cast<uint8_t>(_level);
|
2016-08-11 23:43:34 +03:00
|
|
|
_item->reserve(level, static_cast<unsigned int>(_children.size()));
|
2016-07-10 01:24:31 +03:00
|
|
|
|
2016-08-11 23:43:34 +03:00
|
|
|
const short next_level = _level + 1;
|
|
|
|
bool warned = false;
|
2016-06-30 02:57:57 +03:00
|
|
|
qreal total_duration = 0, prev_end = 0, maxh = 0;
|
2016-07-31 13:13:48 +03:00
|
|
|
qreal start_time = -1;
|
2016-08-30 22:51:18 +03:00
|
|
|
for (auto child_index : _children)
|
2016-06-30 02:57:57 +03:00
|
|
|
{
|
2016-08-30 22:51:18 +03:00
|
|
|
auto& gui_block = easyBlock(child_index);
|
|
|
|
const auto& child = gui_block.tree;
|
|
|
|
|
2016-08-28 18:19:12 +03:00
|
|
|
auto xbegin = time2position(child.node->begin());
|
2016-07-31 13:13:48 +03:00
|
|
|
if (start_time < 0)
|
|
|
|
{
|
|
|
|
start_time = xbegin;
|
|
|
|
}
|
2016-06-30 02:57:57 +03:00
|
|
|
|
2016-08-28 18:19:12 +03:00
|
|
|
auto duration = time2position(child.node->end()) - xbegin;
|
2016-07-31 13:13:48 +03:00
|
|
|
|
2016-08-23 22:42:20 +03:00
|
|
|
//const auto dt = xbegin - prev_end;
|
|
|
|
//if (dt < 0)
|
|
|
|
//{
|
|
|
|
// duration += dt;
|
|
|
|
// xbegin -= dt;
|
|
|
|
//}
|
2016-06-30 02:57:57 +03:00
|
|
|
|
2016-08-01 22:21:59 +03:00
|
|
|
if (duration < MIN_DURATION)
|
2016-06-30 02:57:57 +03:00
|
|
|
{
|
2016-08-01 22:21:59 +03:00
|
|
|
duration = MIN_DURATION;
|
2016-06-30 02:57:57 +03:00
|
|
|
}
|
|
|
|
|
2016-08-11 23:43:34 +03:00
|
|
|
auto i = _item->addItem(level);
|
|
|
|
auto& b = _item->getItem(level, i);
|
2016-07-27 21:50:11 +03:00
|
|
|
|
2016-08-11 23:43:34 +03:00
|
|
|
gui_block.graphics_item = _item->index();
|
|
|
|
gui_block.graphics_item_level = level;
|
2016-08-04 22:38:45 +03:00
|
|
|
gui_block.graphics_item_index = i;
|
|
|
|
|
2016-08-11 23:43:34 +03:00
|
|
|
if (next_level < 256 && next_level < _item->levels() && !child.children.empty())
|
2016-07-27 21:50:11 +03:00
|
|
|
{
|
2016-09-04 14:48:35 +03:00
|
|
|
b.children_begin = static_cast<unsigned int>(_item->items(static_cast<uint8_t>(next_level)).size());
|
2016-07-27 21:50:11 +03:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2016-08-11 23:43:34 +03:00
|
|
|
::profiler_gui::set_max(b.children_begin);
|
2016-07-27 21:50:11 +03:00
|
|
|
}
|
2016-06-30 02:57:57 +03:00
|
|
|
|
|
|
|
qreal h = 0;
|
2016-08-11 23:43:34 +03:00
|
|
|
qreal children_duration = 0;
|
|
|
|
|
|
|
|
if (next_level < 256)
|
|
|
|
{
|
|
|
|
children_duration = setTree(_item, child.children, h, _y + GRAPHICS_ROW_SIZE_FULL, next_level);
|
|
|
|
}
|
|
|
|
else if (!child.children.empty() && !warned)
|
|
|
|
{
|
|
|
|
warned = true;
|
2016-08-23 22:42:20 +03:00
|
|
|
qWarning() << "Warning: Maximum blocks depth (255) exceeded! See EasyGraphicsView::setTree() : " << __LINE__ << " in file " << __FILE__;
|
2016-08-11 23:43:34 +03:00
|
|
|
}
|
|
|
|
|
2016-06-27 23:22:12 +03:00
|
|
|
if (duration < children_duration)
|
|
|
|
{
|
|
|
|
duration = children_duration;
|
|
|
|
}
|
|
|
|
|
2016-06-30 02:57:57 +03:00
|
|
|
if (h > maxh)
|
|
|
|
{
|
|
|
|
maxh = h;
|
|
|
|
}
|
|
|
|
|
2016-08-30 22:51:18 +03:00
|
|
|
b.block = child_index;// &child;
|
2016-09-07 21:48:50 +03:00
|
|
|
b.color = EASY_GLOBALS.descriptors[child.node->id()]->color();// ::profiler_gui::fromProfilerRgb(::profiler::colors::get_red(color), ::profiler::colors::get_green(color), ::profiler::colors::get_blue(color));
|
2016-08-11 23:43:34 +03:00
|
|
|
b.setPos(xbegin, duration);
|
2016-06-30 02:57:57 +03:00
|
|
|
b.totalHeight = GRAPHICS_ROW_SIZE + h;
|
2016-08-23 22:42:20 +03:00
|
|
|
b.state = BLOCK_ITEM_UNCHANGED;
|
2016-06-27 23:22:12 +03:00
|
|
|
|
2016-06-30 02:57:57 +03:00
|
|
|
prev_end = xbegin + duration;
|
2016-07-31 13:13:48 +03:00
|
|
|
total_duration = prev_end - start_time;
|
2016-06-26 18:46:51 +03:00
|
|
|
}
|
2016-06-27 23:22:12 +03:00
|
|
|
|
2016-06-30 02:57:57 +03:00
|
|
|
_height += GRAPHICS_ROW_SIZE_FULL + maxh;
|
|
|
|
|
2016-06-27 23:22:12 +03:00
|
|
|
return total_duration;
|
2016-06-26 18:46:51 +03:00
|
|
|
}
|
|
|
|
|
2016-07-31 18:48:41 +03:00
|
|
|
//////////////////////////////////////////////////////////////////////////
|
2016-06-26 18:46:51 +03:00
|
|
|
|
2016-08-18 23:26:41 +03:00
|
|
|
void EasyGraphicsView::setScrollbar(EasyGraphicsScrollbar* _scrollbar)
|
2016-06-26 18:46:51 +03:00
|
|
|
{
|
2016-08-21 16:48:42 +03:00
|
|
|
auto const prevScrollbar = m_pScrollbar;
|
|
|
|
const bool makeConnect = prevScrollbar == nullptr || prevScrollbar != _scrollbar;
|
|
|
|
|
|
|
|
if (prevScrollbar != nullptr && prevScrollbar != _scrollbar)
|
2016-06-26 18:46:51 +03:00
|
|
|
{
|
2016-08-21 16:48:42 +03:00
|
|
|
disconnect(prevScrollbar, &EasyGraphicsScrollbar::valueChanged, this, &This::onGraphicsScrollbarValueChange);
|
2016-08-21 17:12:28 +03:00
|
|
|
disconnect(prevScrollbar, &EasyGraphicsScrollbar::wheeled, this, &This::onGraphicsScrollbarWheel);
|
2016-06-26 18:46:51 +03:00
|
|
|
}
|
2016-06-26 20:54:16 +03:00
|
|
|
|
2016-07-31 13:13:48 +03:00
|
|
|
m_pScrollbar = _scrollbar;
|
2016-08-03 23:00:04 +03:00
|
|
|
m_pScrollbar->setMinimapFrom(0, nullptr);
|
2016-08-03 00:06:36 +03:00
|
|
|
m_pScrollbar->hideChrono();
|
2016-07-31 13:13:48 +03:00
|
|
|
m_pScrollbar->setRange(0, scene()->width());
|
|
|
|
m_pScrollbar->setSliderWidth(m_visibleSceneRect.width());
|
|
|
|
m_pScrollbar->setValue(0);
|
2016-08-21 16:48:42 +03:00
|
|
|
|
|
|
|
if (makeConnect)
|
|
|
|
{
|
|
|
|
connect(m_pScrollbar, &EasyGraphicsScrollbar::valueChanged, this, &This::onGraphicsScrollbarValueChange);
|
2016-08-21 17:12:28 +03:00
|
|
|
connect(m_pScrollbar, &EasyGraphicsScrollbar::wheeled, this, &This::onGraphicsScrollbarWheel);
|
2016-08-21 16:48:42 +03:00
|
|
|
}
|
2016-08-03 23:00:04 +03:00
|
|
|
|
2016-08-30 22:51:18 +03:00
|
|
|
EASY_GLOBALS.selected_thread = 0;
|
|
|
|
emit EASY_GLOBALS.events.selectedThreadChanged(0);
|
2016-07-31 13:13:48 +03:00
|
|
|
}
|
|
|
|
|
2016-07-31 18:48:41 +03:00
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
|
2016-08-18 23:26:41 +03:00
|
|
|
void EasyGraphicsView::updateVisibleSceneRect()
|
2016-07-31 13:13:48 +03:00
|
|
|
{
|
|
|
|
m_visibleSceneRect = mapToScene(rect()).boundingRect();
|
2016-08-07 19:38:31 +03:00
|
|
|
|
|
|
|
auto vbar = verticalScrollBar();
|
|
|
|
if (vbar && vbar->isVisible())
|
|
|
|
m_visibleSceneRect.setWidth(m_visibleSceneRect.width() - vbar->width() - 2);
|
2016-08-21 14:26:04 +03:00
|
|
|
m_visibleSceneRect.setHeight(m_visibleSceneRect.height() - TIMELINE_ROW_SIZE);
|
2016-07-31 13:13:48 +03:00
|
|
|
}
|
|
|
|
|
2016-08-18 23:26:41 +03:00
|
|
|
void EasyGraphicsView::updateTimelineStep(qreal _windowWidth)
|
2016-08-10 22:08:27 +03:00
|
|
|
{
|
|
|
|
const auto time = units2microseconds(_windowWidth);
|
|
|
|
if (time < 100)
|
|
|
|
m_timelineStep = 1e-2;
|
|
|
|
else if (time < 10e3)
|
|
|
|
m_timelineStep = 1;
|
|
|
|
else if (time < 10e6)
|
|
|
|
m_timelineStep = 1e3;
|
|
|
|
else
|
|
|
|
m_timelineStep = 1e6;
|
|
|
|
|
2016-08-21 14:26:04 +03:00
|
|
|
const auto optimal_steps = static_cast<int>(40 * m_visibleSceneRect.width() / 1500);
|
2016-08-10 22:08:27 +03:00
|
|
|
auto steps = time / m_timelineStep;
|
2016-08-21 14:26:04 +03:00
|
|
|
while (steps > optimal_steps) {
|
2016-08-10 22:08:27 +03:00
|
|
|
m_timelineStep *= 10;
|
|
|
|
steps *= 0.1;
|
|
|
|
}
|
|
|
|
|
|
|
|
m_timelineStep = microseconds2units(m_timelineStep);
|
|
|
|
}
|
|
|
|
|
2016-09-09 00:07:27 +03:00
|
|
|
void EasyGraphicsView::repaintScene()
|
2016-07-31 13:13:48 +03:00
|
|
|
{
|
|
|
|
scene()->update(m_visibleSceneRect);
|
2016-09-09 00:07:27 +03:00
|
|
|
emit sceneUpdated();
|
2016-07-31 13:13:48 +03:00
|
|
|
}
|
|
|
|
|
2016-07-31 18:48:41 +03:00
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
|
2016-08-18 23:26:41 +03:00
|
|
|
void EasyGraphicsView::scaleTo(qreal _scale)
|
2016-08-01 22:21:59 +03:00
|
|
|
{
|
|
|
|
if (m_bEmpty)
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// have to limit scale because of Qt's QPainter feature: it doesn't draw text
|
|
|
|
// with very big coordinates (but it draw rectangles with the same coordinates good).
|
|
|
|
m_scale = clamp(MIN_SCALE, _scale, MAX_SCALE);
|
|
|
|
updateVisibleSceneRect();
|
|
|
|
|
|
|
|
// Update slider width for scrollbar
|
|
|
|
const auto windowWidth = m_visibleSceneRect.width() / m_scale;
|
|
|
|
m_pScrollbar->setSliderWidth(windowWidth);
|
|
|
|
|
2016-08-10 22:08:27 +03:00
|
|
|
updateTimelineStep(windowWidth);
|
2016-09-09 00:07:27 +03:00
|
|
|
repaintScene();
|
2016-08-01 22:21:59 +03:00
|
|
|
}
|
|
|
|
|
2016-08-18 23:26:41 +03:00
|
|
|
void EasyGraphicsView::wheelEvent(QWheelEvent* _event)
|
2016-07-31 13:13:48 +03:00
|
|
|
{
|
2016-08-21 16:48:42 +03:00
|
|
|
if (!m_bEmpty)
|
|
|
|
onWheel(mapToScene(_event->pos()).x(), _event->delta());
|
|
|
|
_event->accept();
|
|
|
|
}
|
2016-07-31 18:48:41 +03:00
|
|
|
|
2016-08-21 17:12:28 +03:00
|
|
|
void EasyGraphicsView::onGraphicsScrollbarWheel(qreal _mouseX, int _wheelDelta)
|
2016-08-21 16:48:42 +03:00
|
|
|
{
|
2016-08-21 17:12:28 +03:00
|
|
|
for (auto item : m_items)
|
|
|
|
{
|
2016-08-30 22:51:18 +03:00
|
|
|
if (item->threadId() == EASY_GLOBALS.selected_thread)
|
2016-08-21 17:12:28 +03:00
|
|
|
{
|
|
|
|
m_bUpdatingRect = true;
|
|
|
|
auto vbar = verticalScrollBar();
|
|
|
|
vbar->setValue(item->y() + (item->boundingRect().height() - vbar->pageStep()) * 0.5);
|
|
|
|
m_bUpdatingRect = false;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2016-08-21 16:48:42 +03:00
|
|
|
|
2016-08-21 17:12:28 +03:00
|
|
|
onWheel(_mouseX, _wheelDelta);
|
|
|
|
}
|
|
|
|
|
|
|
|
void EasyGraphicsView::onWheel(qreal _mouseX, int _wheelDelta)
|
|
|
|
{
|
2016-08-21 22:44:03 +03:00
|
|
|
const decltype(m_scale) scaleCoeff = _wheelDelta > 0 ? ::profiler_gui::SCALING_COEFFICIENT : ::profiler_gui::SCALING_COEFFICIENT_INV;
|
2016-07-31 13:13:48 +03:00
|
|
|
|
|
|
|
// Remember current mouse position
|
2016-09-09 00:07:27 +03:00
|
|
|
_mouseX = clamp(0., _mouseX, scene()->width());
|
2016-08-21 16:48:42 +03:00
|
|
|
const auto mousePosition = m_offset + _mouseX / m_scale;
|
2016-07-31 13:13:48 +03:00
|
|
|
|
2016-08-01 22:21:59 +03:00
|
|
|
// have to limit scale because of Qt's QPainter feature: it doesn't draw text
|
|
|
|
// with very big coordinates (but it draw rectangles with the same coordinates good).
|
|
|
|
m_scale = clamp(MIN_SCALE, m_scale * scaleCoeff, MAX_SCALE);
|
2016-06-26 20:54:16 +03:00
|
|
|
|
2016-08-21 17:12:28 +03:00
|
|
|
//updateVisibleSceneRect(); // Update scene rect
|
2016-06-27 23:22:12 +03:00
|
|
|
|
2016-07-31 13:13:48 +03:00
|
|
|
// Update slider width for scrollbar
|
|
|
|
const auto windowWidth = m_visibleSceneRect.width() / m_scale;
|
|
|
|
m_pScrollbar->setSliderWidth(windowWidth);
|
|
|
|
|
|
|
|
// Calculate new offset to simulate QGraphicsView::AnchorUnderMouse scaling behavior
|
2016-08-21 16:48:42 +03:00
|
|
|
m_offset = clamp(0., mousePosition - _mouseX / m_scale, scene()->width() - windowWidth);
|
2016-07-31 13:13:48 +03:00
|
|
|
|
|
|
|
// Update slider position
|
2016-08-01 22:21:59 +03:00
|
|
|
m_bUpdatingRect = true; // To be sure that updateVisibleSceneRect will not be called by scrollbar change
|
2016-07-31 13:13:48 +03:00
|
|
|
m_pScrollbar->setValue(m_offset);
|
|
|
|
m_bUpdatingRect = false;
|
|
|
|
|
2016-08-21 17:12:28 +03:00
|
|
|
updateVisibleSceneRect(); // Update scene rect
|
2016-08-10 22:08:27 +03:00
|
|
|
updateTimelineStep(windowWidth);
|
2016-09-09 00:07:27 +03:00
|
|
|
repaintScene(); // repaint scene
|
2016-06-26 20:54:16 +03:00
|
|
|
}
|
|
|
|
|
2016-07-31 18:48:41 +03:00
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
|
2016-08-18 23:26:41 +03:00
|
|
|
void EasyGraphicsView::mousePressEvent(QMouseEvent* _event)
|
2016-06-26 20:54:16 +03:00
|
|
|
{
|
2016-07-31 18:48:41 +03:00
|
|
|
if (m_bEmpty)
|
|
|
|
{
|
|
|
|
_event->accept();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2016-08-08 22:45:57 +03:00
|
|
|
m_mouseButtons = _event->buttons();
|
|
|
|
m_mousePressPos = _event->pos();
|
|
|
|
|
|
|
|
if (m_mouseButtons & Qt::RightButton)
|
|
|
|
{
|
|
|
|
const auto mouseX = m_offset + mapToScene(m_mousePressPos).x() / m_scale;
|
|
|
|
m_chronometerItem->setLeftRight(mouseX, mouseX);
|
|
|
|
m_chronometerItem->setReverse(false);
|
|
|
|
m_chronometerItem->hide();
|
|
|
|
m_pScrollbar->hideChrono();
|
|
|
|
}
|
|
|
|
|
|
|
|
_event->accept();
|
2016-06-26 20:54:16 +03:00
|
|
|
}
|
|
|
|
|
2016-08-18 23:26:41 +03:00
|
|
|
void EasyGraphicsView::mouseDoubleClickEvent(QMouseEvent* _event)
|
2016-08-06 14:50:31 +03:00
|
|
|
{
|
|
|
|
if (m_bEmpty)
|
|
|
|
{
|
|
|
|
_event->accept();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
m_mouseButtons = _event->buttons();
|
2016-08-08 22:17:56 +03:00
|
|
|
m_mousePressPos = _event->pos();
|
2016-08-06 14:50:31 +03:00
|
|
|
m_bDoubleClick = true;
|
|
|
|
|
2016-08-08 22:45:57 +03:00
|
|
|
if (m_mouseButtons & Qt::LeftButton)
|
|
|
|
{
|
|
|
|
const auto mouseX = m_offset + mapToScene(m_mousePressPos).x() / m_scale;
|
|
|
|
m_chronometerItemAux->setLeftRight(mouseX, mouseX);
|
|
|
|
m_chronometerItemAux->setReverse(false);
|
|
|
|
m_chronometerItemAux->hide();
|
2016-09-09 00:07:27 +03:00
|
|
|
emit sceneUpdated();
|
2016-08-06 14:50:31 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
_event->accept();
|
|
|
|
}
|
|
|
|
|
2016-07-31 18:48:41 +03:00
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
|
2016-08-18 23:26:41 +03:00
|
|
|
void EasyGraphicsView::mouseReleaseEvent(QMouseEvent* _event)
|
2016-06-26 20:54:16 +03:00
|
|
|
{
|
2016-07-31 18:48:41 +03:00
|
|
|
if (m_bEmpty)
|
|
|
|
{
|
|
|
|
_event->accept();
|
|
|
|
return;
|
|
|
|
}
|
2016-08-08 22:45:57 +03:00
|
|
|
|
2016-09-09 00:07:27 +03:00
|
|
|
bool chronoHidden = false;
|
2016-08-21 14:26:04 +03:00
|
|
|
bool changedSelection = false, changedSelectedItem = false;
|
2016-08-08 22:45:57 +03:00
|
|
|
if (m_mouseButtons & Qt::RightButton)
|
|
|
|
{
|
|
|
|
if (m_chronometerItem->isVisible() && m_chronometerItem->width() < 1e-6)
|
|
|
|
{
|
|
|
|
m_chronometerItem->hide();
|
|
|
|
m_chronometerItem->setHover(false);
|
|
|
|
m_pScrollbar->hideChrono();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!m_selectedBlocks.empty())
|
|
|
|
{
|
|
|
|
changedSelection = true;
|
|
|
|
m_selectedBlocks.clear();
|
|
|
|
}
|
|
|
|
|
2016-08-18 23:26:41 +03:00
|
|
|
if (m_chronometerItem->isVisible())
|
2016-08-08 22:45:57 +03:00
|
|
|
{
|
|
|
|
//printf("INTERVAL: {%lf, %lf} ms\n", m_chronometerItem->left(), m_chronometerItem->right());
|
|
|
|
|
|
|
|
for (auto item : m_items)
|
|
|
|
{
|
|
|
|
item->getBlocks(m_chronometerItem->left(), m_chronometerItem->right(), m_selectedBlocks);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!m_selectedBlocks.empty())
|
|
|
|
{
|
|
|
|
changedSelection = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-08-30 22:51:18 +03:00
|
|
|
const ::profiler_gui::EasyBlockItem* selectedBlock = nullptr;
|
|
|
|
const auto previouslySelectedBlock = EASY_GLOBALS.selected_block;
|
2016-08-08 22:45:57 +03:00
|
|
|
if (m_mouseButtons & Qt::LeftButton)
|
|
|
|
{
|
2016-08-21 14:26:04 +03:00
|
|
|
bool clicked = false;
|
|
|
|
|
2016-08-08 22:45:57 +03:00
|
|
|
if (m_chronometerItemAux->isVisible() && m_chronometerItemAux->width() < 1e-6)
|
|
|
|
{
|
2016-09-09 00:07:27 +03:00
|
|
|
chronoHidden = true;
|
2016-08-08 22:45:57 +03:00
|
|
|
m_chronometerItemAux->hide();
|
|
|
|
}
|
2016-08-18 23:26:41 +03:00
|
|
|
else if (m_chronometerItem->isVisible() && m_chronometerItem->hoverIndicator())
|
2016-08-08 22:45:57 +03:00
|
|
|
{
|
|
|
|
// Jump to selected zone
|
|
|
|
clicked = true;
|
|
|
|
m_flickerSpeedX = m_flickerSpeedY = 0;
|
|
|
|
m_pScrollbar->setValue(m_chronometerItem->left() + m_chronometerItem->width() * 0.5 - m_pScrollbar->sliderHalfWidth());
|
|
|
|
}
|
|
|
|
|
2016-08-18 23:26:41 +03:00
|
|
|
if (!clicked && m_mouseMovePath.manhattanLength() < 5)
|
2016-08-08 22:45:57 +03:00
|
|
|
{
|
|
|
|
// Handle Click
|
|
|
|
|
2016-08-21 14:26:04 +03:00
|
|
|
//clicked = true;
|
2016-08-08 22:45:57 +03:00
|
|
|
auto mouseClickPos = mapToScene(m_mousePressPos);
|
2016-09-08 22:42:35 +03:00
|
|
|
if (mouseClickPos.x() >= 0)
|
2016-08-08 22:45:57 +03:00
|
|
|
{
|
2016-09-08 22:42:35 +03:00
|
|
|
mouseClickPos.setX(m_offset + mouseClickPos.x() / m_scale);
|
|
|
|
|
|
|
|
// Try to select one of item blocks
|
|
|
|
for (auto item : m_items)
|
2016-08-08 22:45:57 +03:00
|
|
|
{
|
2016-09-08 22:42:35 +03:00
|
|
|
auto block = item->intersect(mouseClickPos);
|
|
|
|
if (block)
|
|
|
|
{
|
|
|
|
changedSelectedItem = true;
|
|
|
|
selectedBlock = block;
|
|
|
|
EASY_GLOBALS.selected_block = block->block;
|
|
|
|
break;
|
|
|
|
}
|
2016-08-08 22:45:57 +03:00
|
|
|
}
|
2016-08-21 22:44:03 +03:00
|
|
|
|
2016-09-08 22:42:35 +03:00
|
|
|
if (!changedSelectedItem && EASY_GLOBALS.selected_block != ::profiler_gui::numeric_max(EASY_GLOBALS.selected_block))
|
|
|
|
{
|
|
|
|
changedSelectedItem = true;
|
|
|
|
::profiler_gui::set_max(EASY_GLOBALS.selected_block);
|
|
|
|
}
|
2016-08-21 22:44:03 +03:00
|
|
|
}
|
2016-08-08 22:45:57 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
m_bDoubleClick = false;
|
|
|
|
m_mouseButtons = _event->buttons();
|
|
|
|
m_mouseMovePath = QPoint();
|
|
|
|
_event->accept();
|
|
|
|
|
|
|
|
if (changedSelection)
|
|
|
|
{
|
|
|
|
emit intervalChanged(m_selectedBlocks, m_beginTime, position2time(m_chronometerItem->left()), position2time(m_chronometerItem->right()), m_chronometerItem->reverse());
|
|
|
|
}
|
|
|
|
|
|
|
|
m_chronometerItem->setReverse(false);
|
|
|
|
|
|
|
|
if (changedSelectedItem)
|
|
|
|
{
|
|
|
|
m_bUpdatingRect = true;
|
2016-08-30 22:51:18 +03:00
|
|
|
if (selectedBlock != nullptr && previouslySelectedBlock == EASY_GLOBALS.selected_block && !blocksTree(selectedBlock->block).children.empty())
|
2016-08-21 22:44:03 +03:00
|
|
|
{
|
2016-08-30 22:51:18 +03:00
|
|
|
EASY_GLOBALS.gui_blocks[previouslySelectedBlock].expanded = !EASY_GLOBALS.gui_blocks[previouslySelectedBlock].expanded;
|
|
|
|
emit EASY_GLOBALS.events.itemsExpandStateChanged();
|
2016-08-21 22:44:03 +03:00
|
|
|
}
|
2016-08-30 22:51:18 +03:00
|
|
|
emit EASY_GLOBALS.events.selectedBlockChanged(EASY_GLOBALS.selected_block);
|
2016-08-08 22:45:57 +03:00
|
|
|
m_bUpdatingRect = false;
|
|
|
|
|
2016-09-09 00:07:27 +03:00
|
|
|
repaintScene();
|
|
|
|
}
|
|
|
|
else if (chronoHidden)
|
|
|
|
{
|
|
|
|
emit sceneUpdated();
|
2016-08-08 22:45:57 +03:00
|
|
|
}
|
2016-06-26 20:54:16 +03:00
|
|
|
}
|
|
|
|
|
2016-07-31 18:48:41 +03:00
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
|
2016-08-18 23:26:41 +03:00
|
|
|
bool EasyGraphicsView::moveChrono(EasyChronometerItem* _chronometerItem, qreal _mouseX)
|
2016-08-06 14:50:31 +03:00
|
|
|
{
|
2016-08-08 22:45:57 +03:00
|
|
|
if (_chronometerItem->reverse())
|
|
|
|
{
|
|
|
|
if (_mouseX > _chronometerItem->right())
|
|
|
|
{
|
|
|
|
_chronometerItem->setReverse(false);
|
|
|
|
_chronometerItem->setLeftRight(_chronometerItem->right(), _mouseX);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
_chronometerItem->setLeftRight(_mouseX, _chronometerItem->right());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (_mouseX < _chronometerItem->left())
|
|
|
|
{
|
|
|
|
_chronometerItem->setReverse(true);
|
|
|
|
_chronometerItem->setLeftRight(_mouseX, _chronometerItem->left());
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
_chronometerItem->setLeftRight(_chronometerItem->left(), _mouseX);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!_chronometerItem->isVisible() && _chronometerItem->width() > 1e-6)
|
|
|
|
{
|
|
|
|
_chronometerItem->show();
|
|
|
|
return true;
|
2016-08-06 14:50:31 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2016-08-18 23:26:41 +03:00
|
|
|
void EasyGraphicsView::mouseMoveEvent(QMouseEvent* _event)
|
2016-06-26 20:54:16 +03:00
|
|
|
{
|
2016-08-08 22:17:56 +03:00
|
|
|
if (m_bEmpty || (m_mouseButtons == 0 && !m_chronometerItem->isVisible()))
|
2016-07-31 18:48:41 +03:00
|
|
|
{
|
|
|
|
_event->accept();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool needUpdate = false;
|
2016-08-08 22:45:57 +03:00
|
|
|
const auto pos = _event->pos();
|
|
|
|
const auto delta = pos - m_mousePressPos;
|
2016-08-08 22:17:56 +03:00
|
|
|
m_mousePressPos = pos;
|
|
|
|
|
|
|
|
if (m_mouseButtons != 0)
|
|
|
|
{
|
|
|
|
m_mouseMovePath.setX(m_mouseMovePath.x() + ::std::abs(delta.x()));
|
|
|
|
m_mouseMovePath.setY(m_mouseMovePath.y() + ::std::abs(delta.y()));
|
|
|
|
}
|
|
|
|
|
2016-08-08 22:45:57 +03:00
|
|
|
auto mouseScenePos = mapToScene(m_mousePressPos);
|
2016-08-08 22:17:56 +03:00
|
|
|
mouseScenePos.setX(m_offset + mouseScenePos.x() / m_scale);
|
2016-09-09 00:07:27 +03:00
|
|
|
const auto x = clamp(0., mouseScenePos.x(), scene()->width());
|
2016-07-31 18:48:41 +03:00
|
|
|
|
2016-08-08 22:45:57 +03:00
|
|
|
if (m_mouseButtons & Qt::RightButton)
|
|
|
|
{
|
2016-09-08 22:42:35 +03:00
|
|
|
bool showItem = moveChrono(m_chronometerItem, x);
|
2016-08-08 22:45:57 +03:00
|
|
|
m_pScrollbar->setChronoPos(m_chronometerItem->left(), m_chronometerItem->right());
|
|
|
|
|
|
|
|
if (showItem)
|
|
|
|
{
|
|
|
|
m_pScrollbar->showChrono();
|
|
|
|
}
|
|
|
|
|
|
|
|
needUpdate = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (m_mouseButtons & Qt::LeftButton)
|
|
|
|
{
|
|
|
|
if (m_bDoubleClick)
|
|
|
|
{
|
2016-09-08 22:42:35 +03:00
|
|
|
moveChrono(m_chronometerItemAux, x);
|
2016-08-08 22:45:57 +03:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
auto vbar = verticalScrollBar();
|
|
|
|
|
|
|
|
m_bUpdatingRect = true; // Block scrollbars from updating scene rect to make it possible to do it only once
|
|
|
|
vbar->setValue(vbar->value() - delta.y());
|
|
|
|
m_pScrollbar->setValue(m_pScrollbar->value() - delta.x() / m_scale);
|
|
|
|
m_bUpdatingRect = false;
|
|
|
|
// Seems like an ugly stub, but QSignalBlocker is also a bad decision
|
|
|
|
// because if scrollbar does not emit valueChanged signal then viewport does not move
|
|
|
|
|
|
|
|
updateVisibleSceneRect(); // Update scene visible rect only once
|
|
|
|
|
|
|
|
// Update flicker speed
|
|
|
|
m_flickerSpeedX += delta.x() >> 1;
|
|
|
|
m_flickerSpeedY += delta.y() >> 1;
|
|
|
|
if (!m_flickerTimer.isActive())
|
|
|
|
{
|
|
|
|
// If flicker timer is not started, then start it
|
|
|
|
m_flickerTimer.start(FLICKER_INTERVAL);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
needUpdate = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (m_chronometerItem->isVisible())
|
|
|
|
{
|
2016-08-18 23:26:41 +03:00
|
|
|
auto prevValue = m_chronometerItem->hoverIndicator();
|
2016-08-08 22:45:57 +03:00
|
|
|
m_chronometerItem->setHover(m_chronometerItem->contains(mouseScenePos));
|
2016-08-18 23:26:41 +03:00
|
|
|
needUpdate = needUpdate || (prevValue != m_chronometerItem->hoverIndicator());
|
2016-08-08 22:45:57 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
if (needUpdate)
|
|
|
|
{
|
2016-09-09 00:07:27 +03:00
|
|
|
repaintScene(); // repaint scene
|
2016-08-08 22:45:57 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
_event->accept();
|
2016-06-26 20:54:16 +03:00
|
|
|
}
|
|
|
|
|
2016-07-31 18:48:41 +03:00
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
|
2016-09-09 00:07:27 +03:00
|
|
|
void EasyGraphicsView::keyPressEvent(QKeyEvent* _event)
|
|
|
|
{
|
|
|
|
static const int KeyStep = 100;
|
|
|
|
|
|
|
|
const int key = _event->key();
|
|
|
|
|
|
|
|
switch (key)
|
|
|
|
{
|
|
|
|
case Qt::Key_Right:
|
|
|
|
case Qt::Key_6:
|
|
|
|
{
|
|
|
|
m_pScrollbar->setValue(m_pScrollbar->value() + KeyStep / m_scale);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case Qt::Key_Left:
|
|
|
|
case Qt::Key_4:
|
|
|
|
{
|
|
|
|
m_pScrollbar->setValue(m_pScrollbar->value() - KeyStep / m_scale);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case Qt::Key_Up:
|
|
|
|
case Qt::Key_8:
|
|
|
|
{
|
|
|
|
auto vbar = verticalScrollBar();
|
|
|
|
vbar->setValue(vbar->value() - KeyStep);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case Qt::Key_Down:
|
|
|
|
case Qt::Key_2:
|
|
|
|
{
|
|
|
|
auto vbar = verticalScrollBar();
|
|
|
|
vbar->setValue(vbar->value() + KeyStep);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case Qt::Key_Plus:
|
|
|
|
case Qt::Key_Equal:
|
|
|
|
{
|
|
|
|
onWheel(mapToScene(mapFromGlobal(QCursor::pos())).x(), KeyStep);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case Qt::Key_Minus:
|
|
|
|
{
|
|
|
|
onWheel(mapToScene(mapFromGlobal(QCursor::pos())).x(), -KeyStep);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
m_keys.insert(key);
|
|
|
|
_event->accept();
|
|
|
|
}
|
|
|
|
|
|
|
|
void EasyGraphicsView::keyReleaseEvent(QKeyEvent* _event)
|
|
|
|
{
|
|
|
|
const int key = _event->key();
|
|
|
|
|
|
|
|
m_keys.erase(key);
|
|
|
|
_event->accept();
|
|
|
|
}
|
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
|
2016-08-18 23:26:41 +03:00
|
|
|
void EasyGraphicsView::resizeEvent(QResizeEvent* _event)
|
2016-07-31 18:48:41 +03:00
|
|
|
{
|
2016-09-11 16:53:34 +03:00
|
|
|
Parent::resizeEvent(_event);
|
|
|
|
|
|
|
|
const QRectF previousRect = m_visibleSceneRect;
|
|
|
|
updateVisibleSceneRect(); // Update scene visible rect only once
|
|
|
|
|
|
|
|
// Update slider width for scrollbar
|
|
|
|
const auto windowWidth = m_visibleSceneRect.width() / m_scale;
|
|
|
|
m_pScrollbar->setSliderWidth(windowWidth);
|
|
|
|
|
|
|
|
// Calculate new offset to save old screen center
|
|
|
|
const auto deltaWidth = m_visibleSceneRect.width() - previousRect.width();
|
|
|
|
m_offset = clamp(0., m_offset - deltaWidth * 0.5 / m_scale, scene()->width() - windowWidth);
|
|
|
|
|
|
|
|
// Update slider position
|
|
|
|
m_bUpdatingRect = true; // To be sure that updateVisibleSceneRect will not be called by scrollbar change
|
|
|
|
m_pScrollbar->setValue(m_offset);
|
|
|
|
m_bUpdatingRect = false;
|
|
|
|
|
2016-09-09 00:07:27 +03:00
|
|
|
repaintScene(); // repaint scene
|
2016-07-31 18:48:41 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
|
2016-08-18 23:26:41 +03:00
|
|
|
void EasyGraphicsView::initMode()
|
2016-06-26 20:54:16 +03:00
|
|
|
{
|
2016-06-27 23:22:12 +03:00
|
|
|
// TODO: find mode with least number of bugs :)
|
|
|
|
// There are always some display bugs...
|
|
|
|
|
|
|
|
setCacheMode(QGraphicsView::CacheNone);
|
2016-06-26 20:54:16 +03:00
|
|
|
setTransformationAnchor(QGraphicsView::AnchorUnderMouse);
|
2016-07-10 01:24:31 +03:00
|
|
|
setViewportUpdateMode(QGraphicsView::FullViewportUpdate);
|
|
|
|
setOptimizationFlag(QGraphicsView::DontSavePainterState, true);
|
2016-06-27 23:22:12 +03:00
|
|
|
|
2016-07-31 13:13:48 +03:00
|
|
|
setHorizontalScrollBarPolicy(Qt::ScrollBarAlwaysOff);
|
2016-06-30 02:57:57 +03:00
|
|
|
connect(verticalScrollBar(), &QScrollBar::valueChanged, this, &This::onScrollbarValueChange);
|
2016-07-27 22:52:13 +03:00
|
|
|
connect(&m_flickerTimer, &QTimer::timeout, this, &This::onFlickerTimeout);
|
2016-08-24 01:00:24 +03:00
|
|
|
|
2016-08-30 22:51:18 +03:00
|
|
|
auto globalSignals = &EASY_GLOBALS.events;
|
2016-08-24 01:00:24 +03:00
|
|
|
connect(globalSignals, &::profiler_gui::EasyGlobalSignals::selectedThreadChanged, this, &This::onSelectedThreadChange);
|
|
|
|
connect(globalSignals, &::profiler_gui::EasyGlobalSignals::selectedBlockChanged, this, &This::onSelectedBlockChange);
|
|
|
|
connect(globalSignals, &::profiler_gui::EasyGlobalSignals::itemsExpandStateChanged, this, &This::onItemsEspandStateChange);
|
2016-09-09 00:07:27 +03:00
|
|
|
connect(globalSignals, &::profiler_gui::EasyGlobalSignals::drawBordersChanged, this, &This::repaintScene);
|
|
|
|
connect(globalSignals, &::profiler_gui::EasyGlobalSignals::chronoPositionChanged, this, &This::repaintScene);
|
2016-06-26 20:54:16 +03:00
|
|
|
}
|
|
|
|
|
2016-07-31 18:48:41 +03:00
|
|
|
//////////////////////////////////////////////////////////////////////////
|
2016-06-30 02:57:57 +03:00
|
|
|
|
2016-08-18 23:26:41 +03:00
|
|
|
void EasyGraphicsView::onScrollbarValueChange(int)
|
2016-06-30 02:57:57 +03:00
|
|
|
{
|
2016-07-31 18:48:41 +03:00
|
|
|
if (!m_bUpdatingRect && !m_bEmpty)
|
2016-06-30 02:57:57 +03:00
|
|
|
updateVisibleSceneRect();
|
2016-06-26 18:46:51 +03:00
|
|
|
}
|
|
|
|
|
2016-08-18 23:26:41 +03:00
|
|
|
void EasyGraphicsView::onGraphicsScrollbarValueChange(qreal _value)
|
2016-07-31 13:13:48 +03:00
|
|
|
{
|
2016-07-31 18:48:41 +03:00
|
|
|
if (!m_bEmpty)
|
2016-07-31 13:13:48 +03:00
|
|
|
{
|
2016-07-31 18:48:41 +03:00
|
|
|
m_offset = _value;
|
|
|
|
if (!m_bUpdatingRect)
|
|
|
|
{
|
|
|
|
updateVisibleSceneRect();
|
2016-09-09 00:07:27 +03:00
|
|
|
repaintScene();
|
2016-07-31 18:48:41 +03:00
|
|
|
}
|
2016-07-31 13:13:48 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-07-31 18:48:41 +03:00
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
|
2016-08-18 23:26:41 +03:00
|
|
|
void EasyGraphicsView::onFlickerTimeout()
|
2016-07-27 22:52:13 +03:00
|
|
|
{
|
|
|
|
if (m_mouseButtons & Qt::LeftButton)
|
|
|
|
{
|
|
|
|
// Fast slow-down and stop if mouse button is pressed, no flicking.
|
2016-08-06 14:50:31 +03:00
|
|
|
m_flickerSpeedX >>= 1;
|
|
|
|
m_flickerSpeedY >>= 1;
|
2016-08-21 22:44:03 +03:00
|
|
|
if (m_flickerSpeedX == -1) m_flickerSpeedX = 0;
|
|
|
|
if (m_flickerSpeedY == -1) m_flickerSpeedY = 0;
|
2016-07-27 22:52:13 +03:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// Flick when mouse button is not pressed
|
|
|
|
|
2016-08-06 14:50:31 +03:00
|
|
|
auto vbar = verticalScrollBar();
|
|
|
|
|
2016-08-08 22:45:57 +03:00
|
|
|
m_bUpdatingRect = true; // Block scrollbars from updating scene rect to make it possible to do it only once
|
|
|
|
m_pScrollbar->setValue(m_pScrollbar->value() - m_flickerSpeedX / m_scale);
|
|
|
|
vbar->setValue(vbar->value() - m_flickerSpeedY);
|
|
|
|
m_bUpdatingRect = false;
|
|
|
|
// Seems like an ugly stub, but QSignalBlocker is also a bad decision
|
|
|
|
// because if scrollbar does not emit valueChanged signal then viewport does not move
|
|
|
|
|
2016-07-27 22:52:13 +03:00
|
|
|
updateVisibleSceneRect(); // Update scene visible rect only once
|
2016-09-09 00:07:27 +03:00
|
|
|
repaintScene(); // repaint scene
|
2016-08-06 14:50:31 +03:00
|
|
|
|
|
|
|
m_flickerSpeedX -= absmin(sign(m_flickerSpeedX), m_flickerSpeedX);
|
|
|
|
m_flickerSpeedY -= absmin(sign(m_flickerSpeedY), m_flickerSpeedY);
|
2016-07-27 22:52:13 +03:00
|
|
|
}
|
|
|
|
|
2016-08-06 14:50:31 +03:00
|
|
|
if (m_flickerSpeedX == 0 && m_flickerSpeedY == 0)
|
2016-07-27 22:52:13 +03:00
|
|
|
{
|
|
|
|
// Flicker stopped, no timer needed.
|
|
|
|
m_flickerTimer.stop();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-07-31 18:48:41 +03:00
|
|
|
//////////////////////////////////////////////////////////////////////////
|
2016-07-31 13:13:48 +03:00
|
|
|
|
2016-08-18 23:26:41 +03:00
|
|
|
void EasyGraphicsView::onSelectedThreadChange(::profiler::thread_id_t _id)
|
2016-08-03 23:00:04 +03:00
|
|
|
{
|
2016-08-18 23:26:41 +03:00
|
|
|
if (m_pScrollbar == nullptr || m_pScrollbar->minimapThread() == _id)
|
2016-08-03 23:00:04 +03:00
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (_id == 0)
|
|
|
|
{
|
|
|
|
m_pScrollbar->setMinimapFrom(0, nullptr);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (auto item : m_items)
|
|
|
|
{
|
|
|
|
if (item->threadId() == _id)
|
|
|
|
{
|
|
|
|
m_pScrollbar->setMinimapFrom(_id, item->items(0));
|
2016-09-09 00:07:27 +03:00
|
|
|
repaintScene();
|
2016-08-03 23:00:04 +03:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
m_pScrollbar->setMinimapFrom(0, nullptr);
|
2016-09-09 00:07:27 +03:00
|
|
|
repaintScene();
|
2016-08-03 23:00:04 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
|
2016-08-18 23:26:41 +03:00
|
|
|
void EasyGraphicsView::onSelectedBlockChange(unsigned int _block_index)
|
2016-08-08 22:17:56 +03:00
|
|
|
{
|
|
|
|
if (!m_bUpdatingRect)
|
|
|
|
{
|
2016-08-30 22:51:18 +03:00
|
|
|
if (_block_index < EASY_GLOBALS.gui_blocks.size())
|
2016-08-08 22:17:56 +03:00
|
|
|
{
|
|
|
|
// Scroll to item
|
|
|
|
|
2016-08-30 22:51:18 +03:00
|
|
|
const auto& guiblock = EASY_GLOBALS.gui_blocks[_block_index];
|
2016-08-11 23:43:34 +03:00
|
|
|
const auto thread_item = m_items[guiblock.graphics_item];
|
|
|
|
const auto& item = thread_item->items(guiblock.graphics_item_level)[guiblock.graphics_item_index];
|
2016-08-08 22:17:56 +03:00
|
|
|
|
2016-08-08 22:45:57 +03:00
|
|
|
m_flickerSpeedX = m_flickerSpeedY = 0;
|
2016-08-09 00:15:40 +03:00
|
|
|
|
|
|
|
m_bUpdatingRect = true;
|
2016-08-11 23:43:34 +03:00
|
|
|
verticalScrollBar()->setValue(static_cast<int>(thread_item->levelY(guiblock.graphics_item_level) - m_visibleSceneRect.height() * 0.5));
|
2016-08-08 22:17:56 +03:00
|
|
|
m_pScrollbar->setValue(item.left() + item.width() * 0.5 - m_pScrollbar->sliderHalfWidth());
|
2016-08-09 00:15:40 +03:00
|
|
|
m_bUpdatingRect = false;
|
2016-08-08 22:17:56 +03:00
|
|
|
}
|
|
|
|
|
2016-08-09 00:15:40 +03:00
|
|
|
updateVisibleSceneRect();
|
2016-09-09 00:07:27 +03:00
|
|
|
repaintScene();
|
2016-08-08 22:17:56 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
|
2016-08-18 23:26:41 +03:00
|
|
|
void EasyGraphicsView::onItemsEspandStateChange()
|
|
|
|
{
|
|
|
|
if (!m_bUpdatingRect)
|
|
|
|
{
|
2016-09-09 00:07:27 +03:00
|
|
|
repaintScene();
|
2016-08-18 23:26:41 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
EasyGraphicsViewWidget::EasyGraphicsViewWidget(QWidget* _parent)
|
2016-08-10 22:08:27 +03:00
|
|
|
: QWidget(_parent)
|
2016-09-08 22:42:35 +03:00
|
|
|
, m_scrollbar(new EasyGraphicsScrollbar(this))
|
|
|
|
, m_view(new EasyGraphicsView(this))
|
2016-09-09 00:07:27 +03:00
|
|
|
, m_threadNamesWidget(new EasyThreadNamesWidget(m_view, m_scrollbar->height(), this))
|
2016-07-31 13:13:48 +03:00
|
|
|
{
|
2016-08-09 00:45:45 +03:00
|
|
|
initWidget();
|
|
|
|
}
|
|
|
|
|
2016-08-18 23:26:41 +03:00
|
|
|
void EasyGraphicsViewWidget::initWidget()
|
2016-08-09 00:45:45 +03:00
|
|
|
{
|
|
|
|
auto lay = new QGridLayout(this);
|
2016-08-01 22:21:59 +03:00
|
|
|
lay->setContentsMargins(1, 0, 1, 0);
|
2016-09-09 00:07:27 +03:00
|
|
|
lay->addWidget(m_threadNamesWidget, 0, 0, 2, 1);
|
2016-09-08 22:42:35 +03:00
|
|
|
lay->setSpacing(1);
|
2016-08-10 22:08:27 +03:00
|
|
|
lay->addWidget(m_view, 0, 1);
|
2016-08-01 22:21:59 +03:00
|
|
|
lay->setSpacing(1);
|
2016-08-10 22:08:27 +03:00
|
|
|
lay->addWidget(m_scrollbar, 1, 1);
|
2016-07-31 13:13:48 +03:00
|
|
|
setLayout(lay);
|
2016-08-10 22:08:27 +03:00
|
|
|
|
2016-07-31 13:13:48 +03:00
|
|
|
m_view->setScrollbar(m_scrollbar);
|
|
|
|
}
|
|
|
|
|
2016-08-18 23:26:41 +03:00
|
|
|
EasyGraphicsViewWidget::~EasyGraphicsViewWidget()
|
2016-07-31 13:13:48 +03:00
|
|
|
{
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2016-08-18 23:26:41 +03:00
|
|
|
EasyGraphicsView* EasyGraphicsViewWidget::view()
|
2016-07-31 13:13:48 +03:00
|
|
|
{
|
|
|
|
return m_view;
|
|
|
|
}
|
|
|
|
|
2016-07-31 18:48:41 +03:00
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
2016-08-09 00:45:45 +03:00
|
|
|
|
2016-09-08 22:42:35 +03:00
|
|
|
EasyThreadNameItem::EasyThreadNameItem() : QGraphicsItem(nullptr)
|
2016-08-09 00:45:45 +03:00
|
|
|
{
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2016-09-08 22:42:35 +03:00
|
|
|
EasyThreadNameItem::~EasyThreadNameItem()
|
2016-08-09 00:45:45 +03:00
|
|
|
{
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2016-09-08 22:42:35 +03:00
|
|
|
QRectF EasyThreadNameItem::boundingRect() const
|
2016-08-09 00:45:45 +03:00
|
|
|
{
|
2016-09-08 22:42:35 +03:00
|
|
|
return m_boundingRect;
|
|
|
|
}
|
|
|
|
|
|
|
|
void EasyThreadNameItem::paint(QPainter* _painter, const QStyleOptionGraphicsItem*, QWidget*)
|
|
|
|
{
|
2016-09-11 16:53:34 +03:00
|
|
|
auto const parentView = static_cast<EasyThreadNamesWidget*>(scene()->parent());
|
|
|
|
const auto view = parentView->view();
|
2016-09-08 22:42:35 +03:00
|
|
|
const auto& items = view->getItems();
|
|
|
|
if (items.empty())
|
|
|
|
return;
|
|
|
|
|
|
|
|
const auto visibleSceneRect = view->visibleSceneRect();
|
2016-09-09 00:07:27 +03:00
|
|
|
const auto h = visibleSceneRect.height() + TIMELINE_ROW_SIZE - 2;
|
2016-09-08 22:42:35 +03:00
|
|
|
const auto w = scene()->sceneRect().width();
|
|
|
|
|
2016-09-09 00:07:27 +03:00
|
|
|
static const uint16_t OVERLAP = THREADS_ROW_SPACING >> 1;
|
2016-09-08 22:42:35 +03:00
|
|
|
static const QBrush brushes[2] = {QColor::fromRgb(BACKGROUND_1), QColor::fromRgb(BACKGROUND_2)};
|
|
|
|
int i = -1;
|
|
|
|
|
|
|
|
QRectF rect;
|
|
|
|
|
|
|
|
_painter->resetTransform();
|
|
|
|
|
2016-09-09 00:07:27 +03:00
|
|
|
// Draw thread names
|
|
|
|
_painter->setFont(BG_FONT);
|
2016-09-08 22:42:35 +03:00
|
|
|
for (auto item : items)
|
2016-08-09 00:45:45 +03:00
|
|
|
{
|
2016-09-08 22:42:35 +03:00
|
|
|
++i;
|
|
|
|
|
|
|
|
auto br = item->boundingRect();
|
|
|
|
auto top = item->y() + br.top() - visibleSceneRect.top() - OVERLAP;
|
|
|
|
auto hgt = br.height() + THREADS_ROW_SPACING;
|
|
|
|
auto bottom = top + hgt;
|
|
|
|
|
|
|
|
if (top > h || bottom < 0)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (item->threadId() == EASY_GLOBALS.selected_thread)
|
|
|
|
_painter->setBrush(QBrush(QColor::fromRgb(::profiler_gui::SELECTED_THREAD_BACKGROUND)));
|
|
|
|
else
|
|
|
|
_painter->setBrush(brushes[i & 1]);
|
|
|
|
|
2016-09-09 00:07:27 +03:00
|
|
|
if (top < 0)
|
|
|
|
{
|
|
|
|
hgt += top;
|
|
|
|
top = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
const auto dh = top + hgt - h;
|
|
|
|
if (dh > 0)
|
|
|
|
hgt -= dh;
|
|
|
|
|
2016-09-08 22:42:35 +03:00
|
|
|
rect.setRect(0, top, w, hgt);
|
|
|
|
|
|
|
|
_painter->setPen(Qt::NoPen);
|
|
|
|
_painter->drawRect(rect);
|
|
|
|
|
|
|
|
rect.translate(-5, 0);
|
|
|
|
_painter->setPen(QColor::fromRgb(::profiler::colors::Dark));
|
|
|
|
_painter->drawText(rect, Qt::AlignRight | Qt::AlignVCenter, item->threadName());
|
2016-08-09 00:45:45 +03:00
|
|
|
}
|
2016-09-09 00:07:27 +03:00
|
|
|
|
|
|
|
// Draw separator between thread names area and information area
|
|
|
|
_painter->setPen(Qt::darkGray);
|
|
|
|
_painter->drawLine(QLineF(0, h, w, h));
|
|
|
|
_painter->drawLine(QLineF(0, h + 2, w, h + 2));
|
|
|
|
|
|
|
|
// Draw information
|
|
|
|
_painter->setFont(CHRONOMETER_FONT);
|
2016-09-11 16:53:34 +03:00
|
|
|
QFontMetricsF fm(CHRONOMETER_FONT, parentView);
|
2016-09-09 00:07:27 +03:00
|
|
|
const qreal time1 = view->chronoTime();
|
|
|
|
const qreal time2 = view->chronoTimeAux();
|
|
|
|
|
|
|
|
auto y = h + 2;
|
|
|
|
|
|
|
|
auto drawTimeText = [&rect, &w, &y, &fm, &_painter](qreal time, QRgb color)
|
|
|
|
{
|
|
|
|
if (time > 0)
|
|
|
|
{
|
|
|
|
const QString text = ::profiler_gui::timeStringReal(time); // Displayed text
|
2016-09-11 16:53:34 +03:00
|
|
|
const auto th = fm.height(); // Calculate displayed text height
|
|
|
|
rect.setRect(0, y, w, th);
|
2016-09-09 00:07:27 +03:00
|
|
|
|
|
|
|
_painter->setPen(color);
|
|
|
|
_painter->drawText(rect, Qt::AlignCenter, text);
|
|
|
|
|
2016-09-11 16:53:34 +03:00
|
|
|
y += th;
|
2016-09-09 00:07:27 +03:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
drawTimeText(time1, CHRONOMETER_COLOR.rgb() & 0x00ffffff);
|
|
|
|
drawTimeText(time2, CHRONOMETER_COLOR2.rgb() & 0x00ffffff);
|
2016-09-08 22:42:35 +03:00
|
|
|
}
|
2016-08-09 00:45:45 +03:00
|
|
|
|
2016-09-08 22:42:35 +03:00
|
|
|
void EasyThreadNameItem::setBoundingRect(const QRectF& _rect)
|
|
|
|
{
|
|
|
|
m_boundingRect = _rect;
|
|
|
|
}
|
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
|
2016-09-11 16:53:34 +03:00
|
|
|
EasyThreadNamesWidget::EasyThreadNamesWidget(EasyGraphicsView* _view, int _additionalHeight, QWidget* _parent)
|
|
|
|
: Parent(_parent)
|
2016-09-08 22:42:35 +03:00
|
|
|
, m_view(_view)
|
2016-09-09 00:07:27 +03:00
|
|
|
, m_additionalHeight(_additionalHeight + 1)
|
2016-09-08 22:42:35 +03:00
|
|
|
{
|
2016-09-11 16:53:34 +03:00
|
|
|
setSizePolicy(QSizePolicy::Fixed, QSizePolicy::Ignored);
|
|
|
|
|
2016-09-08 22:42:35 +03:00
|
|
|
setScene(new QGraphicsScene(this));
|
|
|
|
|
|
|
|
setCacheMode(QGraphicsView::CacheNone);
|
|
|
|
setViewportUpdateMode(QGraphicsView::FullViewportUpdate);
|
|
|
|
setHorizontalScrollBarPolicy(Qt::ScrollBarAlwaysOff);
|
|
|
|
setVerticalScrollBarPolicy(Qt::ScrollBarAlwaysOff);
|
|
|
|
setFixedWidth(100);
|
|
|
|
|
|
|
|
connect(&EASY_GLOBALS.events, &::profiler_gui::EasyGlobalSignals::selectedThreadChanged, this, &This::onSelectedThreadChange);
|
|
|
|
connect(m_view, &EasyGraphicsView::treeChanged, this, &This::onTreeChange);
|
2016-09-09 00:07:27 +03:00
|
|
|
connect(m_view, &EasyGraphicsView::sceneUpdated, this, &This::repaintScene);
|
2016-09-08 22:42:35 +03:00
|
|
|
connect(m_view->verticalScrollBar(), &QScrollBar::valueChanged, verticalScrollBar(), &QScrollBar::setValue);
|
2016-09-09 00:07:27 +03:00
|
|
|
connect(m_view->verticalScrollBar(), &QScrollBar::rangeChanged, this, &This::setVerticalScrollbarRange);
|
2016-09-08 22:42:35 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
EasyThreadNamesWidget::~EasyThreadNamesWidget()
|
|
|
|
{
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2016-09-09 00:07:27 +03:00
|
|
|
void EasyThreadNamesWidget::setVerticalScrollbarRange(int _minValue, int _maxValue)
|
|
|
|
{
|
|
|
|
verticalScrollBar()->setRange(_minValue, _maxValue + m_additionalHeight);
|
|
|
|
}
|
|
|
|
|
2016-09-08 22:42:35 +03:00
|
|
|
void EasyThreadNamesWidget::onTreeChange()
|
|
|
|
{
|
2016-09-11 16:53:34 +03:00
|
|
|
const QSignalBlocker b(this);
|
2016-09-08 22:42:35 +03:00
|
|
|
scene()->clear();
|
|
|
|
|
2016-09-11 16:53:34 +03:00
|
|
|
QFontMetricsF fm(BG_FONT, this);
|
2016-09-08 22:42:35 +03:00
|
|
|
qreal maxLength = 0;
|
|
|
|
const auto& graphicsItems = m_view->getItems();
|
|
|
|
for (auto graphicsItem : graphicsItems)
|
2016-09-11 16:53:34 +03:00
|
|
|
maxLength = ::std::max(maxLength, fm.width(graphicsItem->threadName()) * FONT_METRICS_FACTOR);
|
2016-09-08 22:42:35 +03:00
|
|
|
|
|
|
|
auto vbar = verticalScrollBar();
|
|
|
|
auto viewBar = m_view->verticalScrollBar();
|
|
|
|
|
2016-09-09 00:07:27 +03:00
|
|
|
setVerticalScrollbarRange(viewBar->minimum(), viewBar->maximum());
|
2016-09-08 22:42:35 +03:00
|
|
|
vbar->setSingleStep(viewBar->singleStep());
|
|
|
|
vbar->setPageStep(viewBar->pageStep());
|
|
|
|
|
|
|
|
auto r = m_view->sceneRect();
|
2016-09-09 00:07:27 +03:00
|
|
|
setSceneRect(0, r.top(), maxLength, r.height() + m_additionalHeight);
|
2016-09-08 22:42:35 +03:00
|
|
|
|
|
|
|
auto item = new EasyThreadNameItem();
|
|
|
|
item->setPos(0, 0);
|
|
|
|
item->setBoundingRect(sceneRect());
|
|
|
|
scene()->addItem(item);
|
|
|
|
|
|
|
|
setFixedWidth(maxLength);
|
|
|
|
}
|
|
|
|
|
|
|
|
void EasyThreadNamesWidget::onSelectedThreadChange(::profiler::thread_id_t)
|
|
|
|
{
|
|
|
|
scene()->update();
|
|
|
|
}
|
|
|
|
|
2016-09-09 00:07:27 +03:00
|
|
|
void EasyThreadNamesWidget::repaintScene()
|
|
|
|
{
|
|
|
|
scene()->update();
|
|
|
|
}
|
|
|
|
|
2016-09-08 22:42:35 +03:00
|
|
|
void EasyThreadNamesWidget::mousePressEvent(QMouseEvent* _event)
|
|
|
|
{
|
|
|
|
QMouseEvent e(_event->type(), _event->pos() - QPointF(sceneRect().width(), 0), _event->button(), _event->buttons() & ~Qt::RightButton, _event->modifiers());
|
|
|
|
m_view->mousePressEvent(&e);
|
|
|
|
_event->accept();
|
|
|
|
}
|
|
|
|
|
|
|
|
void EasyThreadNamesWidget::mouseDoubleClickEvent(QMouseEvent* _event)
|
|
|
|
{
|
|
|
|
static const auto OVERLAP = THREADS_ROW_SPACING >> 1;
|
|
|
|
|
|
|
|
auto y = mapToScene(_event->pos()).y();
|
2016-08-09 00:45:45 +03:00
|
|
|
const auto& items = m_view->getItems();
|
2016-09-08 22:42:35 +03:00
|
|
|
for (auto item : items)
|
2016-08-09 00:45:45 +03:00
|
|
|
{
|
2016-09-08 22:42:35 +03:00
|
|
|
auto br = item->boundingRect();
|
|
|
|
auto top = item->y() + br.top() - OVERLAP;
|
|
|
|
auto bottom = top + br.height() + OVERLAP;
|
|
|
|
|
|
|
|
if (y < top || y > bottom)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
const auto thread_id = item->threadId();
|
|
|
|
if (thread_id != EASY_GLOBALS.selected_thread)
|
|
|
|
{
|
|
|
|
EASY_GLOBALS.selected_thread = thread_id;
|
|
|
|
emit EASY_GLOBALS.events.selectedThreadChanged(thread_id);
|
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
2016-08-09 00:45:45 +03:00
|
|
|
}
|
|
|
|
|
2016-09-08 22:42:35 +03:00
|
|
|
_event->accept();
|
|
|
|
}
|
|
|
|
|
|
|
|
void EasyThreadNamesWidget::mouseReleaseEvent(QMouseEvent* _event)
|
|
|
|
{
|
|
|
|
QMouseEvent e(_event->type(), _event->pos() - QPointF(sceneRect().width(), 0), _event->button(), _event->buttons() & ~Qt::RightButton, _event->modifiers());
|
|
|
|
m_view->mouseReleaseEvent(&e);
|
|
|
|
_event->accept();
|
2016-08-09 00:45:45 +03:00
|
|
|
}
|
2016-09-08 22:42:35 +03:00
|
|
|
|
|
|
|
void EasyThreadNamesWidget::mouseMoveEvent(QMouseEvent* _event)
|
|
|
|
{
|
|
|
|
QMouseEvent e(_event->type(), _event->pos() - QPointF(sceneRect().width(), 0), _event->button(), _event->buttons() & ~Qt::RightButton, _event->modifiers());
|
|
|
|
m_view->mouseMoveEvent(&e);
|
|
|
|
_event->accept();
|
|
|
|
}
|
|
|
|
|
2016-09-09 00:07:27 +03:00
|
|
|
void EasyThreadNamesWidget::keyPressEvent(QKeyEvent* _event)
|
|
|
|
{
|
|
|
|
m_view->keyPressEvent(_event);
|
|
|
|
}
|
|
|
|
|
|
|
|
void EasyThreadNamesWidget::keyReleaseEvent(QKeyEvent* _event)
|
|
|
|
{
|
|
|
|
m_view->keyReleaseEvent(_event);
|
|
|
|
}
|
|
|
|
|
2016-09-08 22:42:35 +03:00
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
|