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
|
r820 | #include "barlabel_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
|
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 | { | ||
Tero Ahola
|
r737 | connect(series, SIGNAL(showToolTip(QPoint,QString)), this, SLOT(showToolTip(QPoint,QString))); | ||
sauimone
|
r681 | connect(series, SIGNAL(updatedBars()), this, SLOT(handleLayoutChanged())); | ||
sauimone
|
r850 | connect(series, 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
|
r839 | deleteItems(); | ||
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 | |||
sauimone
|
r839 | painter->save(); | ||
painter->setClipRect(m_clipRect); | ||||
foreach (Bar *bar, m_bars) | ||||
bar->paint(painter,option,widget); | ||||
foreach (BarLabel* label, m_labels) | ||||
label->paint(painter,option,widget); | ||||
painter->restore(); | ||||
sauimone
|
r126 | } | ||
sauimone
|
r666 | QRectF BarChartItem::boundingRect() const | ||
sauimone
|
r126 | { | ||
sauimone
|
r839 | return m_rect.translated(-m_rect.topLeft()); | ||
// 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
|
r839 | deleteItems(); | ||
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++) { | ||
QString category = m_series->categoryName(c); | ||||
for (int s = 0; s < m_series->barsetCount(); s++) { | ||||
QBarSet *set = m_series->barsetAt(s); | ||||
sauimone
|
r839 | Bar *bar = new Bar(category,0); // Null parent is best :) | ||
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
|
r820 | // Create labels | ||
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
|
r839 | BarLabel *value = new BarLabel(*set, 0); | ||
sauimone
|
r820 | m_labels.append(value); | ||
connect(set,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
|
r817 | QBarSet* barSet = m_series->barsetAt(set); | ||
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(); | ||
} | ||||
sauimone
|
r666 | |||
sauimone
|
r839 | void BarChartItem::deleteItems() | ||
{ | ||||
foreach (Bar *bar, m_bars) | ||||
delete bar; | ||||
foreach (BarLabel* label, m_labels) | ||||
delete label; | ||||
} | ||||
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 | { | ||
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 | ||