barchartitem.cpp
233 lines
| 6.8 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
|
r263 | #include "barvalue_p.h" | ||
sauimone
|
r240 | #include "qbarset.h" | ||
sauimone
|
r338 | #include "qbarseries.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
|
r126 | #include <QDebug> | ||
sauimone
|
r288 | #include <QToolTip> | ||
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 | { | ||
Tero Ahola
|
r737 | connect(series, SIGNAL(showToolTip(QPoint,QString)), this, SLOT(showToolTip(QPoint,QString))); | ||
sauimone
|
r681 | connect(series, SIGNAL(updatedBars()), this, SLOT(handleLayoutChanged())); | ||
sauimone
|
r671 | connect(series, SIGNAL(restructuredBar(int)), this, SLOT(handleModelChanged(int))); | ||
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
|
r763 | if (!m_layoutSet) { | ||
sauimone
|
r762 | qWarning() << "BarChartItem::paint called without layout set. Aborting."; | ||
sauimone
|
r126 | return; | ||
} | ||||
Tero Ahola
|
r737 | |||
foreach(QGraphicsItem* i, childItems()) | ||||
sauimone
|
r473 | i->paint(painter,option,widget); | ||
sauimone
|
r126 | } | ||
sauimone
|
r666 | QRectF BarChartItem::boundingRect() const | ||
sauimone
|
r126 | { | ||
Michal Klocek
|
r679 | return m_rect; | ||
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
|
r126 | // Delete old bars | ||
Tero Ahola
|
r737 | foreach (QGraphicsItem *item, childItems()) | ||
sauimone
|
r126 | delete item; | ||
sauimone
|
r763 | m_bars.clear(); | ||
sauimone
|
r813 | m_values.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++) { | ||
QString category = m_series->categoryName(c); | ||||
for (int s = 0; s < m_series->barsetCount(); s++) { | ||||
QBarSet *set = m_series->barsetAt(s); | ||||
sauimone
|
r425 | Bar *bar = new Bar(category,this); | ||
sauimone
|
r239 | childItems().append(bar); | ||
sauimone
|
r763 | m_bars.append(bar); | ||
sauimone
|
r812 | connect(bar, SIGNAL(clicked(QString,Qt::MouseButtons)), set, SIGNAL(clicked(QString,Qt::MouseButtons))); | ||
Tero Ahola
|
r737 | connect(bar, SIGNAL(hoverEntered(QPoint)), set, SLOT(barHoverEnterEvent(QPoint))); | ||
connect(bar, SIGNAL(hoverLeaved()), set, SLOT(barHoverLeaveEvent())); | ||||
sauimone
|
r763 | m_layout.append(QRectF(0, 0, 0, 0)); | ||
sauimone
|
r239 | } | ||
sauimone
|
r126 | } | ||
sauimone
|
r654 | |||
sauimone
|
r263 | // Create floating values | ||
sauimone
|
r763 | for (int category = 0; category < m_series->categoryCount(); category++) { | ||
for (int s = 0; s < m_series->barsetCount(); s++) { | ||||
QBarSet *set = m_series->barsetAt(s); | ||||
sauimone
|
r267 | BarValue *value = new BarValue(*set, this); | ||
sauimone
|
r263 | childItems().append(value); | ||
sauimone
|
r813 | m_values.append(value); | ||
connect(set,SIGNAL(valuesVisibleChanged(bool)),value,SLOT(valuesVisibleChanged(bool))); | ||||
sauimone
|
r263 | } | ||
} | ||||
sauimone
|
r126 | } | ||
sauimone
|
r810 | |||
sauimone
|
r681 | QVector<QRectF> BarChartItem::calculateLayout() | ||
sauimone
|
r671 | { | ||
sauimone
|
r681 | QVector<QRectF> layout; | ||
// Use temporary qreals for accurancy (we might get some compiler warnings... :) | ||||
sauimone
|
r763 | qreal categoryCount = m_series->categoryCount(); | ||
qreal setCount = m_series->barsetCount(); | ||||
sauimone
|
r681 | |||
sauimone
|
r682 | qreal width = geometry().width(); | ||
qreal height = geometry().height(); | ||||
sauimone
|
r763 | qreal max = m_series->max(); | ||
sauimone
|
r681 | |||
// Domain: | ||||
sauimone
|
r763 | if (m_domainMaxY > max) { | ||
max = m_domainMaxY; | ||||
sauimone
|
r681 | } | ||
Tero Ahola
|
r737 | qreal scale = (height / max); | ||
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
|
r682 | qreal yPos = height; | ||
sauimone
|
r681 | for (int set = 0; set < setCount; set++) { | ||
sauimone
|
r763 | qreal barHeight = m_series->valueAt(set, category) * scale; | ||
Bar* bar = m_bars.at(itemIndex); | ||||
sauimone
|
r681 | |||
Tero Ahola
|
r737 | QRectF rect(xPos, yPos - barHeight, barWidth, barHeight); | ||
sauimone
|
r681 | layout.append(rect); | ||
sauimone
|
r763 | bar->setPen(m_series->barsetAt(set)->pen()); | ||
bar->setBrush(m_series->barsetAt(set)->brush()); | ||||
sauimone
|
r681 | itemIndex++; | ||
sauimone
|
r682 | xPos += barWidth; | ||
sauimone
|
r681 | } | ||
} | ||||
// Position floating values | ||||
itemIndex = 0; | ||||
sauimone
|
r763 | for (int category = 0; category < m_series->categoryCount(); category++) { | ||
Michal Klocek
|
r683 | qreal xPos = categoryWidth * category + barWidth; | ||
sauimone
|
r682 | qreal yPos = height; | ||
sauimone
|
r763 | for (int set=0; set < m_series->barsetCount(); set++) { | ||
qreal barHeight = m_series->valueAt(set, category) * scale; | ||||
sauimone
|
r813 | BarValue* value = m_values.at(itemIndex); | ||
sauimone
|
r681 | |||
sauimone
|
r763 | QBarSet* barSet = m_series->barsetAt(set); | ||
sauimone
|
r681 | |||
Jani Honkonen
|
r768 | if (!qFuzzyIsNull(m_series->valueAt(set,category))) { | ||
sauimone
|
r776 | value->setText(QString::number(m_series->valueAt(set, category))); | ||
sauimone
|
r681 | } else { | ||
sauimone
|
r776 | value->setText(QString("")); | ||
sauimone
|
r681 | } | ||
sauimone
|
r811 | value->setPos(xPos, yPos-barHeight / 2); | ||
value->setPen(barSet->floatingValuePen()); | ||||
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(); | ||
} | ||||
sauimone
|
r666 | |||
Michal Klocek
|
r139 | //handlers | ||
sauimone
|
r666 | void BarChartItem::handleModelChanged(int index) | ||
Michal Klocek
|
r139 | { | ||
Tero Ahola
|
r611 | Q_UNUSED(index) | ||
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 | { | ||
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
|
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 | ||