barchartitem.cpp
218 lines
| 6.6 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$ | ||||
** | ||||
****************************************************************************/ | ||||
sauimone
|
r666 | #include "barchartitem_p.h" | ||
sauimone
|
r126 | #include "bar_p.h" | ||
sauimone
|
r820 | #include "barlabel_p.h" | ||
sauimone
|
r240 | #include "qbarset.h" | ||
sauimone
|
r962 | #include "qbarset_p.h" | ||
sauimone
|
r338 | #include "qbarseries.h" | ||
sauimone
|
r962 | #include "qbarseries_p.h" | ||
sauimone
|
r487 | #include "qchart.h" | ||
#include "qchartaxis.h" | ||||
#include "qchartaxiscategories.h" | ||||
sauimone
|
r594 | #include "chartpresenter_p.h" | ||
sauimone
|
r671 | #include "chartanimator_p.h" | ||
Michal Klocek
|
r679 | #include "chartdataset_p.h" | ||
sauimone
|
r288 | #include <QToolTip> | ||
sauimone
|
r839 | #include <QPainter> | ||
sauimone
|
r126 | |||
QTCOMMERCIALCHART_BEGIN_NAMESPACE | ||||
Michal Klocek
|
r677 | BarChartItem::BarChartItem(QBarSeries *series, ChartPresenter *presenter) : | ||
ChartItem(presenter), | ||||
sauimone
|
r763 | m_layoutSet(false), | ||
m_series(series) | ||||
sauimone
|
r126 | { | ||
sauimone
|
r852 | setFlag(ItemClipsChildrenToShape); | ||
sauimone
|
r962 | connect(series->d_func(), SIGNAL(showToolTip(QPoint,QString)), this, SLOT(showToolTip(QPoint,QString))); | ||
connect(series->d_func(), SIGNAL(updatedBars()), this, SLOT(handleLayoutChanged())); | ||||
connect(series->d_func(), SIGNAL(restructuredBars()), this, SLOT(handleModelChanged())); | ||||
sauimone
|
r594 | setZValue(ChartPresenter::BarSeriesZValue); | ||
sauimone
|
r167 | dataChanged(); | ||
sauimone
|
r126 | } | ||
sauimone
|
r666 | BarChartItem::~BarChartItem() | ||
sauimone
|
r126 | { | ||
sauimone
|
r288 | disconnect(this,SLOT(showToolTip(QPoint,QString))); | ||
sauimone
|
r126 | } | ||
sauimone
|
r666 | void BarChartItem::paint(QPainter *painter, const QStyleOptionGraphicsItem *option, QWidget *widget) | ||
sauimone
|
r126 | { | ||
sauimone
|
r852 | Q_UNUSED(painter); | ||
Q_UNUSED(option); | ||||
Q_UNUSED(widget); | ||||
sauimone
|
r126 | } | ||
sauimone
|
r666 | QRectF BarChartItem::boundingRect() const | ||
sauimone
|
r126 | { | ||
sauimone
|
r839 | return m_rect.translated(-m_rect.topLeft()); | ||
sauimone
|
r126 | } | ||
sauimone
|
r165 | |||
sauimone
|
r666 | void BarChartItem::dataChanged() | ||
sauimone
|
r126 | { | ||
sauimone
|
r172 | // TODO: performance optimizations. Do we really need to delete and create items every time data is changed or can we reuse them? | ||
sauimone
|
r852 | foreach(QGraphicsItem *item, childItems()) { | ||
delete item; | ||||
} | ||||
sauimone
|
r126 | |||
sauimone
|
r763 | m_bars.clear(); | ||
sauimone
|
r820 | m_labels.clear(); | ||
sauimone
|
r763 | m_layout.clear(); | ||
sauimone
|
r256 | |||
sauimone
|
r126 | // Create new graphic items for bars | ||
sauimone
|
r763 | for (int c = 0; c < m_series->categoryCount(); c++) { | ||
sauimone
|
r962 | QString category = m_series->d_func()->categoryName(c); | ||
sauimone
|
r763 | for (int s = 0; s < m_series->barsetCount(); s++) { | ||
sauimone
|
r962 | QBarSet *set = m_series->d_func()->barsetAt(s); | ||
sauimone
|
r968 | Bar *bar = new Bar(set,category,this); | ||
sauimone
|
r763 | m_bars.append(bar); | ||
sauimone
|
r812 | connect(bar, SIGNAL(clicked(QString,Qt::MouseButtons)), set, SIGNAL(clicked(QString,Qt::MouseButtons))); | ||
sauimone
|
r968 | connect(bar, SIGNAL(clicked(QBarSet*,QString,Qt::MouseButtons)), m_series, SIGNAL(clicked(QBarSet*,QString,Qt::MouseButtons))); | ||
sauimone
|
r962 | connect(bar, SIGNAL(hoverEntered(QPoint)), set->d_ptr.data(), SLOT(barHoverEnterEvent(QPoint))); | ||
connect(bar, SIGNAL(hoverLeaved()), set->d_ptr.data(), SLOT(barHoverLeaveEvent())); | ||||
sauimone
|
r763 | m_layout.append(QRectF(0, 0, 0, 0)); | ||
sauimone
|
r239 | } | ||
sauimone
|
r126 | } | ||
sauimone
|
r654 | |||
sauimone
|
r820 | // Create labels | ||
sauimone
|
r763 | for (int category = 0; category < m_series->categoryCount(); category++) { | ||
for (int s = 0; s < m_series->barsetCount(); s++) { | ||||
sauimone
|
r962 | QBarSet *set = m_series->d_func()->barsetAt(s); | ||
sauimone
|
r852 | BarLabel *value = new BarLabel(*set, this); | ||
sauimone
|
r820 | m_labels.append(value); | ||
sauimone
|
r968 | connect(set->d_ptr.data(),SIGNAL(labelsVisibleChanged(bool)),value,SLOT(labelsVisibleChanged(bool))); | ||
sauimone
|
r263 | } | ||
} | ||||
sauimone
|
r126 | } | ||
sauimone
|
r810 | |||
sauimone
|
r681 | QVector<QRectF> BarChartItem::calculateLayout() | ||
sauimone
|
r671 | { | ||
sauimone
|
r681 | QVector<QRectF> layout; | ||
sauimone
|
r850 | // Use temporary qreals for accurancy | ||
sauimone
|
r763 | qreal categoryCount = m_series->categoryCount(); | ||
qreal setCount = m_series->barsetCount(); | ||||
sauimone
|
r681 | |||
sauimone
|
r850 | // Domain: | ||
sauimone
|
r682 | qreal width = geometry().width(); | ||
qreal height = geometry().height(); | ||||
sauimone
|
r850 | qreal range = m_domainMaxY - m_domainMinY; | ||
qreal scale = (height / range); | ||||
Tero Ahola
|
r737 | qreal categoryWidth = width / categoryCount; | ||
sauimone
|
r682 | qreal barWidth = categoryWidth / (setCount+1); | ||
sauimone
|
r681 | |||
int itemIndex(0); | ||||
Tero Ahola
|
r737 | for (int category = 0; category < categoryCount; category++) { | ||
qreal xPos = categoryWidth * category + barWidth / 2; | ||||
sauimone
|
r850 | qreal yPos = height + scale * m_domainMinY; | ||
sauimone
|
r681 | for (int set = 0; set < setCount; set++) { | ||
sauimone
|
r962 | QBarSet* barSet = m_series->d_func()->barsetAt(set); | ||
sauimone
|
r817 | |||
qreal barHeight = barSet->valueAt(category) * scale; | ||||
sauimone
|
r763 | Bar* bar = m_bars.at(itemIndex); | ||
sauimone
|
r681 | |||
Tero Ahola
|
r737 | QRectF rect(xPos, yPos - barHeight, barWidth, barHeight); | ||
sauimone
|
r681 | layout.append(rect); | ||
sauimone
|
r817 | bar->setPen(barSet->pen()); | ||
bar->setBrush(barSet->brush()); | ||||
sauimone
|
r681 | |||
sauimone
|
r820 | BarLabel* label = m_labels.at(itemIndex); | ||
sauimone
|
r681 | |||
sauimone
|
r817 | if (!qFuzzyIsNull(barSet->valueAt(category))) { | ||
sauimone
|
r820 | label->setText(QString::number(barSet->valueAt(category))); | ||
sauimone
|
r681 | } else { | ||
sauimone
|
r820 | label->setText(QString("")); | ||
sauimone
|
r681 | } | ||
sauimone
|
r820 | label->setPos(xPos + (rect.width()/2 - label->boundingRect().width()/2) | ||
,yPos - barHeight/2 - label->boundingRect().height()/2); | ||||
sauimone
|
r839 | label->setFont(barSet->labelFont()); | ||
sauimone
|
r811 | |||
sauimone
|
r681 | itemIndex++; | ||
sauimone
|
r682 | xPos += barWidth; | ||
sauimone
|
r681 | } | ||
} | ||||
sauimone
|
r671 | return layout; | ||
} | ||||
sauimone
|
r681 | void BarChartItem::applyLayout(const QVector<QRectF> &layout) | ||
sauimone
|
r671 | { | ||
Michal Klocek
|
r679 | if (animator()) | ||
sauimone
|
r763 | animator()->updateLayout(this, m_layout, layout); | ||
sauimone
|
r671 | else | ||
setLayout(layout); | ||||
} | ||||
sauimone
|
r681 | void BarChartItem::setLayout(const QVector<QRectF> &layout) | ||
sauimone
|
r671 | { | ||
sauimone
|
r763 | m_layout = layout; | ||
sauimone
|
r681 | |||
sauimone
|
r763 | for (int i=0; i < m_bars.count(); i++) | ||
m_bars.at(i)->setRect(layout.at(i)); | ||||
sauimone
|
r681 | |||
sauimone
|
r671 | update(); | ||
} | ||||
Michal Klocek
|
r139 | //handlers | ||
sauimone
|
r865 | void BarChartItem::handleModelChanged() | ||
Michal Klocek
|
r139 | { | ||
sauimone
|
r161 | dataChanged(); | ||
Michal Klocek
|
r139 | } | ||
sauimone
|
r666 | void BarChartItem::handleDomainChanged(qreal minX, qreal maxX, qreal minY, qreal maxY) | ||
Michal Klocek
|
r139 | { | ||
sauimone
|
r763 | m_domainMinX = minX; | ||
m_domainMaxX = maxX; | ||||
m_domainMinY = minY; | ||||
m_domainMaxY = maxY; | ||||
sauimone
|
r681 | handleLayoutChanged(); | ||
Michal Klocek
|
r139 | } | ||
Tero Ahola
|
r737 | void BarChartItem::handleGeometryChanged(const QRectF &rect) | ||
Michal Klocek
|
r139 | { | ||
sauimone
|
r850 | prepareGeometryChange(); | ||
sauimone
|
r839 | m_clipRect = rect.translated(-rect.topLeft()); | ||
Tero Ahola
|
r737 | m_rect = rect; | ||
sauimone
|
r682 | handleLayoutChanged(); | ||
sauimone
|
r763 | m_layoutSet = true; | ||
Michal Klocek
|
r139 | setPos(rect.topLeft()); | ||
} | ||||
sauimone
|
r671 | void BarChartItem::handleLayoutChanged() | ||
{ | ||||
sauimone
|
r850 | if ((m_rect.width() <= 0) || (m_rect.height() <= 0)) { | ||
// rect size zero. | ||||
return; | ||||
} | ||||
sauimone
|
r681 | QVector<QRectF> layout = calculateLayout(); | ||
sauimone
|
r671 | applyLayout(layout); | ||
update(); | ||||
} | ||||
sauimone
|
r666 | void BarChartItem::showToolTip(QPoint pos, QString tip) | ||
sauimone
|
r288 | { | ||
sauimone
|
r296 | // TODO: cool tooltip instead of default | ||
Tero Ahola
|
r737 | QToolTip::showText(pos, tip); | ||
sauimone
|
r288 | } | ||
sauimone
|
r280 | |||
sauimone
|
r666 | #include "moc_barchartitem_p.cpp" | ||
Michal Klocek
|
r139 | |||
sauimone
|
r126 | QTCOMMERCIALCHART_END_NAMESPACE | ||