linechartitem.cpp
372 lines
| 16.2 KiB
| text/x-c
|
CppLexer
Jani Honkonen
|
r794 | /**************************************************************************** | ||
** | ||||
Miikka Heikkinen
|
r2432 | ** Copyright (C) 2013 Digia Plc | ||
Jani Honkonen
|
r794 | ** All rights reserved. | ||
** For any questions to Digia, please use contact form at http://qt.digia.com | ||||
** | ||||
** This file is part of the Qt Commercial Charts Add-on. | ||||
** | ||||
** $QT_BEGIN_LICENSE$ | ||||
** Licensees holding valid Qt Commercial licenses may use this file in | ||||
** accordance with the Qt Commercial License Agreement provided with the | ||||
** Software or, alternatively, in accordance with the terms contained in | ||||
** a written agreement between you and Digia. | ||||
** | ||||
** If you have questions regarding the use of this file, please use | ||||
** contact form at http://qt.digia.com | ||||
** $QT_END_LICENSE$ | ||||
** | ||||
****************************************************************************/ | ||||
Michal Klocek
|
r144 | #include "linechartitem_p.h" | ||
Michal Klocek
|
r349 | #include "qlineseries.h" | ||
Michal Klocek
|
r938 | #include "qlineseries_p.h" | ||
Michal Klocek
|
r131 | #include "chartpresenter_p.h" | ||
Miikka Heikkinen
|
r2483 | #include "polardomain_p.h" | ||
Michal Klocek
|
r21 | #include <QPainter> | ||
Michal Klocek
|
r1218 | #include <QGraphicsSceneMouseEvent> | ||
Michal Klocek
|
r144 | |||
Tero Ahola
|
r30 | QTCOMMERCIALCHART_BEGIN_NAMESPACE | ||
Michal Klocek
|
r21 | |||
Tero Ahola
|
r1792 | const qreal mouseEventMinWidth(12); | ||
Michal Klocek
|
r150 | |||
Miikka Heikkinen
|
r2483 | LineChartItem::LineChartItem(QLineSeries *series, QGraphicsItem *item) | ||
Michal Klocek
|
r2273 | : XYChart(series,item), | ||
Jani Honkonen
|
r2097 | m_series(series), | ||
Miikka Heikkinen
|
r2483 | m_pointsVisible(false), | ||
m_chartType(QChart::ChartTypeUndefined) | ||||
Michal Klocek
|
r21 | { | ||
Marek Rosa
|
r2255 | setAcceptHoverEvents(true); | ||
Michal Klocek
|
r262 | setZValue(ChartPresenter::LineChartZValue); | ||
Jani Honkonen
|
r2097 | QObject::connect(series->d_func(), SIGNAL(updated()), this, SLOT(handleUpdated())); | ||
Tero Ahola
|
r1342 | QObject::connect(series, SIGNAL(visibleChanged()), this, SLOT(handleUpdated())); | ||
Tero Ahola
|
r2067 | QObject::connect(series, SIGNAL(opacityChanged()), this, SLOT(handleUpdated())); | ||
Michal Klocek
|
r439 | handleUpdated(); | ||
Michal Klocek
|
r21 | } | ||
Michal Klocek
|
r144 | QRectF LineChartItem::boundingRect() const | ||
Tero Ahola
|
r103 | { | ||
Jani Honkonen
|
r2097 | return m_rect; | ||
Tero Ahola
|
r103 | } | ||
Michal Klocek
|
r85 | |||
Michal Klocek
|
r144 | QPainterPath LineChartItem::shape() const | ||
Michal Klocek
|
r67 | { | ||
Miikka Heikkinen
|
r2483 | return m_shapePath; | ||
Michal Klocek
|
r131 | } | ||
Michal Klocek
|
r85 | |||
Michal Klocek
|
r1217 | void LineChartItem::updateGeometry() | ||
Michal Klocek
|
r131 | { | ||
Michal Klocek
|
r1819 | m_points = geometryPoints(); | ||
Miikka Heikkinen
|
r2483 | const QVector<QPointF> &points = m_points; | ||
Michal Klocek
|
r1217 | |||
Miikka Heikkinen
|
r2483 | if (points.size() == 0) { | ||
Michal Klocek
|
r1269 | prepareGeometryChange(); | ||
Miikka Heikkinen
|
r2483 | m_fullPath = QPainterPath(); | ||
Miikka Heikkinen
|
r2451 | m_linePath = QPainterPath(); | ||
Michal Klocek
|
r1269 | m_rect = QRect(); | ||
Marek Rosa
|
r545 | return; | ||
} | ||||
Michal Klocek
|
r464 | |||
Miikka Heikkinen
|
r2483 | QPainterPath linePath; | ||
QPainterPath fullPath; | ||||
// Use worst case scenario to determine required margin. | ||||
qreal margin = m_linePen.width() * 1.42; | ||||
Michal Klocek
|
r464 | |||
Miikka Heikkinen
|
r2483 | // Area series use component line series that aren't necessarily added to the chart themselves, | ||
// so check if chart type is forced before trying to obtain it from the chart. | ||||
QChart::ChartType chartType = m_chartType; | ||||
if (chartType == QChart::ChartTypeUndefined) | ||||
chartType = m_series->chart()->chartType(); | ||||
Michal Klocek
|
r1819 | |||
Miikka Heikkinen
|
r2483 | // For polar charts, we need special handling for angular (horizontal) | ||
// points that are off-grid. | ||||
if (chartType == QChart::ChartTypePolar) { | ||||
QPainterPath linePathLeft; | ||||
QPainterPath linePathRight; | ||||
QPainterPath *currentSegmentPath = 0; | ||||
QPainterPath *previousSegmentPath = 0; | ||||
qreal minX = domain()->minX(); | ||||
qreal maxX = domain()->maxX(); | ||||
qreal minY = domain()->minY(); | ||||
Miikka Heikkinen
|
r2491 | QPointF currentSeriesPoint = m_series->at(0); | ||
Miikka Heikkinen
|
r2483 | QPointF currentGeometryPoint = points.at(0); | ||
QPointF previousGeometryPoint = points.at(0); | ||||
Michal Klocek
|
r1819 | int size = m_linePen.width(); | ||
Miikka Heikkinen
|
r2483 | bool pointOffGrid = false; | ||
bool previousPointWasOffGrid = (currentSeriesPoint.x() < minX || currentSeriesPoint.x() > maxX); | ||||
qreal domainRadius = domain()->size().height() / 2.0; | ||||
const QPointF centerPoint(domainRadius, domainRadius); | ||||
if (!previousPointWasOffGrid) { | ||||
fullPath.moveTo(points.at(0)); | ||||
if (m_pointsVisible && currentSeriesPoint.y() >= minY) { | ||||
// Do not draw ellipses for points below minimum Y. | ||||
linePath.addEllipse(points.at(0), size, size); | ||||
fullPath.addEllipse(points.at(0), size, size); | ||||
linePath.moveTo(points.at(0)); | ||||
fullPath.moveTo(points.at(0)); | ||||
} | ||||
Michal Klocek
|
r1819 | } | ||
Miikka Heikkinen
|
r2483 | qreal leftMarginLine = centerPoint.x() - margin; | ||
qreal rightMarginLine = centerPoint.x() + margin; | ||||
qreal horizontal = centerPoint.y(); | ||||
Michal Klocek
|
r464 | |||
Miikka Heikkinen
|
r2489 | // See ScatterChartItem::updateGeometry() for explanation why seriesLastIndex is needed | ||
const int seriesLastIndex = m_series->count() - 1; | ||||
Miikka Heikkinen
|
r2483 | for (int i = 1; i < points.size(); i++) { | ||
// Interpolating line fragments would be ugly when thick pen is used, | ||||
// so we work around it by utilizing three separate | ||||
// paths for line segments and clip those with custom regions at paint time. | ||||
// "Right" path contains segments that cross the axis line with visible point on the | ||||
// right side of the axis line, as well as segments that have one point within the margin | ||||
// on the right side of the axis line and another point on the right side of the chart. | ||||
// "Left" path contains points with similarly on the left side. | ||||
// "Full" path contains rest of the points. | ||||
// This doesn't yield perfect results always. E.g. when segment covers more than 90 | ||||
// degrees and both of the points are within the margin, one in the top half and one in the | ||||
// bottom half of the chart, the bottom one gets clipped incorrectly. | ||||
// However, this should be rare occurrence in any sensible chart. | ||||
Miikka Heikkinen
|
r2491 | currentSeriesPoint = m_series->at(qMin(seriesLastIndex, i)); | ||
Miikka Heikkinen
|
r2483 | currentGeometryPoint = points.at(i); | ||
pointOffGrid = (currentSeriesPoint.x() < minX || currentSeriesPoint.x() > maxX); | ||||
// Draw something unless both off-grid | ||||
if (!pointOffGrid || !previousPointWasOffGrid) { | ||||
QPointF intersectionPoint; | ||||
qreal y; | ||||
if (pointOffGrid != previousPointWasOffGrid) { | ||||
if (currentGeometryPoint.x() == previousGeometryPoint.x()) { | ||||
y = currentGeometryPoint.y() + (currentGeometryPoint.y() - previousGeometryPoint.y()) / 2.0; | ||||
} else { | ||||
qreal ratio = (centerPoint.x() - currentGeometryPoint.x()) / (currentGeometryPoint.x() - previousGeometryPoint.x()); | ||||
y = currentGeometryPoint.y() + (currentGeometryPoint.y() - previousGeometryPoint.y()) * ratio; | ||||
} | ||||
intersectionPoint = QPointF(centerPoint.x(), y); | ||||
} | ||||
bool dummyOk; // We know points are ok, but this is needed | ||||
qreal currentAngle = static_cast<PolarDomain *>(domain())->toAngularCoordinate(currentSeriesPoint.x(), dummyOk); | ||||
Miikka Heikkinen
|
r2491 | qreal previousAngle = static_cast<PolarDomain *>(domain())->toAngularCoordinate(m_series->at(i - 1).x(), dummyOk); | ||
Miikka Heikkinen
|
r2483 | |||
if ((qAbs(currentAngle - previousAngle) > 180.0)) { | ||||
// If the angle between two points is over 180 degrees (half X range), | ||||
// any direct segment between them becomes meaningless. | ||||
// In this case two line segments are drawn instead, from previous | ||||
// point to the center and from center to current point. | ||||
if ((previousAngle < 0.0 || (previousAngle <= 180.0 && previousGeometryPoint.x() < rightMarginLine)) | ||||
&& previousGeometryPoint.y() < horizontal) { | ||||
currentSegmentPath = &linePathRight; | ||||
} else if ((previousAngle > 360.0 || (previousAngle > 180.0 && previousGeometryPoint.x() > leftMarginLine)) | ||||
&& previousGeometryPoint.y() < horizontal) { | ||||
currentSegmentPath = &linePathLeft; | ||||
} else if (previousAngle > 0.0 && previousAngle < 360.0) { | ||||
currentSegmentPath = &linePath; | ||||
} else { | ||||
currentSegmentPath = 0; | ||||
} | ||||
if (currentSegmentPath) { | ||||
if (previousSegmentPath != currentSegmentPath) | ||||
currentSegmentPath->moveTo(previousGeometryPoint); | ||||
if (previousPointWasOffGrid) | ||||
fullPath.moveTo(intersectionPoint); | ||||
currentSegmentPath->lineTo(centerPoint); | ||||
fullPath.lineTo(centerPoint); | ||||
} | ||||
previousSegmentPath = currentSegmentPath; | ||||
if ((currentAngle < 0.0 || (currentAngle <= 180.0 && currentGeometryPoint.x() < rightMarginLine)) | ||||
&& currentGeometryPoint.y() < horizontal) { | ||||
currentSegmentPath = &linePathRight; | ||||
} else if ((currentAngle > 360.0 || (currentAngle > 180.0 &¤tGeometryPoint.x() > leftMarginLine)) | ||||
&& currentGeometryPoint.y() < horizontal) { | ||||
currentSegmentPath = &linePathLeft; | ||||
} else if (currentAngle > 0.0 && currentAngle < 360.0) { | ||||
currentSegmentPath = &linePath; | ||||
} else { | ||||
currentSegmentPath = 0; | ||||
} | ||||
if (currentSegmentPath) { | ||||
if (previousSegmentPath != currentSegmentPath) | ||||
currentSegmentPath->moveTo(centerPoint); | ||||
if (!previousSegmentPath) | ||||
fullPath.moveTo(centerPoint); | ||||
currentSegmentPath->lineTo(currentGeometryPoint); | ||||
if (pointOffGrid) | ||||
fullPath.lineTo(intersectionPoint); | ||||
else | ||||
fullPath.lineTo(currentGeometryPoint); | ||||
} | ||||
} else { | ||||
if (previousAngle < 0.0 || currentAngle < 0.0 | ||||
|| ((previousAngle <= 180.0 && currentAngle <= 180.0) | ||||
&& ((previousGeometryPoint.x() < rightMarginLine && previousGeometryPoint.y() < horizontal) | ||||
|| (currentGeometryPoint.x() < rightMarginLine && currentGeometryPoint.y() < horizontal)))) { | ||||
currentSegmentPath = &linePathRight; | ||||
} else if (previousAngle > 360.0 || currentAngle > 360.0 | ||||
|| ((previousAngle > 180.0 && currentAngle > 180.0) | ||||
&& ((previousGeometryPoint.x() > leftMarginLine && previousGeometryPoint.y() < horizontal) | ||||
|| (currentGeometryPoint.x() > leftMarginLine && currentGeometryPoint.y() < horizontal)))) { | ||||
currentSegmentPath = &linePathLeft; | ||||
} else { | ||||
currentSegmentPath = &linePath; | ||||
} | ||||
if (currentSegmentPath != previousSegmentPath) | ||||
currentSegmentPath->moveTo(previousGeometryPoint); | ||||
if (previousPointWasOffGrid) | ||||
fullPath.moveTo(intersectionPoint); | ||||
if (pointOffGrid) | ||||
fullPath.lineTo(intersectionPoint); | ||||
else | ||||
fullPath.lineTo(currentGeometryPoint); | ||||
currentSegmentPath->lineTo(currentGeometryPoint); | ||||
} | ||||
} else { | ||||
currentSegmentPath = 0; | ||||
} | ||||
previousPointWasOffGrid = pointOffGrid; | ||||
if (m_pointsVisible && !pointOffGrid && currentSeriesPoint.y() >= minY) { | ||||
linePath.addEllipse(points.at(i), size, size); | ||||
fullPath.addEllipse(points.at(i), size, size); | ||||
linePath.moveTo(points.at(i)); | ||||
fullPath.moveTo(points.at(i)); | ||||
} | ||||
previousSegmentPath = currentSegmentPath; | ||||
previousGeometryPoint = currentGeometryPoint; | ||||
} | ||||
m_linePathPolarRight = linePathRight; | ||||
m_linePathPolarLeft = linePathLeft; | ||||
// Note: This construction of m_fullpath is not perfect. The partial segments that are | ||||
// outside left/right clip regions at axis boundary still generate hover/click events, | ||||
// because shape doesn't get clipped. It doesn't seem possible to do sensibly. | ||||
} else { // not polar | ||||
linePath.moveTo(points.at(0)); | ||||
if (m_pointsVisible) { | ||||
int size = m_linePen.width(); | ||||
linePath.addEllipse(points.at(0), size, size); | ||||
linePath.moveTo(points.at(0)); | ||||
for (int i = 1; i < points.size(); i++) { | ||||
linePath.lineTo(points.at(i)); | ||||
linePath.addEllipse(points.at(i), size, size); | ||||
linePath.moveTo(points.at(i)); | ||||
} | ||||
} else { | ||||
for (int i = 1; i < points.size(); i++) | ||||
linePath.lineTo(points.at(i)); | ||||
} | ||||
fullPath = linePath; | ||||
} | ||||
Tero Ahola
|
r1823 | |||
Michal Klocek
|
r1819 | QPainterPathStroker stroker; | ||
Tero Ahola
|
r1823 | // QPainter::drawLine does not respect join styles, for example BevelJoin becomes MiterJoin. | ||
// This is why we are prepared for the "worst case" scenario, i.e. use always MiterJoin and | ||||
// multiply line width with square root of two when defining shape and bounding rectangle. | ||||
Miikka Heikkinen
|
r2483 | stroker.setWidth(margin); | ||
Tero Ahola
|
r1823 | stroker.setJoinStyle(Qt::MiterJoin); | ||
stroker.setCapStyle(Qt::SquareCap); | ||||
Michal Klocek
|
r1821 | stroker.setMiterLimit(m_linePen.miterLimit()); | ||
Tero Ahola
|
r1791 | |||
Michal Klocek
|
r1819 | prepareGeometryChange(); | ||
Tero Ahola
|
r1791 | |||
Miikka Heikkinen
|
r2483 | m_linePath = linePath; | ||
m_fullPath = fullPath; | ||||
m_shapePath = stroker.createStroke(fullPath); | ||||
m_rect = m_shapePath.boundingRect(); | ||||
Michal Klocek
|
r21 | } | ||
Michal Klocek
|
r392 | void LineChartItem::handleUpdated() | ||
Michal Klocek
|
r389 | { | ||
Miikka Heikkinen
|
r2451 | // If points visiblity has changed, a geometry update is needed. | ||
// Also, if pen changes when points are visible, geometry update is needed. | ||||
bool doGeometryUpdate = | ||||
(m_pointsVisible != m_series->pointsVisible()) | ||||
|| (m_series->pointsVisible() && (m_linePen != m_series->pen())); | ||||
Tero Ahola
|
r1346 | setVisible(m_series->isVisible()); | ||
Tero Ahola
|
r2067 | setOpacity(m_series->opacity()); | ||
Michal Klocek
|
r544 | m_pointsVisible = m_series->pointsVisible(); | ||
m_linePen = m_series->pen(); | ||||
Miikka Heikkinen
|
r2451 | if (doGeometryUpdate) | ||
updateGeometry(); | ||||
Michal Klocek
|
r139 | update(); | ||
} | ||||
Michal Klocek
|
r391 | void LineChartItem::paint(QPainter *painter, const QStyleOptionGraphicsItem *option, QWidget *widget) | ||
{ | ||||
Tero Ahola
|
r611 | Q_UNUSED(widget) | ||
Q_UNUSED(option) | ||||
Miikka Heikkinen
|
r2483 | QRectF clipRect = QRectF(QPointF(0, 0), domain()->size()); | ||
Tero Ahola
|
r1828 | painter->save(); | ||
Tero Ahola
|
r1346 | painter->setPen(m_linePen); | ||
Miikka Heikkinen
|
r2483 | bool alwaysUsePath = false; | ||
if (m_series->chart()->chartType() == QChart::ChartTypePolar) { | ||||
qreal halfWidth = domain()->size().width() / 2.0; | ||||
QRectF clipRectLeft = QRectF(0, 0, halfWidth, domain()->size().height()); | ||||
QRectF clipRectRight = QRectF(halfWidth, 0, halfWidth, domain()->size().height()); | ||||
QRegion fullPolarClipRegion(clipRect.toRect(), QRegion::Ellipse); | ||||
QRegion clipRegionLeft(fullPolarClipRegion.intersected(clipRectLeft.toRect())); | ||||
QRegion clipRegionRight(fullPolarClipRegion.intersected(clipRectRight.toRect())); | ||||
painter->setClipRegion(clipRegionLeft); | ||||
painter->drawPath(m_linePathPolarLeft); | ||||
painter->setClipRegion(clipRegionRight); | ||||
painter->drawPath(m_linePathPolarRight); | ||||
painter->setClipRegion(fullPolarClipRegion); | ||||
alwaysUsePath = true; // required for proper clipping | ||||
} else { | ||||
painter->setClipRect(clipRect); | ||||
} | ||||
Michal Klocek
|
r1819 | |||
if (m_pointsVisible) { | ||||
Miikka Heikkinen
|
r2457 | painter->setBrush(m_linePen.color()); | ||
Michal Klocek
|
r1819 | painter->drawPath(m_linePath); | ||
Jani Honkonen
|
r2097 | } else { | ||
Miikka Heikkinen
|
r2457 | painter->setBrush(QBrush(Qt::NoBrush)); | ||
Miikka Heikkinen
|
r2483 | if (m_linePen.style() != Qt::SolidLine || alwaysUsePath) { | ||
Miikka Heikkinen
|
r2457 | // If pen style is not solid line, always fall back to path painting | ||
// to ensure proper continuity of the pattern | ||||
painter->drawPath(m_linePath); | ||||
} else { | ||||
for (int i(1); i < m_points.size(); i++) | ||||
painter->drawLine(m_points.at(i - 1), m_points.at(i)); | ||||
} | ||||
Michal Klocek
|
r544 | } | ||
Miikka Heikkinen
|
r2457 | |||
Tero Ahola
|
r1828 | painter->restore(); | ||
Michal Klocek
|
r391 | } | ||
Michal Klocek
|
r1218 | void LineChartItem::mousePressEvent(QGraphicsSceneMouseEvent *event) | ||
{ | ||||
Michal Klocek
|
r2273 | emit XYChart::clicked(domain()->calculateDomainPoint(event->pos())); | ||
Michal Klocek
|
r1747 | QGraphicsItem::mousePressEvent(event); | ||
Michal Klocek
|
r1218 | } | ||
Marek Rosa
|
r2255 | void LineChartItem::hoverEnterEvent(QGraphicsSceneHoverEvent *event) | ||
{ | ||||
Michal Klocek
|
r2273 | emit XYChart::hovered(domain()->calculateDomainPoint(event->pos()), true); | ||
Marek Rosa
|
r2356 | // event->accept(); | ||
QGraphicsItem::hoverEnterEvent(event); | ||||
Marek Rosa
|
r2255 | } | ||
void LineChartItem::hoverLeaveEvent(QGraphicsSceneHoverEvent *event) | ||||
{ | ||||
Michal Klocek
|
r2273 | emit XYChart::hovered(domain()->calculateDomainPoint(event->pos()), false); | ||
Marek Rosa
|
r2356 | // event->accept(); | ||
QGraphicsItem::hoverEnterEvent(event); | ||||
Marek Rosa
|
r2255 | } | ||
Michal Klocek
|
r144 | #include "moc_linechartitem_p.cpp" | ||
Michal Klocek
|
r131 | |||
Tero Ahola
|
r30 | QTCOMMERCIALCHART_END_NAMESPACE | ||