1
0
mirror of https://github.com/tumic0/GPXSee.git synced 2024-10-07 07:13:21 +02:00
GPXSee/src/GUI/pathitem.cpp

276 lines
5.9 KiB
C++
Raw Normal View History

2016-11-18 18:02:40 +01:00
#include <cmath>
2016-09-26 21:01:58 +02:00
#include <QCursor>
#include <QPainter>
#include "common/greatcircle.h"
2017-11-26 18:54:03 +01:00
#include "map/map.h"
2016-09-26 21:01:58 +02:00
#include "pathitem.h"
2018-09-15 08:40:37 +02:00
#define GEOGRAPHICAL_MILE 1855.3248
static unsigned segments(qreal distance)
{
return ceil(distance / GEOGRAPHICAL_MILE);
}
PathItem::PathItem(const Path &path, Map *map, QGraphicsItem *parent)
: QGraphicsObject(parent)
2016-09-26 21:01:58 +02:00
{
Q_ASSERT(path.count() >= 2);
_path = path;
_map = map;
_digitalZoom = 0;
2016-12-06 01:48:26 +01:00
_width = 3;
2016-09-26 21:01:58 +02:00
QBrush brush(Qt::SolidPattern);
2016-12-06 01:48:26 +01:00
_pen = QPen(brush, _width);
2016-09-26 21:01:58 +02:00
2018-09-15 13:40:21 +02:00
updatePainterPath();
updateShape();
2016-09-26 21:01:58 +02:00
_marker = new MarkerItem(this);
_marker->setPos(position(_path.at(0).distance()));
_markerDistance = _path.at(0).distance();
2016-09-26 21:01:58 +02:00
setCursor(Qt::ArrowCursor);
setAcceptHoverEvents(true);
}
void PathItem::updateShape()
{
QPainterPathStroker s;
s.setWidth((_width + 1) * pow(2, -_digitalZoom));
_shape = s.createStroke(_painterPath);
}
2018-09-15 13:40:21 +02:00
void PathItem::addSegment(const Coordinates &c1, const Coordinates &c2)
{
if (fabs(c1.lon() - c2.lon()) > 180.0) {
2018-10-11 18:19:35 +02:00
// Split segment on date line crossing
QPointF p;
if (c2.lon() < 0) {
QLineF l(QPointF(c1.lon(), c1.lat()), QPointF(c2.lon() + 360,
c2.lat()));
QLineF dl(QPointF(180, -90), QPointF(180, 90));
l.intersect(dl, &p);
_painterPath.lineTo(_map->ll2xy(Coordinates(180, p.y())));
_painterPath.moveTo(_map->ll2xy(Coordinates(-180, p.y())));
} else {
QLineF l(QPointF(c1.lon(), c1.lat()), QPointF(c2.lon() - 360,
c2.lat()));
QLineF dl(QPointF(-180, -90), QPointF(-180, 90));
l.intersect(dl, &p);
_painterPath.lineTo(_map->ll2xy(Coordinates(-180, p.y())));
_painterPath.moveTo(_map->ll2xy(Coordinates(180, p.y())));
}
_painterPath.lineTo(_map->ll2xy(c2));
} else
2018-09-15 13:40:21 +02:00
_painterPath.lineTo(_map->ll2xy(c2));
}
void PathItem::updatePainterPath()
{
_painterPath = QPainterPath();
2018-09-15 13:40:21 +02:00
_painterPath.moveTo(_map->ll2xy(_path.first().coordinates()));
for (int i = 1; i < _path.size(); i++) {
const PathPoint &p1 = _path.at(i-1);
const PathPoint &p2 = _path.at(i);
unsigned n = segments(p2.distance() - p1.distance());
2018-09-15 08:40:37 +02:00
if (n > 1) {
2018-09-15 13:40:21 +02:00
GreatCircle gc(p1.coordinates(), p2.coordinates());
Coordinates last = p1.coordinates();
2018-09-13 01:15:43 +02:00
for (unsigned j = 1; j <= n; j++) {
Coordinates c(gc.pointAt(j/(double)n));
2018-09-15 13:40:21 +02:00
addSegment(last, c);
last = c;
}
} else
2018-09-15 13:40:21 +02:00
addSegment(p1.coordinates(), p2.coordinates());
}
}
2016-09-26 21:01:58 +02:00
void PathItem::paint(QPainter *painter, const QStyleOptionGraphicsItem *option,
QWidget *widget)
{
Q_UNUSED(option);
Q_UNUSED(widget);
painter->setPen(_pen);
painter->drawPath(_painterPath);
2016-09-26 21:01:58 +02:00
/*
QPen p = QPen(QBrush(Qt::red), 0);
painter->setPen(p);
painter->drawRect(boundingRect());
*/
}
void PathItem::setMap(Map *map)
2016-09-26 21:01:58 +02:00
{
prepareGeometryChange();
2017-10-04 23:15:39 +02:00
_map = map;
2018-09-15 13:40:21 +02:00
updatePainterPath();
2016-09-26 21:01:58 +02:00
updateShape();
_marker->setPos(position(_markerDistance));
2016-09-26 21:01:58 +02:00
}
void PathItem::setColor(const QColor &color)
{
2017-10-04 23:15:39 +02:00
if (_pen.color() == color)
return;
2016-09-26 21:01:58 +02:00
_pen.setColor(color);
update();
}
2017-04-05 22:53:25 +02:00
void PathItem::setWidth(qreal width)
2016-12-06 01:48:26 +01:00
{
2017-10-04 23:15:39 +02:00
if (_width == width)
return;
2016-12-06 01:48:26 +01:00
prepareGeometryChange();
_width = width;
_pen.setWidthF(_width * pow(2, -_digitalZoom));
2016-12-06 01:48:26 +01:00
updateShape();
}
void PathItem::setStyle(Qt::PenStyle style)
{
2017-10-04 23:15:39 +02:00
if (_pen.style() == style)
return;
2016-12-06 01:48:26 +01:00
_pen.setStyle(style);
update();
}
void PathItem::setDigitalZoom(int zoom)
{
2017-10-04 23:15:39 +02:00
if (_digitalZoom == zoom)
return;
prepareGeometryChange();
_digitalZoom = zoom;
_pen.setWidthF(_width * pow(2, -_digitalZoom));
_marker->setScale(pow(2, -_digitalZoom));
updateShape();
}
2016-11-14 22:12:43 +01:00
QPointF PathItem::position(qreal x) const
{
int low = 0;
int high = _path.count() - 1;
2016-11-14 22:12:43 +01:00
int mid = 0;
Q_ASSERT(high > low);
Q_ASSERT(x >= _path.at(low).distance() && x <= _path.at(high).distance());
2016-11-14 22:12:43 +01:00
while (low <= high) {
mid = low + ((high - low) / 2);
qreal val = _path.at(mid).distance();
2016-11-14 22:12:43 +01:00
if (val > x)
high = mid - 1;
else if (val < x)
low = mid + 1;
else
return _map->ll2xy(_path.at(mid).coordinates());
2016-11-14 22:12:43 +01:00
}
Coordinates c1, c2;
2016-11-14 22:12:43 +01:00
qreal p1, p2;
if (_path.at(mid).distance() < x) {
c1 = _path.at(mid).coordinates(); c2 = _path.at(mid+1).coordinates();
p1 = _path.at(mid).distance(); p2 = _path.at(mid+1).distance();
2016-11-14 22:12:43 +01:00
} else {
c1 = _path.at(mid-1).coordinates(); c2 = _path.at(mid).coordinates();
p1 = _path.at(mid-1).distance(); p2 = _path.at(mid).distance();
2016-11-14 22:12:43 +01:00
}
unsigned n = segments(p2 - p1);
if (n > 1) {
2018-09-13 01:15:43 +02:00
GreatCircle gc(c1, c2);
// Great circle point
double f = (x - p1) / (p2 - p1);
QPointF p(_map->ll2xy(gc.pointAt(f)));
// Segment line of the great circle path
double f1 = floor(n * f) / n;
double f2 = ceil(n * f) / n;
QLineF l(_map->ll2xy(gc.pointAt(f1)), _map->ll2xy(gc.pointAt(f2)));
// Project the great circle point to the segment line
QLineF u = l.unitVector();
double lambda = (u.dx() * (p.x() - l.p1().x())) + (u.dy() * (p.y()
- l.p1().y()));
return QPointF((u.dx() * lambda) + l.p1().x(), (u.dy() * lambda)
+ l.p1().y());
2018-09-13 01:15:43 +02:00
} else {
QLineF l(_map->ll2xy(c1), _map->ll2xy(c2));
return l.pointAt((x - p1) / (p2 - p1));
}
2016-11-14 22:12:43 +01:00
}
2016-09-26 21:01:58 +02:00
void PathItem::moveMarker(qreal distance)
{
if (distance >= _path.first().distance()
&& distance <= _path.last().distance()) {
2016-09-26 21:01:58 +02:00
_marker->setVisible(true);
2016-11-14 22:12:43 +01:00
_marker->setPos(position(distance));
_markerDistance = distance;
2016-11-18 18:02:40 +01:00
} else
_marker->setVisible(false);
2016-09-26 21:01:58 +02:00
}
2017-12-03 00:36:52 +01:00
void PathItem::setMarkerColor(const QColor &color)
{
_marker->setColor(color);
}
void PathItem::hover(bool hover)
{
if (hover) {
_pen.setWidth((_width + 1) * pow(2, -_digitalZoom));
setZValue(zValue() + 1.0);
} else {
_pen.setWidth(_width * pow(2, -_digitalZoom));
setZValue(zValue() - 1.0);
}
update();
}
2016-09-26 21:01:58 +02:00
void PathItem::hoverEnterEvent(QGraphicsSceneHoverEvent *event)
{
Q_UNUSED(event);
_pen.setWidthF((_width + 1) * pow(2, -_digitalZoom));
2016-09-26 21:01:58 +02:00
setZValue(zValue() + 1.0);
update();
emit selected(true);
}
void PathItem::hoverLeaveEvent(QGraphicsSceneHoverEvent *event)
{
Q_UNUSED(event);
_pen.setWidthF(_width * pow(2, -_digitalZoom));
2016-09-26 21:01:58 +02:00
setZValue(zValue() - 1.0);
update();
emit selected(false);
}