0
0
mirror of https://github.com/yse/easy_profiler.git synced 2025-01-14 00:27:55 +08:00
easy_profiler/profiler_gui/graphics_scrollbar.cpp
Victor Zarubkin 4db8d4fbb6 (GraphicsScrollbar) Added mini-map which shows duration of blocks relative to each other;
(GraphicsScrollbar) Also added displaying of chronometer item position in graphics scrollbar.
2016-08-03 00:06:36 +03:00

400 lines
10 KiB
C++

/************************************************************************
* file name : graphics_scrollbar.cpp
* ----------------- :
* creation time : 2016/07/04
* copyright : (c) 2016 Victor Zarubkin
* author : Victor Zarubkin
* email : v.s.zarubkin@gmail.com
* ----------------- :
* description : .
* ----------------- :
* change log : * 2016/07/04 Victor Zarubkin: Initial commit.
* :
* : *
* ----------------- :
* license : TODO: add license text
************************************************************************/
#include <algorithm>
#include <QWheelEvent>
#include <QMouseEvent>
#include <QResizeEvent>
#include "graphics_scrollbar.h"
//////////////////////////////////////////////////////////////////////////
const qreal SCALING_COEFFICIENT = 1.25;
const qreal SCALING_COEFFICIENT_INV = 1.0 / SCALING_COEFFICIENT;
const int DEFAULT_TOP = -40;
const int DEFAULT_HEIGHT = 80;
//////////////////////////////////////////////////////////////////////////
auto const clamp = [](qreal _minValue, qreal _value, qreal _maxValue)
{
return (_value < _minValue ? _minValue : (_value > _maxValue ? _maxValue : _value));
};
//////////////////////////////////////////////////////////////////////////
GraphicsHorizontalSlider::GraphicsHorizontalSlider() : Parent(), m_halfwidth(0)
{
setWidth(1);
setBrush(Qt::SolidPattern);
}
GraphicsHorizontalSlider::~GraphicsHorizontalSlider()
{
}
void GraphicsHorizontalSlider::paint(QPainter* _painter, const QStyleOptionGraphicsItem* _option, QWidget* _widget)
{
//Parent::paint(_painter, _option, _widget);
const auto currentScale = static_cast<const GraphicsHorizontalScrollbar*>(scene()->parent())->getWindowScale();
const auto br = rect();
qreal w = width() * currentScale;
qreal dx = 0;
if (w < 1.0)
{
dx = (w - 1.0) * 0.5;
w = 1.0;
}
_painter->save();
_painter->setTransform(QTransform::fromScale(1.0 / currentScale, 1), true);
_painter->setBrush(brush());
_painter->setPen(Qt::NoPen);
_painter->drawRect(QRectF(dx + br.left() * currentScale, br.top(), w, br.height()));
_painter->restore();
}
qreal GraphicsHorizontalSlider::width() const
{
return m_halfwidth * 2.0;
}
qreal GraphicsHorizontalSlider::halfwidth() const
{
return m_halfwidth;
}
void GraphicsHorizontalSlider::setWidth(qreal _width)
{
m_halfwidth = _width * 0.5;
setRect(-m_halfwidth, DEFAULT_TOP, _width, DEFAULT_HEIGHT);
}
void GraphicsHorizontalSlider::setHalfwidth(qreal _halfwidth)
{
m_halfwidth = _halfwidth;
setRect(-m_halfwidth, DEFAULT_TOP, m_halfwidth * 2.0, DEFAULT_HEIGHT);
}
void GraphicsHorizontalSlider::setColor(QRgb _color)
{
auto b = brush();
b.setColor(QColor::fromRgba(_color));
setBrush(b);
}
//////////////////////////////////////////////////////////////////////////
MinimapItem::MinimapItem() : Parent(), m_pSource(nullptr), m_maxDuration(0)
{
}
MinimapItem::~MinimapItem()
{
}
QRectF MinimapItem::boundingRect() const
{
return m_boundingRect;
}
void MinimapItem::paint(QPainter* _painter, const QStyleOptionGraphicsItem* _option, QWidget* _widget)
{
if (m_pSource == nullptr)
{
return;
}
const auto currentScale = static_cast<const GraphicsHorizontalScrollbar*>(scene()->parent())->getWindowScale();
const auto bottom = m_boundingRect.bottom();
const auto coeff = (m_boundingRect.height() - 5) / m_maxDuration;
QRectF rect;
QBrush brush(Qt::SolidPattern);
QRgb previousColor = 0;
brush.setColor(QColor::fromRgba(0x80808080));
_painter->save();
_painter->setPen(Qt::NoPen);
_painter->setBrush(brush);
_painter->setTransform(QTransform::fromScale(1.0 / currentScale, 1), true);
auto& items = *m_pSource;
for (const auto& item : items)
{
//if (previousColor != item.color)
//{
// // Set background color brush for rectangle
// previousColor = item.color;
// brush.setColor(QColor::fromRgba(0x40000000 | item.color));
// _painter->setBrush(brush);
//}
// Draw rectangle
auto h = 5 + item.width() * coeff;
rect.setRect(item.left() * currentScale, bottom - h, ::std::max(item.width() * currentScale, 1.0), h);
_painter->drawRect(rect);
}
_painter->restore();
}
void MinimapItem::setBoundingRect(const QRectF& _rect)
{
m_boundingRect = _rect;
}
void MinimapItem::setSource(const ProfItems* _items)
{
m_pSource = _items;
if (m_pSource)
{
if (m_pSource->empty())
{
m_pSource = nullptr;
}
m_maxDuration = 0;
for (const auto& item : *m_pSource)
{
if (item.width() > m_maxDuration)
{
m_maxDuration = item.width();
}
}
}
if (m_pSource == nullptr)
{
hide();
}
else
{
show();
}
}
//////////////////////////////////////////////////////////////////////////
GraphicsHorizontalScrollbar::GraphicsHorizontalScrollbar(QWidget* _parent)
: Parent(_parent)
, m_minimumValue(0)
, m_maximumValue(500)
, m_value(10)
, m_windowScale(1)
, m_mouseButtons(Qt::NoButton)
, m_slider(nullptr)
, m_chronometerIndicator(nullptr)
, m_minimap(nullptr)
, m_bScrolling(false)
{
setCacheMode(QGraphicsView::CacheNone);
setTransformationAnchor(QGraphicsView::AnchorUnderMouse);
setViewportUpdateMode(QGraphicsView::BoundingRectViewportUpdate);
setOptimizationFlag(QGraphicsView::DontSavePainterState, true);
setHorizontalScrollBarPolicy(Qt::ScrollBarAlwaysOff);
setVerticalScrollBarPolicy(Qt::ScrollBarAlwaysOff);
setContentsMargins(0, 0, 0, 0);
setMinimumHeight(DEFAULT_HEIGHT + 2);
setMaximumHeight(DEFAULT_HEIGHT + 2);
auto selfScene = new QGraphicsScene(this);
selfScene->setSceneRect(0, DEFAULT_TOP, 500, DEFAULT_HEIGHT);
setScene(selfScene);
m_slider = new GraphicsHorizontalSlider();
m_slider->setPos(0, 0);
m_slider->setZValue(5);
m_slider->setColor(0x90e00000);
selfScene->addItem(m_slider);
m_chronometerIndicator = new GraphicsHorizontalSlider();
m_chronometerIndicator->setPos(0, 0);
m_chronometerIndicator->setZValue(10);
m_chronometerIndicator->setColor(0x90404040);
selfScene->addItem(m_chronometerIndicator);
m_chronometerIndicator->hide();
m_minimap = new MinimapItem();
m_minimap->setPos(0, 0);
m_minimap->setBoundingRect(selfScene->sceneRect());
selfScene->addItem(m_minimap);
m_minimap->hide();
centerOn(0, 0);
}
GraphicsHorizontalScrollbar::~GraphicsHorizontalScrollbar()
{
}
//////////////////////////////////////////////////////////////////////////
qreal GraphicsHorizontalScrollbar::getWindowScale() const
{
return m_windowScale;
}
qreal GraphicsHorizontalScrollbar::minimum() const
{
return m_minimumValue;
}
qreal GraphicsHorizontalScrollbar::maximum() const
{
return m_maximumValue;
}
qreal GraphicsHorizontalScrollbar::range() const
{
return m_maximumValue - m_minimumValue;
}
qreal GraphicsHorizontalScrollbar::value() const
{
return m_value;
}
//////////////////////////////////////////////////////////////////////////
void GraphicsHorizontalScrollbar::setValue(qreal _value)
{
m_value = clamp(m_minimumValue, _value, ::std::max(m_minimumValue, m_maximumValue - m_slider->width()));
m_slider->setX(m_value + m_slider->halfwidth());
emit valueChanged(m_value);
}
void GraphicsHorizontalScrollbar::setRange(qreal _minValue, qreal _maxValue)
{
const auto oldRange = range();
const auto oldValue = oldRange < 1e-3 ? 0.0 : m_value / oldRange;
m_minimumValue = _minValue;
m_maximumValue = _maxValue;
scene()->setSceneRect(_minValue, DEFAULT_TOP, _maxValue - _minValue, DEFAULT_HEIGHT);
m_minimap->setBoundingRect(scene()->sceneRect());
emit rangeChanged();
setValue(_minValue + oldValue * range());
onWindowWidthChange(width());
}
void GraphicsHorizontalScrollbar::setSliderWidth(qreal _width)
{
m_slider->setWidth(_width);
setValue(m_value);
}
//////////////////////////////////////////////////////////////////////////
void GraphicsHorizontalScrollbar::setChronoPos(qreal _left, qreal _right)
{
m_chronometerIndicator->setWidth(_right - _left);
m_chronometerIndicator->setX(_left + m_chronometerIndicator->halfwidth());
}
void GraphicsHorizontalScrollbar::showChrono()
{
m_chronometerIndicator->show();
}
void GraphicsHorizontalScrollbar::hideChrono()
{
m_chronometerIndicator->hide();
}
//////////////////////////////////////////////////////////////////////////
void GraphicsHorizontalScrollbar::setMinimapFrom(const ProfItems* _items)
{
m_minimap->setSource(_items);
}
//////////////////////////////////////////////////////////////////////////
void GraphicsHorizontalScrollbar::mousePressEvent(QMouseEvent* _event)
{
m_mouseButtons = _event->buttons();
if (m_mouseButtons & Qt::LeftButton)
{
m_bScrolling = true;
m_mousePressPos = _event->pos();
setValue(mapToScene(m_mousePressPos).x() - m_minimumValue);
}
_event->accept();
//QGraphicsView::mousePressEvent(_event);
}
void GraphicsHorizontalScrollbar::mouseReleaseEvent(QMouseEvent* _event)
{
m_mouseButtons = _event->buttons();
m_bScrolling = false;
_event->accept();
//QGraphicsView::mouseReleaseEvent(_event);
}
void GraphicsHorizontalScrollbar::mouseMoveEvent(QMouseEvent* _event)
{
if (m_mouseButtons & Qt::LeftButton)
{
const auto pos = _event->pos();
const auto delta = pos - m_mousePressPos;
m_mousePressPos = pos;
if (m_bScrolling)
{
setValue(m_value + delta.x() / m_windowScale);
}
}
}
void GraphicsHorizontalScrollbar::resizeEvent(QResizeEvent* _event)
{
onWindowWidthChange(_event->size().width());
}
void GraphicsHorizontalScrollbar::onWindowWidthChange(qreal _width)
{
const auto oldScale = m_windowScale;
const auto scrollingRange = range();
if (scrollingRange < 1e-3)
{
m_windowScale = 1;
}
else
{
m_windowScale = _width / scrollingRange;
}
scale(m_windowScale / oldScale, 1);
}
//////////////////////////////////////////////////////////////////////////