mirror of
https://github.com/tumic0/GPXSee.git
synced 2025-01-19 04:02:09 +01:00
196 lines
4.0 KiB
C++
196 lines
4.0 KiB
C++
#include <QPainter>
|
|
#include "graphitem.h"
|
|
|
|
|
|
#define GRAPH_WIDTH 1
|
|
#define HOVER_WIDTH 2
|
|
|
|
static bool hasTime(const Graph &graph)
|
|
{
|
|
for (int i = 0; i < graph.count(); i++)
|
|
if (std::isnan(graph.at(i).t()))
|
|
return false;
|
|
|
|
return true;
|
|
}
|
|
|
|
GraphItem::GraphItem(const Graph &graph, QGraphicsItem *parent)
|
|
: QGraphicsObject(parent)
|
|
{
|
|
_id = 0;
|
|
|
|
_pen = QPen(Qt::black, GRAPH_WIDTH);
|
|
|
|
_type = Distance;
|
|
_graph = graph;
|
|
_sx = 1.0; _sy = 1.0;
|
|
_time = hasTime(_graph);
|
|
|
|
updatePath();
|
|
updateBounds();
|
|
}
|
|
|
|
void GraphItem::paint(QPainter *painter, const QStyleOptionGraphicsItem *option,
|
|
QWidget *widget)
|
|
{
|
|
Q_UNUSED(option);
|
|
Q_UNUSED(widget);
|
|
|
|
painter->setPen(_pen);
|
|
painter->drawPath(_path);
|
|
|
|
/*
|
|
QPen p = QPen(QBrush(Qt::red), 0);
|
|
painter->setPen(p);
|
|
painter->drawRect(boundingRect());
|
|
*/
|
|
}
|
|
|
|
void GraphItem::setGraphType(GraphType type)
|
|
{
|
|
prepareGeometryChange();
|
|
|
|
_type = type;
|
|
updatePath();
|
|
updateBounds();
|
|
}
|
|
|
|
void GraphItem::setColor(const QColor &color)
|
|
{
|
|
_pen.setColor(color);
|
|
update();
|
|
}
|
|
|
|
qreal GraphItem::yAtX(qreal x)
|
|
{
|
|
int low = 0;
|
|
int high = _graph.count() - 1;
|
|
int mid = 0;
|
|
|
|
Q_ASSERT(high > low);
|
|
Q_ASSERT(x >= _graph.at(low).x(_type) && x <= _graph.at(high).x(_type));
|
|
|
|
while (low <= high) {
|
|
mid = low + ((high - low) / 2);
|
|
const GraphPoint &p = _graph.at(mid);
|
|
if (p.x(_type) > x)
|
|
high = mid - 1;
|
|
else if (p.x(_type) < x)
|
|
low = mid + 1;
|
|
else
|
|
return -p.y();
|
|
}
|
|
|
|
QLineF l;
|
|
if (_graph.at(mid).x(_type) < x)
|
|
l = QLineF(_graph.at(mid).x(_type), _graph.at(mid).y(),
|
|
_graph.at(mid+1).x(_type), _graph.at(mid+1).y());
|
|
else
|
|
l = QLineF(_graph.at(mid-1).x(_type), _graph.at(mid-1).y(),
|
|
_graph.at(mid).x(_type), _graph.at(mid).y());
|
|
|
|
return -l.pointAt((x - l.p1().x()) / (l.p2().x() - l.p1().x())).y();
|
|
}
|
|
|
|
qreal GraphItem::distanceAtTime(qreal time)
|
|
{
|
|
int low = 0;
|
|
int high = _graph.count() - 1;
|
|
int mid = 0;
|
|
|
|
Q_ASSERT(high > low);
|
|
Q_ASSERT(time >= _graph.at(low).t() && time <= _graph.at(high).t());
|
|
|
|
while (low <= high) {
|
|
mid = low + ((high - low) / 2);
|
|
const GraphPoint &p = _graph.at(mid);
|
|
if (p.t() > time)
|
|
high = mid - 1;
|
|
else if (p.t() < time)
|
|
low = mid + 1;
|
|
else
|
|
return _graph.at(mid).s();
|
|
}
|
|
|
|
QLineF l;
|
|
if (_graph.at(mid).t() < time)
|
|
l = QLineF(_graph.at(mid).t(), _graph.at(mid).s(), _graph.at(mid+1).t(),
|
|
_graph.at(mid+1).s());
|
|
else
|
|
l = QLineF(_graph.at(mid-1).t(), _graph.at(mid-1).s(),
|
|
_graph.at(mid).t(), _graph.at(mid).s());
|
|
|
|
return l.pointAt((time - l.p1().x()) / (l.p2().x() - l.p1().x())).y();
|
|
}
|
|
|
|
void GraphItem::emitSliderPositionChanged(qreal pos)
|
|
{
|
|
if (_type == Time) {
|
|
if (_time) {
|
|
if (pos <= _graph.last().t())
|
|
emit sliderPositionChanged(distanceAtTime(pos));
|
|
else
|
|
emit sliderPositionChanged(_graph.last().s() + 1);
|
|
} else
|
|
emit sliderPositionChanged(_graph.last().s() + 1);
|
|
} else
|
|
emit sliderPositionChanged(pos);
|
|
}
|
|
|
|
void GraphItem::selected(bool selected)
|
|
{
|
|
if (selected) {
|
|
_pen.setWidth(HOVER_WIDTH);
|
|
setZValue(zValue() + 1.0);
|
|
} else {
|
|
_pen.setWidth(GRAPH_WIDTH);
|
|
setZValue(zValue() - 1.0);
|
|
}
|
|
|
|
update();
|
|
}
|
|
|
|
void GraphItem::setScale(qreal sx, qreal sy)
|
|
{
|
|
if (_sx == sx && _sy == sy)
|
|
return;
|
|
|
|
prepareGeometryChange();
|
|
|
|
_sx = sx; _sy = sy;
|
|
updatePath();
|
|
}
|
|
|
|
void GraphItem::updatePath()
|
|
{
|
|
_path = QPainterPath();
|
|
|
|
if (_type == Time && !_time)
|
|
return;
|
|
|
|
_path.moveTo(_graph.first().x(_type) * _sx, -_graph.first().y() * _sy);
|
|
for (int i = 1; i < _graph.size(); i++)
|
|
_path.lineTo(_graph.at(i).x(_type) * _sx, -_graph.at(i).y() * _sy);
|
|
}
|
|
|
|
void GraphItem::updateBounds()
|
|
{
|
|
if (_type == Time && !_time) {
|
|
_bounds = QRectF();
|
|
return;
|
|
}
|
|
|
|
qreal bottom, top, left, right;
|
|
|
|
QPointF p = QPointF(_graph.first().x(_type), -_graph.first().y());
|
|
bottom = p.y(); top = p.y(); left = p.x(); right = p.x();
|
|
|
|
for (int i = 1; i < _graph.size(); i++) {
|
|
p = QPointF(_graph.at(i).x(_type), -_graph.at(i).y());
|
|
bottom = qMax(bottom, p.y()); top = qMin(top, p.y());
|
|
right = qMax(right, p.x()); left = qMin(left, p.x());
|
|
}
|
|
|
|
_bounds = QRectF(QPointF(left, top), QPointF(right, bottom));
|
|
}
|