mirror of
https://github.com/python-LimeReport/LimeReport.git
synced 2025-01-26 10:51:03 +03:00
1461 lines
54 KiB
C++
1461 lines
54 KiB
C++
|
/************************************************************************
|
||
|
* file name : easy_graphics_item.cpp
|
||
|
* ----------------- :
|
||
|
* creation time : 2016/09/15
|
||
|
* author : Victor Zarubkin
|
||
|
* email : v.s.zarubkin@gmail.com
|
||
|
* ----------------- :
|
||
|
* description : The file contains implementation of EasyGraphicsItem.
|
||
|
* ----------------- :
|
||
|
* change log : * 2016/09/15 Victor Zarubkin: Moved sources from blocks_graphics_view.cpp
|
||
|
* :
|
||
|
* : *
|
||
|
* ----------------- :
|
||
|
* license : Lightweight profiler library for c++
|
||
|
* : Copyright(C) 2016-2017 Sergey Yagovtsev, Victor Zarubkin
|
||
|
* :
|
||
|
* : Licensed under either of
|
||
|
* : * MIT license (LICENSE.MIT or http://opensource.org/licenses/MIT)
|
||
|
* : * Apache License, Version 2.0, (LICENSE.APACHE or http://www.apache.org/licenses/LICENSE-2.0)
|
||
|
* : at your option.
|
||
|
* :
|
||
|
* : The MIT License
|
||
|
* :
|
||
|
* : Permission is hereby granted, free of charge, to any person obtaining a copy
|
||
|
* : of this software and associated documentation files (the "Software"), to deal
|
||
|
* : in the Software without restriction, including without limitation the rights
|
||
|
* : to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies
|
||
|
* : of the Software, and to permit persons to whom the Software is furnished
|
||
|
* : to do so, subject to the following conditions:
|
||
|
* :
|
||
|
* : The above copyright notice and this permission notice shall be included in all
|
||
|
* : copies or substantial portions of the Software.
|
||
|
* :
|
||
|
* : THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED,
|
||
|
* : INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR
|
||
|
* : PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
|
||
|
* : LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,
|
||
|
* : TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE
|
||
|
* : USE OR OTHER DEALINGS IN THE SOFTWARE.
|
||
|
* :
|
||
|
* : The Apache License, Version 2.0 (the "License")
|
||
|
* :
|
||
|
* : You may not use this file except in compliance with the License.
|
||
|
* : You may obtain a copy of the License at
|
||
|
* :
|
||
|
* : http://www.apache.org/licenses/LICENSE-2.0
|
||
|
* :
|
||
|
* : Unless required by applicable law or agreed to in writing, software
|
||
|
* : distributed under the License is distributed on an "AS IS" BASIS,
|
||
|
* : WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||
|
* : See the License for the specific language governing permissions and
|
||
|
* : limitations under the License.
|
||
|
************************************************************************/
|
||
|
|
||
|
#include <QGraphicsScene>
|
||
|
#include <QDebug>
|
||
|
#include <algorithm>
|
||
|
#include "easy_graphics_item.h"
|
||
|
#include "blocks_graphics_view.h"
|
||
|
#include "globals.h"
|
||
|
|
||
|
//////////////////////////////////////////////////////////////////////////
|
||
|
//////////////////////////////////////////////////////////////////////////
|
||
|
|
||
|
enum BlockItemState : int8_t
|
||
|
{
|
||
|
BLOCK_ITEM_DO_PAINT_FIRST = -2,
|
||
|
BLOCK_ITEM_DO_NOT_PAINT = -1,
|
||
|
BLOCK_ITEM_UNCHANGED,
|
||
|
BLOCK_ITEM_DO_PAINT
|
||
|
};
|
||
|
|
||
|
//////////////////////////////////////////////////////////////////////////
|
||
|
|
||
|
EASY_CONSTEXPR int MIN_SYNC_SPACING = 1;
|
||
|
EASY_CONSTEXPR int MIN_SYNC_SIZE = 3;
|
||
|
EASY_CONSTEXPR int EVENT_HEIGHT = 4;
|
||
|
EASY_CONSTEXPR QRgb BORDERS_COLOR = ::profiler::colors::Grey600 & 0x00ffffff;// 0x00686868;
|
||
|
|
||
|
inline QRgb selectedItemBorderColor(::profiler::color_t _color) {
|
||
|
return ::profiler_gui::isLightColor(_color, 192) ? ::profiler::colors::Black : ::profiler::colors::RichRed;
|
||
|
}
|
||
|
|
||
|
const QPen HIGHLIGHTER_PEN = ([]() -> QPen { QPen p(::profiler::colors::Black); p.setStyle(Qt::DotLine); p.setWidth(2); return p; })();
|
||
|
|
||
|
#ifdef max
|
||
|
#undef max
|
||
|
#endif
|
||
|
|
||
|
#ifdef min
|
||
|
#undef min
|
||
|
#endif
|
||
|
|
||
|
//////////////////////////////////////////////////////////////////////////
|
||
|
|
||
|
EasyGraphicsItem::EasyGraphicsItem(uint8_t _index, const::profiler::BlocksTreeRoot& _root)
|
||
|
: QGraphicsItem(nullptr)
|
||
|
, m_threadName(::profiler_gui::decoratedThreadName(EASY_GLOBALS.use_decorated_thread_name, _root, EASY_GLOBALS.hex_thread_id))
|
||
|
, m_pRoot(&_root)
|
||
|
, m_index(_index)
|
||
|
{
|
||
|
}
|
||
|
|
||
|
EasyGraphicsItem::~EasyGraphicsItem()
|
||
|
{
|
||
|
}
|
||
|
|
||
|
void EasyGraphicsItem::validateName()
|
||
|
{
|
||
|
m_threadName = ::profiler_gui::decoratedThreadName(EASY_GLOBALS.use_decorated_thread_name, *m_pRoot, EASY_GLOBALS.hex_thread_id);
|
||
|
}
|
||
|
|
||
|
const EasyGraphicsView* EasyGraphicsItem::view() const
|
||
|
{
|
||
|
return static_cast<const EasyGraphicsView*>(scene()->parent());
|
||
|
}
|
||
|
|
||
|
//////////////////////////////////////////////////////////////////////////
|
||
|
|
||
|
QRectF EasyGraphicsItem::boundingRect() const
|
||
|
{
|
||
|
return m_boundingRect;
|
||
|
}
|
||
|
|
||
|
//////////////////////////////////////////////////////////////////////////
|
||
|
|
||
|
struct EasyPainterInformation EASY_FINAL
|
||
|
{
|
||
|
const QRectF visibleSceneRect;
|
||
|
QRectF rect;
|
||
|
QBrush brush;
|
||
|
const qreal visibleBottom;
|
||
|
const qreal currentScale;
|
||
|
const qreal offset;
|
||
|
const qreal sceneLeft;
|
||
|
const qreal sceneRight;
|
||
|
const qreal dx;
|
||
|
QRgb previousColor;
|
||
|
QRgb textColor;
|
||
|
Qt::PenStyle previousPenStyle;
|
||
|
bool is_light;
|
||
|
bool selectedItemsWasPainted;
|
||
|
|
||
|
explicit EasyPainterInformation(const EasyGraphicsView* sceneView)
|
||
|
: visibleSceneRect(sceneView->visibleSceneRect())
|
||
|
, visibleBottom(visibleSceneRect.bottom() - 1)
|
||
|
, currentScale(sceneView->scale())
|
||
|
, offset(sceneView->offset())
|
||
|
, sceneLeft(offset)
|
||
|
, sceneRight(offset + visibleSceneRect.width() / currentScale)
|
||
|
, dx(offset * currentScale)
|
||
|
, previousColor(0)
|
||
|
, textColor(0)
|
||
|
, previousPenStyle(Qt::NoPen)
|
||
|
, is_light(false)
|
||
|
, selectedItemsWasPainted(false)
|
||
|
{
|
||
|
brush.setStyle(Qt::SolidPattern);
|
||
|
}
|
||
|
|
||
|
EasyPainterInformation() = delete;
|
||
|
};
|
||
|
|
||
|
#ifdef EASY_GRAPHICS_ITEM_RECURSIVE_PAINT
|
||
|
void EasyGraphicsItem::paintChildren(const float _minWidth, const int _narrowSizeHalf, const uint8_t _levelsNumber, QPainter* _painter, struct EasyPainterInformation& p, ::profiler_gui::EasyBlockItem& _item, const ::profiler_gui::EasyBlock& _itemBlock, RightBounds& _rightBounds, uint8_t _level, int8_t _mode)
|
||
|
{
|
||
|
if (_level >= _levelsNumber || _itemBlock.tree.children.empty())
|
||
|
return;
|
||
|
|
||
|
const auto top = levelY(_level);
|
||
|
if (top > p.visibleBottom)
|
||
|
return;
|
||
|
|
||
|
qreal& prevRight = _rightBounds[_level];
|
||
|
auto& level = m_levels[_level];
|
||
|
const short next_level = _level + 1;
|
||
|
|
||
|
uint32_t neighbours = (uint32_t)_itemBlock.tree.children.size();
|
||
|
uint32_t last = neighbours - 1;
|
||
|
uint32_t neighbour = 0;
|
||
|
|
||
|
if (_mode == BLOCK_ITEM_DO_PAINT_FIRST)
|
||
|
{
|
||
|
neighbour = last = _item.max_depth_child;
|
||
|
neighbours = neighbour + 1;
|
||
|
}
|
||
|
|
||
|
for (uint32_t i = _item.children_begin + neighbour; neighbour < neighbours; ++i, ++neighbour)
|
||
|
{
|
||
|
auto& item = level[i];
|
||
|
|
||
|
if (item.left() > p.sceneRight)
|
||
|
break; // This is first totally invisible item. No need to check other items.
|
||
|
|
||
|
if (item.right() < p.sceneLeft)
|
||
|
continue; // This item is not visible
|
||
|
|
||
|
const auto& itemBlock = easyBlock(item.block);
|
||
|
const uint16_t totalHeight = itemBlock.tree.depth * ::profiler_gui::GRAPHICS_ROW_SIZE_FULL + ::profiler_gui::GRAPHICS_ROW_SIZE;
|
||
|
if ((top + totalHeight) < p.visibleSceneRect.top())
|
||
|
continue; // This item is not visible
|
||
|
|
||
|
const auto item_width = ::std::max(item.width(), _minWidth);
|
||
|
auto x = item.left() * p.currentScale - p.dx;
|
||
|
auto w = item_width * p.currentScale;
|
||
|
//const auto right = x + w;
|
||
|
if ((x + w) <= prevRight)
|
||
|
{
|
||
|
// This item is not visible
|
||
|
if (!(EASY_GLOBALS.hide_narrow_children && w < EASY_GLOBALS.blocks_narrow_size))
|
||
|
paintChildren(_minWidth, _narrowSizeHalf, _levelsNumber, _painter, p, item, itemBlock, _rightBounds, next_level, BLOCK_ITEM_DO_PAINT_FIRST);
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
if (x < prevRight)
|
||
|
{
|
||
|
w -= prevRight - x;
|
||
|
x = prevRight;
|
||
|
}
|
||
|
|
||
|
if (EASY_GLOBALS.hide_minsize_blocks && w < EASY_GLOBALS.blocks_size_min)
|
||
|
continue; // Hide blocks (except top-level blocks) which width is less than 1 pixel
|
||
|
|
||
|
const auto& itemDesc = easyDescriptor(itemBlock.tree.node->id());
|
||
|
|
||
|
int h = 0;
|
||
|
bool do_paint_children = false;
|
||
|
if ((EASY_GLOBALS.hide_narrow_children && w < EASY_GLOBALS.blocks_narrow_size) || !itemBlock.expanded)
|
||
|
{
|
||
|
// Items which width is less than 20 will be painted as big rectangles which are hiding it's children
|
||
|
|
||
|
//x = item.left() * p.currentScale - p.dx;
|
||
|
h = totalHeight;
|
||
|
const auto dh = top + h - p.visibleBottom;
|
||
|
if (dh > 0)
|
||
|
h -= dh;
|
||
|
|
||
|
if (item.block == EASY_GLOBALS.selected_block)
|
||
|
p.selectedItemsWasPainted = true;
|
||
|
|
||
|
const bool colorChange = (p.previousColor != itemDesc.color());
|
||
|
if (colorChange)
|
||
|
{
|
||
|
// Set background color brush for rectangle
|
||
|
p.previousColor = itemDesc.color();
|
||
|
//p.inverseColor = 0xffffffff - p.previousColor;
|
||
|
p.is_light = ::profiler_gui::isLightColor(p.previousColor);
|
||
|
p.textColor = ::profiler_gui::textColorForFlag(p.is_light);
|
||
|
p.brush.setColor(QColor::fromRgba(p.previousColor));
|
||
|
_painter->setBrush(p.brush);
|
||
|
}
|
||
|
|
||
|
if (EASY_GLOBALS.highlight_blocks_with_same_id && (EASY_GLOBALS.selected_block_id == itemBlock.tree.node->id()
|
||
|
|| (::profiler_gui::is_max(EASY_GLOBALS.selected_block) && EASY_GLOBALS.selected_block_id == itemDesc.id())))
|
||
|
{
|
||
|
if (p.previousPenStyle != Qt::DotLine)
|
||
|
{
|
||
|
p.previousPenStyle = Qt::DotLine;
|
||
|
_painter->setPen(HIGHLIGHTER_PEN);
|
||
|
}
|
||
|
}
|
||
|
else if (EASY_GLOBALS.draw_graphics_items_borders)
|
||
|
{
|
||
|
if (p.previousPenStyle != Qt::SolidLine)// || colorChange)
|
||
|
{
|
||
|
// Restore pen for item which is wide enough to paint borders
|
||
|
p.previousPenStyle = Qt::SolidLine;
|
||
|
_painter->setPen(BORDERS_COLOR);//BORDERS_COLOR & inverseColor);
|
||
|
}
|
||
|
}
|
||
|
else if (p.previousPenStyle != Qt::NoPen)
|
||
|
{
|
||
|
p.previousPenStyle = Qt::NoPen;
|
||
|
_painter->setPen(Qt::NoPen);
|
||
|
}
|
||
|
|
||
|
const auto wprev = w;
|
||
|
decltype(w) dw = 0;
|
||
|
if (item.left() < p.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() - p.sceneLeft) * p.currentScale;
|
||
|
x = p.sceneLeft * p.currentScale - p.dx - 2;
|
||
|
w += 2;
|
||
|
dw = 2;
|
||
|
}
|
||
|
|
||
|
if (item.right() > p.sceneRight)
|
||
|
{
|
||
|
w -= (item.right() - p.sceneRight) * p.currentScale;
|
||
|
w += 2;
|
||
|
dw += 2;
|
||
|
}
|
||
|
|
||
|
if (w < EASY_GLOBALS.blocks_size_min)
|
||
|
w = EASY_GLOBALS.blocks_size_min;
|
||
|
|
||
|
// Draw rectangle
|
||
|
p.rect.setRect(x, top, w, h);
|
||
|
_painter->drawRect(p.rect);
|
||
|
|
||
|
prevRight = p.rect.right() + EASY_GLOBALS.blocks_spacing;
|
||
|
//skip_children(next_level, item.children_begin);
|
||
|
if (wprev < EASY_GLOBALS.blocks_narrow_size)
|
||
|
continue;
|
||
|
|
||
|
if (dw > 1)
|
||
|
{
|
||
|
w -= dw;
|
||
|
x += 2;
|
||
|
}
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
if (item.block == EASY_GLOBALS.selected_block)
|
||
|
p.selectedItemsWasPainted = true;
|
||
|
|
||
|
const bool colorChange = (p.previousColor != itemDesc.color());
|
||
|
if (colorChange)
|
||
|
{
|
||
|
// Set background color brush for rectangle
|
||
|
p.previousColor = itemDesc.color();
|
||
|
//p.inverseColor = 0xffffffff - p.previousColor;
|
||
|
p.is_light = ::profiler_gui::isLightColor(p.previousColor);
|
||
|
p.textColor = ::profiler_gui::textColorForFlag(p.is_light);
|
||
|
p.brush.setColor(QColor::fromRgba(p.previousColor));
|
||
|
_painter->setBrush(p.brush);
|
||
|
}
|
||
|
|
||
|
if (EASY_GLOBALS.highlight_blocks_with_same_id && (EASY_GLOBALS.selected_block_id == itemBlock.tree.node->id()
|
||
|
|| (::profiler_gui::is_max(EASY_GLOBALS.selected_block) && EASY_GLOBALS.selected_block_id == itemDesc.id())))
|
||
|
{
|
||
|
if (p.previousPenStyle != Qt::DotLine)
|
||
|
{
|
||
|
p.previousPenStyle = Qt::DotLine;
|
||
|
_painter->setPen(HIGHLIGHTER_PEN);
|
||
|
}
|
||
|
}
|
||
|
else if (EASY_GLOBALS.draw_graphics_items_borders)
|
||
|
{
|
||
|
if (p.previousPenStyle != Qt::SolidLine)// || colorChange)
|
||
|
{
|
||
|
// Restore pen for item which is wide enough to paint borders
|
||
|
p.previousPenStyle = Qt::SolidLine;
|
||
|
_painter->setPen(BORDERS_COLOR);// BORDERS_COLOR & inverseColor);
|
||
|
}
|
||
|
}
|
||
|
else if (p.previousPenStyle != Qt::NoPen)
|
||
|
{
|
||
|
p.previousPenStyle = Qt::NoPen;
|
||
|
_painter->setPen(Qt::NoPen);
|
||
|
}
|
||
|
|
||
|
// Draw rectangle
|
||
|
//x = item.left() * currentScale - p.dx;
|
||
|
h = ::profiler_gui::GRAPHICS_ROW_SIZE;
|
||
|
const auto dh = top + h - p.visibleBottom;
|
||
|
if (dh > 0)
|
||
|
h -= dh;
|
||
|
|
||
|
const auto wprev = w;
|
||
|
decltype(w) dw = 0;
|
||
|
if (item.left() < p.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() - p.sceneLeft) * p.currentScale;
|
||
|
x = p.sceneLeft * p.currentScale - p.dx - 2;
|
||
|
w += 2;
|
||
|
dw = 2;
|
||
|
}
|
||
|
|
||
|
if (item.right() > p.sceneRight)
|
||
|
{
|
||
|
w -= (item.right() - p.sceneRight) * p.currentScale;
|
||
|
w += 2;
|
||
|
dw += 2;
|
||
|
}
|
||
|
|
||
|
if (w < EASY_GLOBALS.blocks_size_min)
|
||
|
w = EASY_GLOBALS.blocks_size_min;
|
||
|
|
||
|
p.rect.setRect(x, top, w, h);
|
||
|
_painter->drawRect(p.rect);
|
||
|
|
||
|
prevRight = p.rect.right() + EASY_GLOBALS.blocks_spacing;
|
||
|
if (wprev < EASY_GLOBALS.blocks_narrow_size)
|
||
|
{
|
||
|
paintChildren(_minWidth, _narrowSizeHalf, _levelsNumber, _painter, p, item, itemBlock, _rightBounds, next_level, wprev < _narrowSizeHalf ? BLOCK_ITEM_DO_PAINT_FIRST : BLOCK_ITEM_DO_PAINT);
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
if (dw > 1)
|
||
|
{
|
||
|
w -= dw;
|
||
|
x += 2;
|
||
|
}
|
||
|
|
||
|
do_paint_children = true;
|
||
|
}
|
||
|
|
||
|
// Draw text-----------------------------------
|
||
|
p.rect.setRect(x + 1, top, w - 1, h);
|
||
|
|
||
|
// text will be painted with inverse color
|
||
|
//auto textColor = inverseColor < 0x00808080 ? profiler::colors::Black : profiler::colors::White;
|
||
|
//if (textColor == previousColor) textColor = 0;
|
||
|
_painter->setPen(p.textColor);
|
||
|
|
||
|
if (item.block == EASY_GLOBALS.selected_block)
|
||
|
_painter->setFont(EASY_GLOBALS.selected_item_font);
|
||
|
|
||
|
// drawing text
|
||
|
auto name = easyBlockName(itemBlock.tree, itemDesc);
|
||
|
_painter->drawText(p.rect, Qt::AlignCenter, ::profiler_gui::toUnicode(name));
|
||
|
|
||
|
// restore previous pen color
|
||
|
if (p.previousPenStyle == Qt::NoPen)
|
||
|
_painter->setPen(Qt::NoPen);
|
||
|
else if (p.previousPenStyle == Qt::DotLine)
|
||
|
{
|
||
|
_painter->setPen(HIGHLIGHTER_PEN);
|
||
|
}
|
||
|
else
|
||
|
_painter->setPen(BORDERS_COLOR);// BORDERS_COLOR & inverseColor); // restore pen for rectangle painting
|
||
|
|
||
|
// restore font
|
||
|
if (item.block == EASY_GLOBALS.selected_block)
|
||
|
_painter->setFont(EASY_GLOBALS.items_font);
|
||
|
// END Draw text~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||
|
|
||
|
if (do_paint_children)
|
||
|
paintChildren(_minWidth, _narrowSizeHalf, _levelsNumber, _painter, p, item, itemBlock, _rightBounds, next_level, _mode);
|
||
|
}
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
void EasyGraphicsItem::paint(QPainter* _painter, const QStyleOptionGraphicsItem*, QWidget*)
|
||
|
{
|
||
|
const bool gotItems = !m_levels.empty() && !m_levels.front().empty();
|
||
|
const bool gotSync = !m_pRoot->sync.empty();
|
||
|
|
||
|
if (!gotItems && !gotSync)
|
||
|
{
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
EasyPainterInformation p(view());
|
||
|
|
||
|
_painter->save();
|
||
|
_painter->setFont(EASY_GLOBALS.items_font);
|
||
|
|
||
|
// Reset indices of first visible item for each layer
|
||
|
const auto levelsNumber = levels();
|
||
|
m_rightBounds[0] = -1e100;
|
||
|
for (uint8_t i = 1; i < levelsNumber; ++i) {
|
||
|
::profiler_gui::set_max(m_levelsIndexes[i]);
|
||
|
m_rightBounds[i] = -1e100;
|
||
|
}
|
||
|
|
||
|
|
||
|
// Search for first visible top-level item
|
||
|
if (gotItems)
|
||
|
{
|
||
|
auto& level0 = m_levels.front();
|
||
|
auto first = ::std::lower_bound(level0.begin(), level0.end(), p.sceneLeft, [](const ::profiler_gui::EasyBlockItem& _item, qreal _value)
|
||
|
{
|
||
|
return _item.left() < _value;
|
||
|
});
|
||
|
|
||
|
if (first != level0.end())
|
||
|
{
|
||
|
m_levelsIndexes[0] = first - level0.begin();
|
||
|
if (m_levelsIndexes[0] > 0)
|
||
|
m_levelsIndexes[0] -= 1;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
m_levelsIndexes[0] = static_cast<unsigned int>(level0.size() - 1);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
// 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
|
||
|
// works fine without using this additional shifting)
|
||
|
//const auto dx = p.offset * p.currentScale;
|
||
|
|
||
|
// Shifting coordinates to current screen offset
|
||
|
_painter->setTransform(QTransform::fromTranslate(0, -y()), true);
|
||
|
|
||
|
|
||
|
|
||
|
if (EASY_GLOBALS.draw_graphics_items_borders)
|
||
|
{
|
||
|
p.previousPenStyle = Qt::SolidLine;
|
||
|
_painter->setPen(BORDERS_COLOR);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
_painter->setPen(Qt::NoPen);
|
||
|
}
|
||
|
|
||
|
|
||
|
const auto MIN_WIDTH = EASY_GLOBALS.enable_zero_length ? 0.f : 0.25f;
|
||
|
|
||
|
|
||
|
// Iterate through layers and draw visible items
|
||
|
if (gotItems)
|
||
|
{
|
||
|
const int narrow_size_half = EASY_GLOBALS.blocks_narrow_size >> 1;
|
||
|
|
||
|
#ifndef EASY_GRAPHICS_ITEM_RECURSIVE_PAINT
|
||
|
static const auto MAX_CHILD_INDEX = ::profiler_gui::numeric_max<decltype(::profiler_gui::EasyBlockItem::children_begin)>();
|
||
|
auto const dont_skip_children = [this, &levelsNumber](short next_level, decltype(::profiler_gui::EasyBlockItem::children_begin) children_begin, int8_t _state)
|
||
|
{
|
||
|
if (next_level < levelsNumber && children_begin != MAX_CHILD_INDEX)
|
||
|
{
|
||
|
if (m_levelsIndexes[next_level] == MAX_CHILD_INDEX)
|
||
|
{
|
||
|
// Mark first potentially visible child item on next sublevel
|
||
|
m_levelsIndexes[next_level] = children_begin;
|
||
|
}
|
||
|
|
||
|
// Mark children items that we want to draw them
|
||
|
m_levels[next_level][children_begin].state = _state;
|
||
|
}
|
||
|
};
|
||
|
#endif
|
||
|
|
||
|
//size_t iterations = 0;
|
||
|
#ifndef EASY_GRAPHICS_ITEM_RECURSIVE_PAINT
|
||
|
for (uint8_t l = 0; l < levelsNumber; ++l)
|
||
|
#else
|
||
|
for (uint8_t l = 0; l < 1; ++l)
|
||
|
#endif
|
||
|
{
|
||
|
auto& level = m_levels[l];
|
||
|
const short next_level = l + 1;
|
||
|
|
||
|
const auto top = levelY(l);
|
||
|
if (top > p.visibleBottom)
|
||
|
break;
|
||
|
|
||
|
//qreal& prevRight = m_rightBounds[l];
|
||
|
qreal prevRight = -1e100;
|
||
|
uint32_t neighbour = 0;
|
||
|
for (uint32_t i = m_levelsIndexes[l], end = static_cast<uint32_t>(level.size()); i < end; ++i, ++neighbour)
|
||
|
{
|
||
|
//++iterations;
|
||
|
|
||
|
auto& item = level[i];
|
||
|
|
||
|
if (item.left() > p.sceneRight)
|
||
|
break; // This is first totally invisible item. No need to check other items.
|
||
|
|
||
|
#ifndef EASY_GRAPHICS_ITEM_RECURSIVE_PAINT
|
||
|
char state = BLOCK_ITEM_DO_PAINT;
|
||
|
if (item.state != BLOCK_ITEM_UNCHANGED)
|
||
|
{
|
||
|
neighbour = 0; // first block in parent's children list
|
||
|
state = item.state;
|
||
|
item.state = BLOCK_ITEM_DO_NOT_PAINT;
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
if (item.right() < p.sceneLeft)
|
||
|
continue; // This item is not visible
|
||
|
|
||
|
#ifndef EASY_GRAPHICS_ITEM_RECURSIVE_PAINT
|
||
|
if (state == BLOCK_ITEM_DO_NOT_PAINT)
|
||
|
{
|
||
|
// This item is not visible
|
||
|
if (neighbour < item.neighbours)
|
||
|
i += item.neighbours - neighbour - 1; // Skip all neighbours
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
if (state == BLOCK_ITEM_DO_PAINT_FIRST && item.children_begin == MAX_CHILD_INDEX && next_level < levelsNumber && neighbour < (item.neighbours-1))
|
||
|
// Paint only first child which has own children
|
||
|
continue; // This item has no children and would not be painted
|
||
|
#endif
|
||
|
|
||
|
const auto& itemBlock = easyBlock(item.block);
|
||
|
const uint16_t totalHeight = itemBlock.tree.depth * ::profiler_gui::GRAPHICS_ROW_SIZE_FULL + ::profiler_gui::GRAPHICS_ROW_SIZE;
|
||
|
if ((top + totalHeight) < p.visibleSceneRect.top())
|
||
|
continue; // This item is not visible
|
||
|
|
||
|
const auto item_width = ::std::max(item.width(), MIN_WIDTH);
|
||
|
auto x = item.left() * p.currentScale - p.dx;
|
||
|
auto w = item_width * p.currentScale;
|
||
|
if ((x + w) <= prevRight)
|
||
|
{
|
||
|
// This item is not visible
|
||
|
#ifdef EASY_GRAPHICS_ITEM_RECURSIVE_PAINT
|
||
|
if (!EASY_GLOBALS.hide_narrow_children || w >= EASY_GLOBALS.blocks_narrow_size)
|
||
|
paintChildren(MIN_WIDTH, narrow_size_half, levelsNumber, _painter, p, item, itemBlock, m_rightBounds, next_level, BLOCK_ITEM_DO_PAINT_FIRST);
|
||
|
#else
|
||
|
if (!(EASY_GLOBALS.hide_narrow_children && w < EASY_GLOBALS.blocks_narrow_size) && l > 0)
|
||
|
dont_skip_children(next_level, item.children_begin, BLOCK_ITEM_DO_PAINT_FIRST);
|
||
|
#endif
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
if (x < prevRight)
|
||
|
{
|
||
|
w -= prevRight - x;
|
||
|
x = prevRight;
|
||
|
}
|
||
|
|
||
|
#ifndef EASY_GRAPHICS_ITEM_RECURSIVE_PAINT
|
||
|
if (EASY_GLOBALS.hide_minsize_blocks && w < EASY_GLOBALS.blocks_size_min && l > 0)
|
||
|
continue; // Hide blocks (except top-level blocks) which width is less than 1 pixel
|
||
|
|
||
|
if (state == BLOCK_ITEM_DO_PAINT_FIRST && neighbour < item.neighbours)
|
||
|
{
|
||
|
// Paint only first child which has own children
|
||
|
i += item.neighbours - neighbour - 1; // Skip all neighbours
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
const auto& itemDesc = easyDescriptor(itemBlock.tree.node->id());
|
||
|
int h = 0;
|
||
|
|
||
|
#ifdef EASY_GRAPHICS_ITEM_RECURSIVE_PAINT
|
||
|
bool do_paint_children = false;
|
||
|
#endif
|
||
|
|
||
|
if ((EASY_GLOBALS.hide_narrow_children && w < EASY_GLOBALS.blocks_narrow_size) || !itemBlock.expanded)
|
||
|
{
|
||
|
// Items which width is less than 20 will be painted as big rectangles which are hiding it's children
|
||
|
|
||
|
//x = item.left() * p.currentScale - p.dx;
|
||
|
h = totalHeight;
|
||
|
const auto dh = top + h - p.visibleBottom;
|
||
|
if (dh > 0)
|
||
|
h -= dh;
|
||
|
|
||
|
if (item.block == EASY_GLOBALS.selected_block)
|
||
|
p.selectedItemsWasPainted = true;
|
||
|
|
||
|
const bool colorChange = (p.previousColor != itemDesc.color());
|
||
|
if (colorChange)
|
||
|
{
|
||
|
// Set background color brush for rectangle
|
||
|
p.previousColor = itemDesc.color();
|
||
|
//p.inverseColor = 0xffffffff - p.previousColor;
|
||
|
p.is_light = ::profiler_gui::isLightColor(p.previousColor);
|
||
|
p.textColor = ::profiler_gui::textColorForFlag(p.is_light);
|
||
|
p.brush.setColor(QColor::fromRgba(p.previousColor));
|
||
|
_painter->setBrush(p.brush);
|
||
|
}
|
||
|
|
||
|
if (EASY_GLOBALS.highlight_blocks_with_same_id && (EASY_GLOBALS.selected_block_id == itemBlock.tree.node->id()
|
||
|
|| (::profiler_gui::is_max(EASY_GLOBALS.selected_block) && EASY_GLOBALS.selected_block_id == itemDesc.id())))
|
||
|
{
|
||
|
if (p.previousPenStyle != Qt::DotLine)
|
||
|
{
|
||
|
p.previousPenStyle = Qt::DotLine;
|
||
|
_painter->setPen(HIGHLIGHTER_PEN);
|
||
|
}
|
||
|
}
|
||
|
else if (EASY_GLOBALS.draw_graphics_items_borders)
|
||
|
{
|
||
|
if (p.previousPenStyle != Qt::SolidLine)// || colorChange)
|
||
|
{
|
||
|
// Restore pen for item which is wide enough to paint borders
|
||
|
p.previousPenStyle = Qt::SolidLine;
|
||
|
_painter->setPen(BORDERS_COLOR);//BORDERS_COLOR & inverseColor);
|
||
|
}
|
||
|
}
|
||
|
else if (p.previousPenStyle != Qt::NoPen)
|
||
|
{
|
||
|
p.previousPenStyle = Qt::NoPen;
|
||
|
_painter->setPen(Qt::NoPen);
|
||
|
}
|
||
|
|
||
|
const auto wprev = w;
|
||
|
decltype(w) dw = 0;
|
||
|
if (item.left() < p.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() - p.sceneLeft) * p.currentScale;
|
||
|
x = p.sceneLeft * p.currentScale - p.dx - 2;
|
||
|
w += 2;
|
||
|
dw = 2;
|
||
|
}
|
||
|
|
||
|
if (item.right() > p.sceneRight)
|
||
|
{
|
||
|
w -= (item.right() - p.sceneRight) * p.currentScale;
|
||
|
w += 2;
|
||
|
dw += 2;
|
||
|
}
|
||
|
|
||
|
if (w < EASY_GLOBALS.blocks_size_min)
|
||
|
w = EASY_GLOBALS.blocks_size_min;
|
||
|
|
||
|
// Draw rectangle
|
||
|
p.rect.setRect(x, top, w, h);
|
||
|
_painter->drawRect(p.rect);
|
||
|
|
||
|
prevRight = p.rect.right() + EASY_GLOBALS.blocks_spacing;
|
||
|
//skip_children(next_level, item.children_begin);
|
||
|
if (wprev < EASY_GLOBALS.blocks_narrow_size)
|
||
|
continue;
|
||
|
|
||
|
if (dw > 1) {
|
||
|
w -= dw;
|
||
|
x += 2;
|
||
|
}
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
if (item.block == EASY_GLOBALS.selected_block)
|
||
|
p.selectedItemsWasPainted = true;
|
||
|
|
||
|
const bool colorChange = (p.previousColor != itemDesc.color());
|
||
|
if (colorChange)
|
||
|
{
|
||
|
// Set background color brush for rectangle
|
||
|
p.previousColor = itemDesc.color();
|
||
|
//p.inverseColor = 0xffffffff - p.previousColor;
|
||
|
p.is_light = ::profiler_gui::isLightColor(p.previousColor);
|
||
|
p.textColor = ::profiler_gui::textColorForFlag(p.is_light);
|
||
|
p.brush.setColor(QColor::fromRgba(p.previousColor));
|
||
|
_painter->setBrush(p.brush);
|
||
|
}
|
||
|
|
||
|
if (EASY_GLOBALS.highlight_blocks_with_same_id && (EASY_GLOBALS.selected_block_id == itemBlock.tree.node->id()
|
||
|
|| (::profiler_gui::is_max(EASY_GLOBALS.selected_block) && EASY_GLOBALS.selected_block_id == itemDesc.id())))
|
||
|
{
|
||
|
if (p.previousPenStyle != Qt::DotLine)
|
||
|
{
|
||
|
p.previousPenStyle = Qt::DotLine;
|
||
|
_painter->setPen(HIGHLIGHTER_PEN);
|
||
|
}
|
||
|
}
|
||
|
else if (EASY_GLOBALS.draw_graphics_items_borders)
|
||
|
{
|
||
|
if (p.previousPenStyle != Qt::SolidLine)// || colorChange)
|
||
|
{
|
||
|
// Restore pen for item which is wide enough to paint borders
|
||
|
p.previousPenStyle = Qt::SolidLine;
|
||
|
_painter->setPen(BORDERS_COLOR);// BORDERS_COLOR & inverseColor);
|
||
|
}
|
||
|
}
|
||
|
else if (p.previousPenStyle != Qt::NoPen)
|
||
|
{
|
||
|
p.previousPenStyle = Qt::NoPen;
|
||
|
_painter->setPen(Qt::NoPen);
|
||
|
}
|
||
|
|
||
|
// Draw rectangle
|
||
|
//x = item.left() * currentScale - p.dx;
|
||
|
h = ::profiler_gui::GRAPHICS_ROW_SIZE;
|
||
|
const auto dh = top + h - p.visibleBottom;
|
||
|
if (dh > 0)
|
||
|
h -= dh;
|
||
|
|
||
|
const auto wprev = w;
|
||
|
decltype(w) dw = 0;
|
||
|
if (item.left() < p.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() - p.sceneLeft) * p.currentScale;
|
||
|
x = p.sceneLeft * p.currentScale - p.dx - 2;
|
||
|
w += 2;
|
||
|
dw = 2;
|
||
|
}
|
||
|
|
||
|
if (item.right() > p.sceneRight)
|
||
|
{
|
||
|
w -= (item.right() - p.sceneRight) * p.currentScale;
|
||
|
w += 2;
|
||
|
dw += 2;
|
||
|
}
|
||
|
|
||
|
if (w < EASY_GLOBALS.blocks_size_min)
|
||
|
w = EASY_GLOBALS.blocks_size_min;
|
||
|
|
||
|
p.rect.setRect(x, top, w, h);
|
||
|
_painter->drawRect(p.rect);
|
||
|
|
||
|
prevRight = p.rect.right() + EASY_GLOBALS.blocks_spacing;
|
||
|
if (wprev < EASY_GLOBALS.blocks_narrow_size)
|
||
|
{
|
||
|
#ifndef EASY_GRAPHICS_ITEM_RECURSIVE_PAINT
|
||
|
dont_skip_children(next_level, item.children_begin, wprev < narrow_size_half ? BLOCK_ITEM_DO_PAINT_FIRST : BLOCK_ITEM_DO_PAINT);
|
||
|
#else
|
||
|
paintChildren(MIN_WIDTH, narrow_size_half, levelsNumber, _painter, p, item, itemBlock, m_rightBounds, next_level, wprev < narrow_size_half ? BLOCK_ITEM_DO_PAINT_FIRST : BLOCK_ITEM_DO_PAINT);
|
||
|
#endif
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
#ifndef EASY_GRAPHICS_ITEM_RECURSIVE_PAINT
|
||
|
dont_skip_children(next_level, item.children_begin, BLOCK_ITEM_DO_PAINT);
|
||
|
#endif
|
||
|
|
||
|
if (dw > 1) {
|
||
|
w -= dw;
|
||
|
x += 2;
|
||
|
}
|
||
|
|
||
|
#ifdef EASY_GRAPHICS_ITEM_RECURSIVE_PAINT
|
||
|
do_paint_children = true;
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
// Draw text-----------------------------------
|
||
|
p.rect.setRect(x + 1, top, w - 1, h);
|
||
|
|
||
|
// text will be painted with inverse color
|
||
|
//auto textColor = inverseColor < 0x00808080 ? profiler::colors::Black : profiler::colors::White;
|
||
|
//if (textColor == previousColor) textColor = 0;
|
||
|
_painter->setPen(p.textColor);
|
||
|
|
||
|
if (item.block == EASY_GLOBALS.selected_block)
|
||
|
_painter->setFont(EASY_GLOBALS.selected_item_font);
|
||
|
|
||
|
// drawing text
|
||
|
auto name = easyBlockName(itemBlock.tree, itemDesc);
|
||
|
_painter->drawText(p.rect, Qt::AlignCenter, ::profiler_gui::toUnicode(name));
|
||
|
|
||
|
// restore previous pen color
|
||
|
if (p.previousPenStyle == Qt::NoPen)
|
||
|
_painter->setPen(Qt::NoPen);
|
||
|
else if (p.previousPenStyle == Qt::DotLine)
|
||
|
{
|
||
|
_painter->setPen(HIGHLIGHTER_PEN);
|
||
|
}
|
||
|
else
|
||
|
_painter->setPen(BORDERS_COLOR);// BORDERS_COLOR & inverseColor); // restore pen for rectangle painting
|
||
|
|
||
|
// restore font
|
||
|
if (item.block == EASY_GLOBALS.selected_block)
|
||
|
_painter->setFont(EASY_GLOBALS.items_font);
|
||
|
// END Draw text~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||
|
|
||
|
#ifdef EASY_GRAPHICS_ITEM_RECURSIVE_PAINT
|
||
|
if (do_paint_children)
|
||
|
paintChildren(MIN_WIDTH, narrow_size_half, levelsNumber, _painter, p, item, itemBlock, m_rightBounds, next_level, BLOCK_ITEM_DO_PAINT);
|
||
|
#endif
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if (EASY_GLOBALS.selected_block < EASY_GLOBALS.gui_blocks.size())
|
||
|
{
|
||
|
const auto& guiblock = EASY_GLOBALS.gui_blocks[EASY_GLOBALS.selected_block];
|
||
|
if (guiblock.graphics_item == m_index)
|
||
|
{
|
||
|
const auto& item = m_levels[guiblock.graphics_item_level][guiblock.graphics_item_index];
|
||
|
if (item.left() < p.sceneRight && item.right() > p.sceneLeft)
|
||
|
{
|
||
|
const auto& itemBlock = easyBlock(item.block);
|
||
|
const auto item_width = ::std::max(item.width(), MIN_WIDTH);
|
||
|
auto top = levelY(guiblock.graphics_item_level);
|
||
|
auto w = ::std::max(item_width * p.currentScale, 1.0);
|
||
|
decltype(top) h = (!itemBlock.expanded ||
|
||
|
(w < EASY_GLOBALS.blocks_narrow_size && EASY_GLOBALS.hide_narrow_children))
|
||
|
? (itemBlock.tree.depth * ::profiler_gui::GRAPHICS_ROW_SIZE_FULL + ::profiler_gui::GRAPHICS_ROW_SIZE)
|
||
|
: ::profiler_gui::GRAPHICS_ROW_SIZE;
|
||
|
|
||
|
auto dh = top + h - p.visibleBottom;
|
||
|
if (dh < h)
|
||
|
{
|
||
|
if (dh > 0)
|
||
|
h -= dh;
|
||
|
|
||
|
const auto& itemDesc = easyDescriptor(itemBlock.tree.node->id());
|
||
|
|
||
|
QPen pen(Qt::SolidLine);
|
||
|
pen.setJoinStyle(Qt::MiterJoin);
|
||
|
pen.setColor(selectedItemBorderColor(itemDesc.color()));//Qt::red);
|
||
|
pen.setWidth(3);
|
||
|
_painter->setPen(pen);
|
||
|
|
||
|
if (!p.selectedItemsWasPainted)
|
||
|
{
|
||
|
p.brush.setColor(QColor::fromRgba(itemDesc.color()));// SELECTED_ITEM_COLOR);
|
||
|
_painter->setBrush(p.brush);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
_painter->setBrush(Qt::NoBrush);
|
||
|
}
|
||
|
|
||
|
auto x = item.left() * p.currentScale - p.dx;
|
||
|
decltype(w) dw = 0;
|
||
|
if (item.left() < p.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() - p.sceneLeft) * p.currentScale;
|
||
|
x = p.sceneLeft * p.currentScale - p.dx - 2;
|
||
|
w += 2;
|
||
|
dw = 2;
|
||
|
}
|
||
|
|
||
|
if (item.right() > p.sceneRight)
|
||
|
{
|
||
|
w -= (item.right() - p.sceneRight) * p.currentScale;
|
||
|
w += 2;
|
||
|
dw += 2;
|
||
|
}
|
||
|
|
||
|
p.rect.setRect(x, top, w, h);
|
||
|
_painter->drawRect(p.rect);
|
||
|
|
||
|
if (!p.selectedItemsWasPainted && w > EASY_GLOBALS.blocks_narrow_size)
|
||
|
{
|
||
|
if (dw > 1) {
|
||
|
w -= dw;
|
||
|
x += 2;
|
||
|
}
|
||
|
|
||
|
// Draw text-----------------------------------
|
||
|
p.rect.setRect(x + 1, top, w - 1, h);
|
||
|
|
||
|
// text will be painted with inverse color
|
||
|
//auto textColor = 0x00ffffff - previousColor;
|
||
|
//if (textColor == previousColor) textColor = 0;
|
||
|
p.textColor = ::profiler_gui::textColorForRgb(itemDesc.color());// SELECTED_ITEM_COLOR);
|
||
|
_painter->setPen(p.textColor);
|
||
|
|
||
|
_painter->setFont(EASY_GLOBALS.selected_item_font);
|
||
|
|
||
|
// drawing text
|
||
|
auto name = easyBlockName(itemBlock.tree, itemDesc);
|
||
|
_painter->drawText(p.rect, Qt::AlignCenter, ::profiler_gui::toUnicode(name));
|
||
|
// END Draw text~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
//printf("%u: %llu\n", m_index, iterations);
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
if (gotSync)
|
||
|
{
|
||
|
const auto sceneView = view();
|
||
|
auto firstSync = ::std::lower_bound(m_pRoot->sync.begin(), m_pRoot->sync.end(), p.sceneLeft, [&sceneView](::profiler::block_index_t _index, qreal _value)
|
||
|
{
|
||
|
return sceneView->time2position(easyBlocksTree(_index).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();
|
||
|
|
||
|
p.previousColor = 0;
|
||
|
qreal prevRight = -1e100;
|
||
|
const qreal top = y() + 1 - EVENT_HEIGHT;
|
||
|
if (top + EVENT_HEIGHT < p.visibleBottom)
|
||
|
{
|
||
|
_painter->setPen(BORDERS_COLOR);
|
||
|
|
||
|
for (auto it = firstSync, end = m_pRoot->sync.end(); it != end; ++it)
|
||
|
{
|
||
|
const auto& item = easyBlocksTree(*it);
|
||
|
auto left = sceneView->time2position(item.node->begin());
|
||
|
|
||
|
if (left > p.sceneRight)
|
||
|
break; // This is first totally invisible item. No need to check other items.
|
||
|
|
||
|
decltype(left) width = sceneView->time2position(item.node->end()) - left;
|
||
|
if (left + width < p.sceneLeft) // This item is not visible
|
||
|
continue;
|
||
|
|
||
|
left *= p.currentScale;
|
||
|
left -= p.dx;
|
||
|
width *= p.currentScale;
|
||
|
if (left + width <= prevRight) // This item is not visible
|
||
|
continue;
|
||
|
|
||
|
if (left < prevRight)
|
||
|
{
|
||
|
width -= prevRight - left;
|
||
|
left = prevRight;
|
||
|
}
|
||
|
|
||
|
if (width < MIN_SYNC_SIZE)
|
||
|
width = MIN_SYNC_SIZE;
|
||
|
|
||
|
const ::profiler::thread_id_t tid = EASY_GLOBALS.version < ::profiler_gui::V130 ? item.node->id() : item.cs->tid();
|
||
|
const bool self_thread = tid != 0 && EASY_GLOBALS.profiler_blocks.find(tid) != EASY_GLOBALS.profiler_blocks.end();
|
||
|
|
||
|
::profiler::color_t color = 0;
|
||
|
if (self_thread)
|
||
|
color = ::profiler::colors::Coral;
|
||
|
else if (item.node->id() == 0)
|
||
|
color = ::profiler::colors::Black;
|
||
|
else
|
||
|
color = ::profiler::colors::RedA400;
|
||
|
|
||
|
if (p.previousColor != color)
|
||
|
{
|
||
|
p.previousColor = color;
|
||
|
_painter->setBrush(QColor::fromRgb(color));
|
||
|
}
|
||
|
|
||
|
p.rect.setRect(left, top, width, EVENT_HEIGHT);
|
||
|
_painter->drawRect(p.rect);
|
||
|
prevRight = left + width + MIN_SYNC_SPACING;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
if (EASY_GLOBALS.enable_event_markers && !m_pRoot->events.empty())
|
||
|
{
|
||
|
const auto sceneView = view();
|
||
|
auto first = ::std::lower_bound(m_pRoot->events.begin(), m_pRoot->events.end(), p.offset, [&sceneView](::profiler::block_index_t _index, qreal _value)
|
||
|
{
|
||
|
return sceneView->time2position(easyBlocksTree(_index).node->begin()) < _value;
|
||
|
});
|
||
|
|
||
|
if (first != m_pRoot->events.end())
|
||
|
{
|
||
|
if (first != m_pRoot->events.begin())
|
||
|
--first;
|
||
|
}
|
||
|
else if (!m_pRoot->events.empty())
|
||
|
{
|
||
|
first = m_pRoot->events.begin() + m_pRoot->events.size() - 1;
|
||
|
}
|
||
|
|
||
|
p.previousColor = 0;
|
||
|
qreal prevRight = -1e100;
|
||
|
const qreal top = y() + boundingRect().height() - 1;
|
||
|
if (top + EVENT_HEIGHT < p.visibleBottom)
|
||
|
{
|
||
|
_painter->setPen(BORDERS_COLOR);
|
||
|
|
||
|
for (auto it = first, end = m_pRoot->events.end(); it != end; ++it)
|
||
|
{
|
||
|
const auto& item = easyBlocksTree(*it);
|
||
|
auto left = sceneView->time2position(item.node->begin());
|
||
|
|
||
|
if (left > p.sceneRight)
|
||
|
break; // This is first totally invisible item. No need to check other items.
|
||
|
|
||
|
decltype(left) width = MIN_WIDTH;
|
||
|
if (left + width < p.sceneLeft) // This item is not visible
|
||
|
continue;
|
||
|
|
||
|
left *= p.currentScale;
|
||
|
left -= p.dx;
|
||
|
width *= p.currentScale;
|
||
|
if (width < 2)
|
||
|
width = 2;
|
||
|
|
||
|
if (left + width <= prevRight) // This item is not visible
|
||
|
continue;
|
||
|
|
||
|
if (left < prevRight)
|
||
|
{
|
||
|
width -= prevRight - left;
|
||
|
left = prevRight;
|
||
|
}
|
||
|
|
||
|
if (width < 2)
|
||
|
width = 2;
|
||
|
|
||
|
::profiler::color_t color = easyDescriptor(item.node->id()).color();
|
||
|
if (p.previousColor != color)
|
||
|
{
|
||
|
p.previousColor = color;
|
||
|
_painter->setBrush(QColor::fromRgb(color));
|
||
|
}
|
||
|
|
||
|
p.rect.setRect(left, top, width, EVENT_HEIGHT);
|
||
|
_painter->drawRect(p.rect);
|
||
|
prevRight = left + width + 2;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
_painter->restore();
|
||
|
}
|
||
|
|
||
|
//////////////////////////////////////////////////////////////////////////
|
||
|
|
||
|
const ::profiler::BlocksTreeRoot* EasyGraphicsItem::root() const
|
||
|
{
|
||
|
return m_pRoot;
|
||
|
}
|
||
|
|
||
|
const QString& EasyGraphicsItem::threadName() const
|
||
|
{
|
||
|
return m_threadName;
|
||
|
}
|
||
|
|
||
|
//////////////////////////////////////////////////////////////////////////
|
||
|
|
||
|
QRect EasyGraphicsItem::getRect() const
|
||
|
{
|
||
|
return view()->mapFromScene(m_boundingRect).boundingRect();
|
||
|
}
|
||
|
|
||
|
//////////////////////////////////////////////////////////////////////////
|
||
|
|
||
|
void EasyGraphicsItem::getBlocks(qreal _left, qreal _right, ::profiler_gui::TreeBlocks& _blocks) const
|
||
|
{
|
||
|
// Search for first visible top-level item
|
||
|
auto& level0 = m_levels.front();
|
||
|
auto first = ::std::lower_bound(level0.begin(), level0.end(), _left, [](const ::profiler_gui::EasyBlockItem& _item, qreal _value)
|
||
|
{
|
||
|
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;
|
||
|
}
|
||
|
|
||
|
// Add all visible top-level items into array of visible blocks
|
||
|
for (size_t i = itemIndex, end = level0.size(); i < end; ++i)
|
||
|
{
|
||
|
const auto& item = level0[i];
|
||
|
|
||
|
if (item.left() > _right)
|
||
|
{
|
||
|
// First invisible item. No need to check other items.
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
if (item.right() < _left)
|
||
|
{
|
||
|
// This item is not visible yet
|
||
|
// This is just to be sure
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
_blocks.emplace_back(m_pRoot, item.block);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
//////////////////////////////////////////////////////////////////////////
|
||
|
|
||
|
const ::profiler_gui::EasyBlock* EasyGraphicsItem::intersect(const QPointF& _pos, ::profiler::block_index_t& _blockIndex) const
|
||
|
{
|
||
|
if (m_levels.empty() || m_levels.front().empty())
|
||
|
{
|
||
|
return nullptr;
|
||
|
}
|
||
|
|
||
|
const auto& level0 = m_levels.front();
|
||
|
const auto top = y();
|
||
|
|
||
|
if (top > _pos.y())
|
||
|
{
|
||
|
return nullptr;
|
||
|
}
|
||
|
|
||
|
EASY_STATIC_CONSTEXPR auto OVERLAP = (::profiler_gui::THREADS_ROW_SPACING >> 1) + 2;
|
||
|
const auto bottom = top + m_levels.size() * ::profiler_gui::GRAPHICS_ROW_SIZE_FULL + OVERLAP;
|
||
|
if (bottom < _pos.y())
|
||
|
{
|
||
|
return nullptr;
|
||
|
}
|
||
|
|
||
|
const unsigned int levelIndex = static_cast<unsigned int>(_pos.y() - top) / ::profiler_gui::GRAPHICS_ROW_SIZE_FULL;
|
||
|
if (levelIndex >= m_levels.size())
|
||
|
{
|
||
|
// The Y position is out of blocks range
|
||
|
|
||
|
if (EASY_GLOBALS.enable_event_markers && !m_pRoot->events.empty())
|
||
|
{
|
||
|
// If event indicators are enabled then try to intersect with one of event indicators
|
||
|
|
||
|
const auto& sceneView = view();
|
||
|
auto first = ::std::lower_bound(m_pRoot->events.begin(), m_pRoot->events.end(), _pos.x(), [&sceneView](::profiler::block_index_t _index, qreal _value)
|
||
|
{
|
||
|
return sceneView->time2position(easyBlocksTree(_index).node->begin()) < _value;
|
||
|
});
|
||
|
|
||
|
if (first != m_pRoot->events.end())
|
||
|
{
|
||
|
if (first != m_pRoot->events.begin())
|
||
|
--first;
|
||
|
}
|
||
|
else if (!m_pRoot->events.empty())
|
||
|
{
|
||
|
first = m_pRoot->events.begin() + m_pRoot->events.size() - 1;
|
||
|
}
|
||
|
|
||
|
const auto MIN_WIDTH = EASY_GLOBALS.enable_zero_length ? 0.f : 0.25f;
|
||
|
const auto currentScale = sceneView->scale();
|
||
|
const auto dw = 5. / currentScale;
|
||
|
|
||
|
for (auto it = first, end = m_pRoot->events.end(); it != end; ++it)
|
||
|
{
|
||
|
_blockIndex = *it;
|
||
|
const auto& item = easyBlock(_blockIndex);
|
||
|
auto left = sceneView->time2position(item.tree.node->begin());
|
||
|
|
||
|
if (left - dw > _pos.x())
|
||
|
break; // This is first totally invisible item. No need to check other items.
|
||
|
|
||
|
decltype(left) width = MIN_WIDTH;
|
||
|
if (left + width + dw < _pos.x()) // This item is not visible
|
||
|
continue;
|
||
|
|
||
|
return &item;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return nullptr;
|
||
|
}
|
||
|
|
||
|
// The Y position is inside blocks range
|
||
|
|
||
|
const auto MIN_WIDTH = EASY_GLOBALS.enable_zero_length ? 0.f : 0.25f;
|
||
|
|
||
|
const auto currentScale = view()->scale();
|
||
|
const auto dw = 5. / currentScale;
|
||
|
unsigned int i = 0;
|
||
|
size_t itemIndex = ::std::numeric_limits<size_t>::max();
|
||
|
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
|
||
|
auto first = ::std::lower_bound(level.begin() + firstItem, level.begin() + lastItem, _pos.x(), [](const ::profiler_gui::EasyBlockItem& _item, qreal _value)
|
||
|
{
|
||
|
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];
|
||
|
static const auto MAX_CHILD_INDEX = ::profiler_gui::numeric_max(item.children_begin);
|
||
|
|
||
|
if (item.left() - dw > _pos.x())
|
||
|
{
|
||
|
return nullptr;
|
||
|
}
|
||
|
|
||
|
const auto item_width = ::std::max(item.width(), MIN_WIDTH);
|
||
|
if (item.left() + item_width + dw < _pos.x())
|
||
|
{
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
const auto w = item_width * currentScale;
|
||
|
const auto& guiItem = easyBlock(item.block);
|
||
|
if (i == levelIndex || (w < EASY_GLOBALS.blocks_narrow_size && EASY_GLOBALS.hide_narrow_children) || !guiItem.expanded)
|
||
|
{
|
||
|
_blockIndex = item.block;
|
||
|
return &guiItem;
|
||
|
}
|
||
|
|
||
|
if (item.children_begin == MAX_CHILD_INDEX)
|
||
|
{
|
||
|
if (itemIndex != 0)
|
||
|
{
|
||
|
auto j = itemIndex;
|
||
|
firstItem = 0;
|
||
|
do {
|
||
|
|
||
|
--j;
|
||
|
const auto& item2 = level[j];
|
||
|
if (item2.children_begin != MAX_CHILD_INDEX)
|
||
|
{
|
||
|
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];
|
||
|
if (item2.children_begin != MAX_CHILD_INDEX)
|
||
|
{
|
||
|
lastItem = item2.children_begin;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
++i;
|
||
|
}
|
||
|
|
||
|
return nullptr;
|
||
|
}
|
||
|
|
||
|
const ::profiler_gui::EasyBlock* EasyGraphicsItem::intersectEvent(const QPointF& _pos) const
|
||
|
{
|
||
|
if (m_pRoot->sync.empty())
|
||
|
{
|
||
|
return nullptr;
|
||
|
}
|
||
|
|
||
|
const auto top = y() - EVENT_HEIGHT;
|
||
|
if (top > _pos.y())
|
||
|
{
|
||
|
return nullptr;
|
||
|
}
|
||
|
|
||
|
const auto bottom = top + EVENT_HEIGHT + 2;
|
||
|
if (bottom < _pos.y())
|
||
|
{
|
||
|
return nullptr;
|
||
|
}
|
||
|
|
||
|
const auto sceneView = view();
|
||
|
auto firstSync = ::std::lower_bound(m_pRoot->sync.begin(), m_pRoot->sync.end(), _pos.x(), [&sceneView](::profiler::block_index_t _index, qreal _value)
|
||
|
{
|
||
|
return sceneView->time2position(easyBlocksTree(_index).node->begin()) < _value;
|
||
|
});
|
||
|
|
||
|
if (firstSync == m_pRoot->sync.end())
|
||
|
firstSync = m_pRoot->sync.begin() + m_pRoot->sync.size() - 1;
|
||
|
else if (firstSync != m_pRoot->sync.begin())
|
||
|
--firstSync;
|
||
|
|
||
|
const auto dw = 4. / view()->scale();
|
||
|
for (auto it = firstSync, end = m_pRoot->sync.end(); it != end; ++it)
|
||
|
{
|
||
|
const auto& item = easyBlock(*it);
|
||
|
|
||
|
const auto left = sceneView->time2position(item.tree.node->begin()) - dw;
|
||
|
if (left > _pos.x())
|
||
|
break;
|
||
|
|
||
|
const auto right = sceneView->time2position(item.tree.node->end()) + dw;
|
||
|
if (right < _pos.x())
|
||
|
continue;
|
||
|
|
||
|
return &item;
|
||
|
}
|
||
|
|
||
|
return nullptr;
|
||
|
}
|
||
|
|
||
|
//////////////////////////////////////////////////////////////////////////
|
||
|
|
||
|
void EasyGraphicsItem::setBoundingRect(qreal x, qreal y, qreal w, qreal h)
|
||
|
{
|
||
|
m_boundingRect.setRect(x, y, w, h);
|
||
|
}
|
||
|
|
||
|
void EasyGraphicsItem::setBoundingRect(const QRectF& _rect)
|
||
|
{
|
||
|
m_boundingRect = _rect;
|
||
|
}
|
||
|
|
||
|
//////////////////////////////////////////////////////////////////////////
|
||
|
|
||
|
::profiler::thread_id_t EasyGraphicsItem::threadId() const
|
||
|
{
|
||
|
return m_pRoot->thread_id;
|
||
|
}
|
||
|
|
||
|
//////////////////////////////////////////////////////////////////////////
|
||
|
|
||
|
uint8_t EasyGraphicsItem::levels() const
|
||
|
{
|
||
|
return static_cast<uint8_t>(m_levels.size());
|
||
|
}
|
||
|
|
||
|
float EasyGraphicsItem::levelY(uint8_t _level) const
|
||
|
{
|
||
|
return y() + static_cast<int>(_level) * static_cast<int>(::profiler_gui::GRAPHICS_ROW_SIZE_FULL);
|
||
|
}
|
||
|
|
||
|
void EasyGraphicsItem::setLevels(uint8_t _levels)
|
||
|
{
|
||
|
typedef decltype(m_levelsIndexes) IndexesT;
|
||
|
static const auto MAX_CHILD_INDEX = ::profiler_gui::numeric_max<IndexesT::value_type>();
|
||
|
|
||
|
m_levels.resize(_levels);
|
||
|
m_levelsIndexes.resize(_levels, MAX_CHILD_INDEX);
|
||
|
m_rightBounds.resize(_levels, -1e100);
|
||
|
}
|
||
|
|
||
|
void EasyGraphicsItem::reserve(uint8_t _level, unsigned int _items)
|
||
|
{
|
||
|
m_levels[_level].reserve(_items);
|
||
|
}
|
||
|
|
||
|
//////////////////////////////////////////////////////////////////////////
|
||
|
|
||
|
const EasyGraphicsItem::Children& EasyGraphicsItem::items(uint8_t _level) const
|
||
|
{
|
||
|
return m_levels[_level];
|
||
|
}
|
||
|
|
||
|
const ::profiler_gui::EasyBlockItem& EasyGraphicsItem::getItem(uint8_t _level, unsigned int _index) const
|
||
|
{
|
||
|
return m_levels[_level][_index];
|
||
|
}
|
||
|
|
||
|
::profiler_gui::EasyBlockItem& EasyGraphicsItem::getItem(uint8_t _level, unsigned int _index)
|
||
|
{
|
||
|
return m_levels[_level][_index];
|
||
|
}
|
||
|
|
||
|
unsigned int EasyGraphicsItem::addItem(uint8_t _level)
|
||
|
{
|
||
|
m_levels[_level].emplace_back();
|
||
|
return static_cast<unsigned int>(m_levels[_level].size() - 1);
|
||
|
}
|
||
|
|
||
|
//////////////////////////////////////////////////////////////////////////
|
||
|
//////////////////////////////////////////////////////////////////////////
|
||
|
|