barchartitem.cpp
280 lines
| 7.6 KiB
| text/x-c
|
CppLexer
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" | ||
sauimone
|
r126 | #include <QDebug> | ||
sauimone
|
r288 | #include <QToolTip> | ||
sauimone
|
r126 | |||
QTCOMMERCIALCHART_BEGIN_NAMESPACE | ||||
Michal Klocek
|
r677 | BarChartItem::BarChartItem(QBarSeries *series, ChartPresenter *presenter) : | ||
ChartItem(presenter), | ||||
Jani Honkonen
|
r609 | mHeight(0), | ||
mWidth(0), | ||||
Tero Ahola
|
r518 | mLayoutSet(false), | ||
Michal Klocek
|
r677 | mSeries(series) | ||
sauimone
|
r126 | { | ||
sauimone
|
r288 | connect(series,SIGNAL(showToolTip(QPoint,QString)),this,SLOT(showToolTip(QPoint,QString))); | ||
sauimone
|
r666 | connect(series, SIGNAL(updatedBars()), this, SLOT(layoutChanged())); | ||
sauimone
|
r671 | //TODO: connect(series,SIGNAL("position or size has changed"), this, SLOT(handleLayoutChanged())); | ||
connect(series, SIGNAL(restructuredBar(int)), this, SLOT(handleModelChanged(int))); | ||||
sauimone
|
r594 | setZValue(ChartPresenter::BarSeriesZValue); | ||
sauimone
|
r487 | initAxisLabels(); | ||
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 | { | ||
if (!mLayoutSet) { | ||||
sauimone
|
r666 | qDebug() << "BarChartItem::paint called without layout set. Aborting."; | ||
sauimone
|
r126 | return; | ||
} | ||||
sauimone
|
r473 | foreach(QGraphicsItem* i, childItems()) { | ||
i->paint(painter,option,widget); | ||||
} | ||||
sauimone
|
r126 | } | ||
sauimone
|
r666 | QRectF BarChartItem::boundingRect() const | ||
sauimone
|
r126 | { | ||
Tero Ahola
|
r518 | return QRectF(0, 0, mWidth, mHeight); | ||
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 | ||
foreach (QGraphicsItem* item, childItems()) { | ||||
delete item; | ||||
} | ||||
sauimone
|
r256 | mBars.clear(); | ||
sauimone
|
r263 | mFloatingValues.clear(); | ||
sauimone
|
r256 | |||
sauimone
|
r126 | // Create new graphic items for bars | ||
sauimone
|
r366 | for (int c=0; c<mSeries->categoryCount(); c++) { | ||
sauimone
|
r425 | QString category = mSeries->categoryName(c); | ||
sauimone
|
r366 | for (int s=0; s<mSeries->barsetCount(); s++) { | ||
sauimone
|
r357 | QBarSet *set = mSeries->barsetAt(s); | ||
sauimone
|
r425 | Bar *bar = new Bar(category,this); | ||
sauimone
|
r239 | childItems().append(bar); | ||
sauimone
|
r256 | mBars.append(bar); | ||
sauimone
|
r425 | connect(bar,SIGNAL(clicked(QString)),set,SIGNAL(clicked(QString))); | ||
connect(bar,SIGNAL(rightClicked(QString)),set,SIGNAL(rightClicked(QString))); | ||||
sauimone
|
r412 | connect(bar,SIGNAL(hoverEntered(QPoint)),set,SLOT(barHoverEnterEvent(QPoint))); | ||
connect(bar,SIGNAL(hoverLeaved()),set,SLOT(barHoverLeaveEvent())); | ||||
sauimone
|
r239 | } | ||
sauimone
|
r126 | } | ||
sauimone
|
r654 | |||
sauimone
|
r263 | // Create floating values | ||
sauimone
|
r366 | for (int category=0; category<mSeries->categoryCount(); category++) { | ||
for (int s=0; s<mSeries->barsetCount(); s++) { | ||||
sauimone
|
r357 | QBarSet *set = mSeries->barsetAt(s); | ||
sauimone
|
r267 | BarValue *value = new BarValue(*set, this); | ||
sauimone
|
r263 | childItems().append(value); | ||
mFloatingValues.append(value); | ||||
sauimone
|
r276 | connect(set,SIGNAL(toggleFloatingValues()),value,SLOT(toggleVisible())); | ||
sauimone
|
r263 | } | ||
} | ||||
sauimone
|
r126 | } | ||
sauimone
|
r666 | void BarChartItem::layoutChanged() | ||
{ | ||||
// Scale bars to new layout | ||||
// Layout for bars: | ||||
if (mSeries->barsetCount() <= 0) { | ||||
qDebug() << "No sets in model!"; | ||||
return; | ||||
} | ||||
if (childItems().count() == 0) { | ||||
qDebug() << "WARNING: BarChartitem::layoutChanged called before graphics items are created!"; | ||||
return; | ||||
} | ||||
// Use temporary qreals for accurancy (we might get some compiler warnings... :) | ||||
int categoryCount = mSeries->categoryCount(); | ||||
int setCount = mSeries->barsetCount(); | ||||
qreal tW = mWidth; | ||||
qreal tH = mHeight; | ||||
qreal tM = mSeries->max(); | ||||
sauimone
|
r674 | |||
// Domain: | ||||
if (mDomainMaxY > tM) { | ||||
tM = mDomainMaxY; | ||||
} | ||||
sauimone
|
r666 | qreal scale = (tH/tM); | ||
qreal tC = categoryCount + 1; | ||||
qreal categoryWidth = tW/tC; | ||||
mBarWidth = categoryWidth / (setCount+1); | ||||
int itemIndex(0); | ||||
for (int category=0; category < categoryCount; category++) { | ||||
qreal xPos = categoryWidth * category + categoryWidth /2 + mBarWidth/2; | ||||
qreal yPos = mHeight; | ||||
for (int set = 0; set < setCount; set++) { | ||||
qreal barHeight = mSeries->valueAt(set,category) * scale; | ||||
Bar* bar = mBars.at(itemIndex); | ||||
// TODO: width settable per bar? | ||||
bar->resize(mBarWidth, barHeight); | ||||
bar->setPen(mSeries->barsetAt(set)->pen()); | ||||
bar->setBrush(mSeries->barsetAt(set)->brush()); | ||||
bar->setPos(xPos, yPos-barHeight); | ||||
itemIndex++; | ||||
xPos += mBarWidth; | ||||
} | ||||
} | ||||
// Position floating values | ||||
itemIndex = 0; | ||||
for (int category=0; category < mSeries->categoryCount(); category++) { | ||||
qreal xPos = categoryWidth * category + categoryWidth/2 + mBarWidth; | ||||
qreal yPos = mHeight; | ||||
for (int set=0; set < mSeries->barsetCount(); set++) { | ||||
qreal barHeight = mSeries->valueAt(set,category) * scale; | ||||
BarValue* value = mFloatingValues.at(itemIndex); | ||||
QBarSet* barSet = mSeries->barsetAt(set); | ||||
value->resize(100,50); // TODO: proper layout for this. | ||||
value->setPos(xPos, yPos-barHeight/2); | ||||
value->setPen(barSet->floatingValuePen()); | ||||
if (mSeries->valueAt(set,category) != 0) { | ||||
value->setValueString(QString::number(mSeries->valueAt(set,category))); | ||||
} else { | ||||
value->setValueString(QString("")); | ||||
} | ||||
itemIndex++; | ||||
xPos += mBarWidth; | ||||
} | ||||
} | ||||
update(); | ||||
} | ||||
sauimone
|
r671 | BarLayout BarChartItem::calculateLayout() | ||
{ | ||||
BarLayout layout; | ||||
foreach(Bar* bar, mBars) { | ||||
layout.insert(bar,bar->boundingRect().size()); | ||||
} | ||||
return layout; | ||||
} | ||||
void BarChartItem::applyLayout(const BarLayout &layout) | ||||
{ | ||||
if (m_animator) | ||||
m_animator->updateLayout(this, layout); | ||||
else | ||||
setLayout(layout); | ||||
} | ||||
void BarChartItem::setLayout(const BarLayout &layout) | ||||
{ | ||||
foreach (Bar *bar, layout.keys()) { | ||||
bar->setSize(layout.value(bar)); | ||||
} | ||||
update(); | ||||
} | ||||
sauimone
|
r666 | |||
void BarChartItem::initAxisLabels() | ||||
sauimone
|
r487 | { | ||
int count = mSeries->categoryCount(); | ||||
if (0 == count) { | ||||
return; | ||||
} | ||||
Michal Klocek
|
r502 | mChart->axisX()->setTicksCount(count+2); | ||
sauimone
|
r487 | |||
qreal min = 0; | ||||
Michal Klocek
|
r502 | qreal max = count+1; | ||
sauimone
|
r487 | |||
mChart->axisX()->setMin(min); | ||||
mChart->axisX()->setMax(max); | ||||
Michal Klocek
|
r502 | |||
Michal Klocek
|
r497 | QChartAxisCategories* categories = mChart->axisX()->categories(); | ||
categories->clear(); | ||||
Michal Klocek
|
r502 | for (int i=0; i<count; i++) { | ||
categories->insert(i+1,mSeries->categoryName(i)); | ||||
sauimone
|
r487 | } | ||
Michal Klocek
|
r502 | |||
sauimone
|
r487 | mChart->axisX()->setLabelsVisible(true); | ||
} | ||||
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
|
r674 | mDomainMinX = minX; | ||
mDomainMaxX = maxX; | ||||
mDomainMinY = minY; | ||||
mDomainMaxY = maxY; | ||||
layoutChanged(); | ||||
Tero Ahola
|
r611 | |||
sauimone
|
r487 | /* | ||
int count = mSeries->categoryCount(); | ||||
if (0 == count) { | ||||
return; | ||||
} | ||||
// Position labels to domain | ||||
qreal min = domain.minX(); | ||||
qreal max = domain.maxX(); | ||||
qreal step = (max-min)/count; | ||||
QChartAxisCategories& categories = mChart->axisX()->categories(); | ||||
categories.clear(); | ||||
for (int i=0; i<count; i++) { | ||||
categories.insert(min,mSeries->categoryName(i)); | ||||
min += step; | ||||
} | ||||
*/ | ||||
Michal Klocek
|
r139 | } | ||
sauimone
|
r666 | void BarChartItem::handleGeometryChanged(const QRectF& rect) | ||
Michal Klocek
|
r139 | { | ||
mWidth = rect.width(); | ||||
mHeight = rect.height(); | ||||
layoutChanged(); | ||||
mLayoutSet = true; | ||||
setPos(rect.topLeft()); | ||||
} | ||||
sauimone
|
r671 | void BarChartItem::handleLayoutChanged() | ||
{ | ||||
BarLayout layout = calculateLayout(); | ||||
applyLayout(layout); | ||||
update(); | ||||
} | ||||
sauimone
|
r666 | void BarChartItem::showToolTip(QPoint pos, QString tip) | ||
sauimone
|
r288 | { | ||
sauimone
|
r296 | // TODO: cool tooltip instead of default | ||
QToolTip::showText(pos,tip); | ||||
sauimone
|
r288 | } | ||
sauimone
|
r280 | |||
sauimone
|
r666 | #include "moc_barchartitem_p.cpp" | ||
Michal Klocek
|
r139 | |||
sauimone
|
r126 | QTCOMMERCIALCHART_END_NAMESPACE | ||