xychartitem.cpp
249 lines
| 8.2 KiB
| text/x-c
|
CppLexer
Jani Honkonen
|
r794 | /**************************************************************************** | ||
** | ||||
** Copyright (C) 2012 Digia Plc | ||||
** 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
|
r466 | #include "xychartitem_p.h" | ||
#include "qxyseries.h" | ||||
Michal Klocek
|
r938 | #include "qxyseries_p.h" | ||
Michal Klocek
|
r466 | #include "chartpresenter_p.h" | ||
Michal Klocek
|
r530 | #include "chartanimator_p.h" | ||
Michal Klocek
|
r466 | #include <QPainter> | ||
Michal Klocek
|
r571 | #include <QGraphicsSceneMouseEvent> | ||
Marek Rosa
|
r1055 | #include <QAbstractItemModel> | ||
Michal Klocek
|
r466 | |||
QTCOMMERCIALCHART_BEGIN_NAMESPACE | ||||
//TODO: optimize : remove points which are not visible | ||||
sauimone
|
r743 | XYChartItem::XYChartItem(QXYSeries *series, ChartPresenter *presenter):ChartItem(presenter), | ||
m_minX(0), | ||||
m_maxX(0), | ||||
m_minY(0), | ||||
m_maxY(0), | ||||
m_series(series) | ||||
Michal Klocek
|
r466 | { | ||
Michal Klocek
|
r938 | connect(series->d_func(),SIGNAL(pointReplaced(int)),this,SLOT(handlePointReplaced(int))); | ||
connect(series->d_func(),SIGNAL(pointAdded(int)),this,SLOT(handlePointAdded(int))); | ||||
Marek Rosa
|
r1054 | connect(series->d_func(),SIGNAL(pointsAdded(int, int)),this,SLOT(handlePointsAdded(int, int))); | ||
Michal Klocek
|
r938 | connect(series->d_func(),SIGNAL(pointRemoved(int)),this,SLOT(handlePointRemoved(int))); | ||
Marek Rosa
|
r1054 | connect(series->d_func(),SIGNAL(pointsRemoved(int, int)),this,SLOT(handlePointsRemoved(int, int))); | ||
Michal Klocek
|
r967 | connect(this,SIGNAL(clicked(QPointF)),series,SIGNAL(clicked(QPointF))); | ||
Michal Klocek
|
r466 | } | ||
sauimone
|
r743 | QPointF XYChartItem::calculateGeometryPoint(const QPointF &point) const | ||
Michal Klocek
|
r466 | { | ||
const qreal deltaX = m_size.width()/(m_maxX-m_minX); | ||||
const qreal deltaY = m_size.height()/(m_maxY-m_minY); | ||||
qreal x = (point.x() - m_minX)* deltaX; | ||||
qreal y = (point.y() - m_minY)*-deltaY + m_size.height(); | ||||
return QPointF(x,y); | ||||
} | ||||
QPointF XYChartItem::calculateGeometryPoint(int index) const | ||||
{ | ||||
const qreal deltaX = m_size.width()/(m_maxX-m_minX); | ||||
const qreal deltaY = m_size.height()/(m_maxY-m_minY); | ||||
Michal Klocek
|
r1057 | const QList<QPointF>& vector = m_series->points(); | ||
qreal x = (vector[index].x() - m_minX)* deltaX; | ||||
qreal y = (vector[index].y() - m_minY)*-deltaY + m_size.height(); | ||||
Michal Klocek
|
r466 | return QPointF(x,y); | ||
} | ||||
QVector<QPointF> XYChartItem::calculateGeometryPoints() const | ||||
{ | ||||
const qreal deltaX = m_size.width()/(m_maxX-m_minX); | ||||
const qreal deltaY = m_size.height()/(m_maxY-m_minY); | ||||
Michal Klocek
|
r1057 | QVector<QPointF> result; | ||
result.resize(m_series->count()); | ||||
const QList<QPointF>& vector = m_series->points(); | ||||
Michal Klocek
|
r466 | for (int i = 0; i < m_series->count(); ++i) { | ||
Michal Klocek
|
r1057 | qreal x = (vector[i].x() - m_minX)* deltaX; | ||
qreal y = (vector[i].y() - m_minY)*-deltaY + m_size.height(); | ||||
result[i].setX(x); | ||||
result[i].setY(y); | ||||
Michal Klocek
|
r466 | } | ||
Michal Klocek
|
r1057 | return result; | ||
Michal Klocek
|
r466 | } | ||
sauimone
|
r743 | QPointF XYChartItem::calculateDomainPoint(const QPointF &point) const | ||
Michal Klocek
|
r544 | { | ||
const qreal deltaX = m_size.width()/(m_maxX-m_minX); | ||||
const qreal deltaY = m_size.height()/(m_maxY-m_minY); | ||||
qreal x = point.x()/deltaX +m_minX; | ||||
qreal y = (point.y()-m_size.height())/(-deltaY)+ m_minY; | ||||
return QPointF(x,y); | ||||
} | ||||
sauimone
|
r743 | void XYChartItem::updateLayout(QVector<QPointF> &oldPoints, QVector<QPointF> &newPoints,int index) | ||
Michal Klocek
|
r466 | { | ||
sauimone
|
r743 | if (animator()) { | ||
animator()->updateLayout(this,oldPoints,newPoints,index); | ||||
} else { | ||||
Michal Klocek
|
r622 | setLayout(newPoints); | ||
Michal Klocek
|
r530 | } | ||
Michal Klocek
|
r466 | } | ||
sauimone
|
r743 | void XYChartItem::setLayout(QVector<QPointF> &points) | ||
Michal Klocek
|
r466 | { | ||
m_points = points; | ||||
Michal Klocek
|
r622 | update(); | ||
Michal Klocek
|
r466 | } | ||
//handlers | ||||
void XYChartItem::handlePointAdded(int index) | ||||
{ | ||||
Marek Rosa
|
r1054 | if (m_series->model() == 0) { | ||
Q_ASSERT(index<m_series->count()); | ||||
Q_ASSERT(index>=0); | ||||
} | ||||
Marek Rosa
|
r1018 | QVector<QPointF> points = m_points; | ||
QPointF point; | ||||
Marek Rosa
|
r990 | point = calculateGeometryPoint(index); | ||
points.insert(index, point); | ||||
updateLayout(m_points, points, index); | ||||
Michal Klocek
|
r466 | update(); | ||
} | ||||
Marek Rosa
|
r1054 | |||
void XYChartItem::handlePointsAdded(int start, int end) | ||||
{ | ||||
if (m_series->model() == 0) { | ||||
for (int i = start; i <= end; i++) | ||||
handlePointAdded(i); | ||||
Marek Rosa
|
r1060 | } else if (m_series->mapCount() != -1 && start >= m_series->mapFirst() + m_series->mapCount()) { | ||
return; | ||||
Marek Rosa
|
r1054 | } else { | ||
int addedCount = end - start + 1; | ||||
Marek Rosa
|
r1060 | if (m_series->mapCount() != -1 && addedCount > m_series->mapCount()) | ||
addedCount = m_series->mapCount(); | ||||
int first = qMax(start, m_series->mapFirst()); | ||||
Marek Rosa
|
r1065 | int last = qMin(first + addedCount - 1, m_series->count() + m_series->mapFirst() - 1); | ||
Marek Rosa
|
r1060 | for (int i = first; i <= last; i++) { | ||
handlePointAdded(i - m_series->mapFirst()); | ||||
Marek Rosa
|
r1054 | } | ||
Marek Rosa
|
r1060 | if (m_series->mapCount() != -1 && m_points.size() > m_series->mapCount()) | ||
for (int i = m_points.size() - 1; i >= m_series->mapCount(); i--) | ||||
handlePointRemoved(i); | ||||
Marek Rosa
|
r1054 | } | ||
} | ||||
Michal Klocek
|
r466 | void XYChartItem::handlePointRemoved(int index) | ||
Michal Klocek
|
r938 | { | ||
Marek Rosa
|
r1054 | if (m_series->model() == 0) { | ||
Q_ASSERT(index<m_series->count() + 1); | ||||
Q_ASSERT(index>=0); | ||||
} | ||||
Marek Rosa
|
r1018 | QVector<QPointF> points = m_points; | ||
Marek Rosa
|
r990 | points.remove(index); | ||
updateLayout(m_points, points, index); | ||||
Michal Klocek
|
r466 | update(); | ||
} | ||||
Marek Rosa
|
r1054 | void XYChartItem::handlePointsRemoved(int start, int end) | ||
{ | ||||
Q_UNUSED(start) | ||||
Q_UNUSED(end) | ||||
Marek Rosa
|
r1055 | if (m_series->model() == 0) { | ||
for (int i = end; i >= start; i--) | ||||
handlePointRemoved(i); | ||||
} else { | ||||
// series uses model as a data source | ||||
Marek Rosa
|
r1065 | int mapFirst = m_series->mapFirst(); | ||
int mapCount = m_series->mapCount(); | ||||
Marek Rosa
|
r1055 | int removedCount = end - start + 1; | ||
Marek Rosa
|
r1065 | if (mapCount != -1 && start >= mapFirst + mapCount) { | ||
Marek Rosa
|
r1055 | return; | ||
} else { | ||||
Marek Rosa
|
r1065 | int toRemove = qMin(m_points.size(), removedCount); // first find how many items can actually be removed | ||
int first = qMax(start, mapFirst); // get the index of the first item that will be removed. | ||||
int last = qMin(first + toRemove - 1, m_points.size() + mapFirst - 1); // get the index of the last item that will be removed. | ||||
for (int i = last; i >= first; i--) | ||||
handlePointRemoved(i - mapFirst); | ||||
if (mapCount != -1) { | ||||
int itemsAvailable; // check how many are available to be added | ||||
if (m_series->mapOrientation() == Qt::Vertical) | ||||
itemsAvailable = m_series->model()->rowCount() - mapFirst - m_points.size(); | ||||
else | ||||
itemsAvailable = m_series->model()->columnCount() - mapFirst - m_points.size(); | ||||
int toBeAdded = qMin(itemsAvailable, mapCount - m_points.size()); // add not more items than there is space left to be filled. | ||||
int currentSize = m_points.size(); | ||||
if (toBeAdded > 0) | ||||
for (int i = m_points.size(); i < currentSize + toBeAdded; i++) { | ||||
handlePointAdded(i); | ||||
} | ||||
} | ||||
Marek Rosa
|
r1055 | } | ||
} | ||||
Marek Rosa
|
r1054 | } | ||
Michal Klocek
|
r466 | void XYChartItem::handlePointReplaced(int index) | ||
{ | ||||
Q_ASSERT(index<m_series->count()); | ||||
Q_ASSERT(index>=0); | ||||
QPointF point = calculateGeometryPoint(index); | ||||
Michal Klocek
|
r476 | QVector<QPointF> points = m_points; | ||
Marek Rosa
|
r527 | points.replace(index,point); | ||
Michal Klocek
|
r622 | updateLayout(m_points,points,index); | ||
Michal Klocek
|
r466 | update(); | ||
} | ||||
void XYChartItem::handleDomainChanged(qreal minX, qreal maxX, qreal minY, qreal maxY) | ||||
{ | ||||
m_minX=minX; | ||||
m_maxX=maxX; | ||||
m_minY=minY; | ||||
m_maxY=maxY; | ||||
sauimone
|
r743 | if (isEmpty()) return; | ||
Michal Klocek
|
r466 | QVector<QPointF> points = calculateGeometryPoints(); | ||
Michal Klocek
|
r622 | updateLayout(m_points,points); | ||
Michal Klocek
|
r466 | update(); | ||
} | ||||
sauimone
|
r743 | void XYChartItem::handleGeometryChanged(const QRectF &rect) | ||
Michal Klocek
|
r466 | { | ||
Marek Rosa
|
r833 | Q_ASSERT(rect.isValid()); | ||
m_size=rect.size(); | ||||
m_clipRect=rect.translated(-rect.topLeft()); | ||||
setPos(rect.topLeft()); | ||||
Michal Klocek
|
r466 | |||
sauimone
|
r743 | if (isEmpty()) return; | ||
Marek Rosa
|
r833 | QVector<QPointF> points = calculateGeometryPoints(); | ||
updateLayout(m_points,points); | ||||
update(); | ||||
Michal Klocek
|
r466 | } | ||
bool XYChartItem::isEmpty() | ||||
{ | ||||
Marek Rosa
|
r833 | return !m_clipRect.isValid() || qFuzzyIsNull(m_maxX - m_minX) || qFuzzyIsNull(m_maxY - m_minY); | ||
Michal Klocek
|
r466 | } | ||
sauimone
|
r743 | void XYChartItem::mousePressEvent(QGraphicsSceneMouseEvent *event) | ||
Michal Klocek
|
r571 | { | ||
emit clicked(calculateDomainPoint(event->pos())); | ||||
} | ||||
Michal Klocek
|
r466 | #include "moc_xychartitem_p.cpp" | ||
QTCOMMERCIALCHART_END_NAMESPACE | ||||