qlegend.cpp
491 lines
| 14.0 KiB
| text/x-c
|
CppLexer
/ src / qlegend.cpp
sauimone
|
r524 | #include "qchartglobal.h" | ||
#include "qlegend.h" | ||||
#include "qseries.h" | ||||
sauimone
|
r547 | #include "legendmarker_p.h" | ||
sauimone
|
r565 | #include "qxyseries.h" | ||
#include "qlineseries.h" | ||||
#include "qareaseries.h" | ||||
#include "qscatterseries.h" | ||||
#include "qsplineseries.h" | ||||
#include "qbarseries.h" | ||||
#include "qstackedbarseries.h" | ||||
#include "qpercentbarseries.h" | ||||
#include "qbarset.h" | ||||
#include "qpieseries.h" | ||||
#include "qpieslice.h" | ||||
sauimone
|
r626 | #include "chartpresenter_p.h" | ||
sauimone
|
r529 | #include <QPainter> | ||
#include <QPen> | ||||
sauimone
|
r524 | |||
sauimone
|
r547 | #include <QGraphicsSceneEvent> | ||
sauimone
|
r529 | |||
sauimone
|
r547 | QTCOMMERCIALCHART_BEGIN_NAMESPACE | ||
sauimone
|
r529 | |||
sauimone
|
r524 | QLegend::QLegend(QGraphicsItem *parent) | ||
: QGraphicsObject(parent) | ||||
sauimone
|
r626 | ,mPos(0,0) | ||
,mSize(0,0) | ||||
sauimone
|
r582 | ,mMinimumSize(50,20) // TODO: magic numbers | ||
sauimone
|
r616 | ,mMaximumSize(150,100) | ||
Michal Klocek
|
r645 | ,m_brush(Qt::darkGray) // TODO: from theme? | ||
sauimone
|
r616 | ,mPreferredLayout(QLegend::PreferredLayoutVertical) | ||
sauimone
|
r524 | { | ||
sauimone
|
r652 | setVisible(false); | ||
sauimone
|
r626 | setZValue(ChartPresenter::LegendZValue); | ||
sauimone
|
r524 | } | ||
void QLegend::paint(QPainter *painter, const QStyleOptionGraphicsItem *option, QWidget *widget) | ||||
{ | ||||
Tero Ahola
|
r611 | Q_UNUSED(option) | ||
Q_UNUSED(widget) | ||||
sauimone
|
r626 | painter->setOpacity(0.5); | ||
Michal Klocek
|
r645 | painter->setPen(m_pen); | ||
painter->setBrush(m_brush); | ||||
sauimone
|
r626 | painter->drawRect(boundingRect()); | ||
sauimone
|
r524 | } | ||
QRectF QLegend::boundingRect() const | ||||
{ | ||||
sauimone
|
r626 | return QRectF(mPos,mSize); | ||
sauimone
|
r524 | } | ||
Michal Klocek
|
r645 | void QLegend::setBrush(const QBrush& brush) | ||
sauimone
|
r540 | { | ||
Michal Klocek
|
r645 | if(m_brush!=brush){ | ||
m_brush = brush; | ||||
update(); | ||||
} | ||||
} | ||||
QBrush QLegend::brush() const | ||||
{ | ||||
return m_brush; | ||||
} | ||||
void QLegend::setPen(const QPen& pen) | ||||
{ | ||||
if(m_pen!=pen){ | ||||
m_pen = pen; | ||||
update(); | ||||
} | ||||
sauimone
|
r540 | } | ||
Michal Klocek
|
r645 | QPen QLegend::pen() const | ||
sauimone
|
r540 | { | ||
Michal Klocek
|
r645 | return m_pen; | ||
sauimone
|
r540 | } | ||
sauimone
|
r616 | void QLegend::setPreferredLayout(QLegend::PreferredLayout preferred) | ||
{ | ||||
mPreferredLayout = preferred; | ||||
layoutChanged(); | ||||
} | ||||
sauimone
|
r626 | QSizeF QLegend::maximumSize() const | ||
sauimone
|
r582 | { | ||
sauimone
|
r626 | return mMaximumSize; | ||
sauimone
|
r582 | } | ||
sauimone
|
r626 | void QLegend::setMaximumSize(const QSizeF size) | ||
sauimone
|
r582 | { | ||
sauimone
|
r626 | mMaximumSize = size; | ||
sauimone
|
r652 | layoutChanged(); | ||
sauimone
|
r626 | } | ||
void QLegend::setSize(const QSizeF size) | ||||
{ | ||||
mSize = size; | ||||
if (mSize.width() > mMaximumSize.width()) { | ||||
mSize.setWidth(mMaximumSize.width()); | ||||
} | ||||
if (mSize.height() > mMaximumSize.height()) { | ||||
mSize.setHeight(mMaximumSize.height()); | ||||
} | ||||
} | ||||
void QLegend::setPos(const QPointF &pos) | ||||
{ | ||||
mPos = pos; | ||||
sauimone
|
r652 | layoutChanged(); | ||
sauimone
|
r582 | } | ||
Tero Ahola
|
r611 | void QLegend::handleSeriesAdded(QSeries* series, Domain* domain) | ||
sauimone
|
r524 | { | ||
Tero Ahola
|
r611 | Q_UNUSED(domain) | ||
sauimone
|
r586 | createMarkers(series); | ||
sauimone
|
r637 | connectSeries(series); | ||
sauimone
|
r586 | layoutChanged(); | ||
} | ||||
void QLegend::handleSeriesRemoved(QSeries* series) | ||||
{ | ||||
sauimone
|
r637 | disconnectSeries(series); | ||
sauimone
|
r586 | if (series->type() == QSeries::SeriesTypeArea) | ||
{ | ||||
// This is special case. Area series has upper and lower series, which each have markers | ||||
QAreaSeries* s = static_cast<QAreaSeries*> (series); | ||||
deleteMarkers(s->upperSeries()); | ||||
deleteMarkers(s->lowerSeries()); | ||||
} else { | ||||
deleteMarkers(series); | ||||
} | ||||
layoutChanged(); | ||||
} | ||||
sauimone
|
r637 | void QLegend::handleAdded(QList<QPieSlice*> slices) | ||
{ | ||||
QPieSeries* series = static_cast<QPieSeries*> (sender()); | ||||
foreach(QPieSlice* s, slices) { | ||||
LegendMarker* marker = new LegendMarker(series,s,this); | ||||
marker->setName(s->label()); | ||||
marker->setBrush(s->sliceBrush()); | ||||
connect(marker,SIGNAL(clicked(QPieSlice*,Qt::MouseButton)),this,SIGNAL(clicked(QPieSlice*,Qt::MouseButton))); | ||||
connect(s,SIGNAL(changed()),marker,SLOT(changed())); | ||||
connect(s,SIGNAL(destroyed()),marker,SLOT(deleteLater())); | ||||
sauimone
|
r643 | connect(marker,SIGNAL(destroyed()),this,SLOT(handleMarkerDestroyed())); | ||
sauimone
|
r637 | mMarkers.append(marker); | ||
childItems().append(marker); | ||||
} | ||||
layoutChanged(); | ||||
} | ||||
sauimone
|
r643 | void QLegend::handleRemoved(QList<QPieSlice *> slices) | ||
{ | ||||
Tero Ahola
|
r647 | Q_UNUSED(slices) | ||
sauimone
|
r643 | // Propably no need to listen for this, since removed slices are deleted and we listen destroyed signal | ||
// qDebug() << "QLegend::handleRemoved(QList<QPieSlice*> slices) count:" << slices.count(); | ||||
} | ||||
sauimone
|
r637 | void QLegend::handleMarkerDestroyed() | ||
{ | ||||
// TODO: what if more than one markers are destroyed and we update layout after first one? | ||||
LegendMarker* m = static_cast<LegendMarker*> (sender()); | ||||
mMarkers.removeOne(m); | ||||
layoutChanged(); | ||||
} | ||||
void QLegend::connectSeries(QSeries *series) | ||||
{ | ||||
// Connect relevant signals from series | ||||
switch (series->type()) | ||||
{ | ||||
case QSeries::SeriesTypeLine: { | ||||
// QLineSeries* lineSeries = static_cast<QLineSeries*>(series); | ||||
break; | ||||
} | ||||
case QSeries::SeriesTypeArea: { | ||||
// QAreaSeries* areaSeries = static_cast<QAreaSeries*>(series); | ||||
break; | ||||
} | ||||
case QSeries::SeriesTypeBar: { | ||||
// QBarSeries* barSeries = static_cast<QBarSeries*>(series); | ||||
break; | ||||
} | ||||
case QSeries::SeriesTypeStackedBar: { | ||||
// QStackedBarSeries* stackedBarSeries = static_cast<QStackedBarSeries*>(series); | ||||
break; | ||||
} | ||||
case QSeries::SeriesTypePercentBar: { | ||||
// QPercentBarSeries* percentBarSeries = static_cast<QPercentBarSeries*>(series); | ||||
break; | ||||
} | ||||
case QSeries::SeriesTypeScatter: { | ||||
// QScatterSeries *scatterSeries = static_cast<QScatterSeries *>(series); | ||||
break; | ||||
} | ||||
case QSeries::SeriesTypePie: { | ||||
QPieSeries *pieSeries = static_cast<QPieSeries *>(series); | ||||
connect(pieSeries,SIGNAL(added(QList<QPieSlice*>)),this,SLOT(handleAdded(QList<QPieSlice*>))); | ||||
// connect(pieSeries,SIGNAL(removed(QList<QPieSlice*>)),this,SLOT(handleRemoved(QList<QPieSlice*>))); | ||||
break; | ||||
} | ||||
case QSeries::SeriesTypeSpline: { | ||||
// QSplineSeries* splineSeries = static_cast<QSplineSeries*>(series); | ||||
break; | ||||
} | ||||
default: { | ||||
qDebug()<< "QLegend::connectSeries" << series->type() << "not implemented."; | ||||
break; | ||||
} | ||||
} | ||||
} | ||||
void QLegend::disconnectSeries(QSeries *series) | ||||
{ | ||||
// Connect relevant signals from series | ||||
switch (series->type()) | ||||
{ | ||||
case QSeries::SeriesTypeLine: { | ||||
// QLineSeries* lineSeries = static_cast<QLineSeries*>(series); | ||||
break; | ||||
} | ||||
case QSeries::SeriesTypeArea: { | ||||
// QAreaSeries* areaSeries = static_cast<QAreaSeries*>(series); | ||||
break; | ||||
} | ||||
case QSeries::SeriesTypeBar: { | ||||
// QBarSeries* barSeries = static_cast<QBarSeries*>(series); | ||||
break; | ||||
} | ||||
case QSeries::SeriesTypeStackedBar: { | ||||
// QStackedBarSeries* stackedBarSeries = static_cast<QStackedBarSeries*>(series); | ||||
break; | ||||
} | ||||
case QSeries::SeriesTypePercentBar: { | ||||
// QPercentBarSeries* percentBarSeries = static_cast<QPercentBarSeries*>(series); | ||||
break; | ||||
} | ||||
case QSeries::SeriesTypeScatter: { | ||||
// QScatterSeries *scatterSeries = static_cast<QScatterSeries *>(series); | ||||
break; | ||||
} | ||||
case QSeries::SeriesTypePie: { | ||||
QPieSeries *pieSeries = static_cast<QPieSeries *>(series); | ||||
disconnect(pieSeries,SIGNAL(added(QList<QPieSlice*>)),this,SLOT(handleAdded(QList<QPieSlice*>))); | ||||
// disconnect(pieSeries,SIGNAL(removed(QList<QPieSlice*>)),this,SLOT(handleRemoved(QList<QPieSlice*>))); | ||||
break; | ||||
} | ||||
case QSeries::SeriesTypeSpline: { | ||||
// QSplineSeries* splineSeries = static_cast<QSplineSeries*>(series); | ||||
break; | ||||
} | ||||
default: { | ||||
qDebug()<< "QLegend::disconnectSeries" << series->type() << "not implemented."; | ||||
break; | ||||
} | ||||
} | ||||
} | ||||
sauimone
|
r586 | void QLegend::createMarkers(QSeries *series) | ||
{ | ||||
sauimone
|
r565 | switch (series->type()) | ||
{ | ||||
case QSeries::SeriesTypeLine: { | ||||
QLineSeries* lineSeries = static_cast<QLineSeries*>(series); | ||||
sauimone
|
r586 | appendMarkers(lineSeries); | ||
sauimone
|
r565 | break; | ||
} | ||||
case QSeries::SeriesTypeArea: { | ||||
QAreaSeries* areaSeries = static_cast<QAreaSeries*>(series); | ||||
sauimone
|
r586 | appendMarkers(areaSeries->upperSeries()); | ||
Michal Klocek
|
r572 | if(areaSeries->lowerSeries()) | ||
sauimone
|
r586 | appendMarkers(areaSeries->lowerSeries()); | ||
sauimone
|
r565 | break; | ||
} | ||||
case QSeries::SeriesTypeBar: { | ||||
QBarSeries* barSeries = static_cast<QBarSeries*>(series); | ||||
sauimone
|
r586 | appendMarkers(barSeries); | ||
sauimone
|
r565 | break; | ||
} | ||||
case QSeries::SeriesTypeStackedBar: { | ||||
QStackedBarSeries* stackedBarSeries = static_cast<QStackedBarSeries*>(series); | ||||
sauimone
|
r586 | appendMarkers(stackedBarSeries); | ||
sauimone
|
r565 | break; | ||
} | ||||
case QSeries::SeriesTypePercentBar: { | ||||
QPercentBarSeries* percentBarSeries = static_cast<QPercentBarSeries*>(series); | ||||
sauimone
|
r586 | appendMarkers(percentBarSeries); | ||
sauimone
|
r565 | break; | ||
} | ||||
case QSeries::SeriesTypeScatter: { | ||||
QScatterSeries *scatterSeries = static_cast<QScatterSeries *>(series); | ||||
sauimone
|
r586 | appendMarkers(scatterSeries); | ||
sauimone
|
r565 | break; | ||
} | ||||
case QSeries::SeriesTypePie: { | ||||
QPieSeries *pieSeries = static_cast<QPieSeries *>(series); | ||||
sauimone
|
r586 | appendMarkers(pieSeries); | ||
sauimone
|
r565 | break; | ||
} | ||||
case QSeries::SeriesTypeSpline: { | ||||
QSplineSeries* splineSeries = static_cast<QSplineSeries*>(series); | ||||
sauimone
|
r586 | appendMarkers(splineSeries); | ||
sauimone
|
r565 | break; | ||
} | ||||
default: { | ||||
sauimone
|
r586 | qDebug()<< "QLegend::createMarkers" << series->type() << "not implemented."; | ||
sauimone
|
r565 | break; | ||
} | ||||
} | ||||
sauimone
|
r576 | } | ||
sauimone
|
r586 | void QLegend::appendMarkers(QXYSeries* series) | ||
sauimone
|
r529 | { | ||
sauimone
|
r565 | LegendMarker* marker = new LegendMarker(series,this); | ||
marker->setName(series->name()); | ||||
marker->setBrush(series->brush()); | ||||
sauimone
|
r567 | connect(marker,SIGNAL(clicked(QSeries*,Qt::MouseButton)),this,SIGNAL(clicked(QSeries*,Qt::MouseButton))); | ||
sauimone
|
r637 | connect(marker,SIGNAL(destroyed()),this,SLOT(handleMarkerDestroyed())); | ||
sauimone
|
r565 | mMarkers.append(marker); | ||
childItems().append(marker); | ||||
} | ||||
sauimone
|
r586 | void QLegend::appendMarkers(QBarSeries *series) | ||
sauimone
|
r565 | { | ||
foreach(QBarSet* s, series->barSets()) { | ||||
sauimone
|
r576 | LegendMarker* marker = new LegendMarker(series,s,this); | ||
sauimone
|
r565 | marker->setName(s->name()); | ||
marker->setBrush(s->brush()); | ||||
sauimone
|
r567 | connect(marker,SIGNAL(clicked(QBarSet*,Qt::MouseButton)),this,SIGNAL(clicked(QBarSet*,Qt::MouseButton))); | ||
sauimone
|
r587 | connect(s,SIGNAL(changed()),marker,SLOT(changed())); | ||
sauimone
|
r637 | connect(marker,SIGNAL(destroyed()),this,SLOT(handleMarkerDestroyed())); | ||
sauimone
|
r565 | mMarkers.append(marker); | ||
childItems().append(marker); | ||||
sauimone
|
r529 | } | ||
sauimone
|
r565 | } | ||
sauimone
|
r529 | |||
sauimone
|
r586 | void QLegend::appendMarkers(QPieSeries *series) | ||
sauimone
|
r565 | { | ||
foreach(QPieSlice* s, series->slices()) { | ||||
sauimone
|
r576 | LegendMarker* marker = new LegendMarker(series,s,this); | ||
sauimone
|
r565 | marker->setName(s->label()); | ||
marker->setBrush(s->sliceBrush()); | ||||
sauimone
|
r567 | connect(marker,SIGNAL(clicked(QPieSlice*,Qt::MouseButton)),this,SIGNAL(clicked(QPieSlice*,Qt::MouseButton))); | ||
sauimone
|
r587 | connect(s,SIGNAL(changed()),marker,SLOT(changed())); | ||
sauimone
|
r637 | connect(s,SIGNAL(destroyed()),marker,SLOT(deleteLater())); | ||
connect(marker,SIGNAL(destroyed()),this,SLOT(handleMarkerDestroyed())); | ||||
sauimone
|
r565 | mMarkers.append(marker); | ||
childItems().append(marker); | ||||
sauimone
|
r529 | } | ||
} | ||||
sauimone
|
r586 | void QLegend::deleteMarkers(QSeries *series) | ||
{ | ||||
// Search all markers that belong to given series and delete them. | ||||
foreach (LegendMarker *m, mMarkers) { | ||||
if (m->series() == series) { | ||||
mMarkers.removeOne(m); | ||||
delete m; | ||||
} | ||||
} | ||||
} | ||||
sauimone
|
r529 | void QLegend::layoutChanged() | ||
{ | ||||
// Calculate layout for markers and text | ||||
if (mMarkers.count() <= 0) { | ||||
// Nothing to do | ||||
return; | ||||
} | ||||
sauimone
|
r626 | // Find out widest item. | ||
qreal itemMaxWidth = 0; | ||||
qreal itemMaxHeight = 0; | ||||
foreach (LegendMarker* m, mMarkers) { | ||||
if (m->boundingRect().width() > itemMaxWidth) { | ||||
itemMaxWidth = m->boundingRect().width(); | ||||
} | ||||
if (m->boundingRect().height() > itemMaxHeight) { | ||||
itemMaxHeight = m->boundingRect().height(); | ||||
} | ||||
} | ||||
int maxHorizontalItems = boundingRect().width() / itemMaxWidth; | ||||
int maxVerticalItems = boundingRect().height() / itemMaxHeight; | ||||
sauimone
|
r616 | |||
sauimone
|
r626 | if (mMarkers.count() > maxHorizontalItems * maxVerticalItems) { | ||
// TODO: overlapping layout | ||||
qDebug() << "Warning. Not enough space to layout all legend items properly."; | ||||
} | ||||
sauimone
|
r616 | |||
sauimone
|
r626 | qreal margin = 5; | ||
qreal totalWidth = 0; | ||||
qreal totalHeight = 0; | ||||
sauimone
|
r616 | switch (mPreferredLayout) | ||
{ | ||||
case QLegend::PreferredLayoutHorizontal: { | ||||
sauimone
|
r626 | /* | ||
qreal xStep = mMaximumSize.width() / (mMarkers.count()+1); | ||||
if (xStep > itemMaxWidth) { | ||||
xStep = itemMaxWidth; | ||||
} | ||||
qreal yStep = mMaximumSize.height() / (mMarkers.count()+1); | ||||
if (yStep > itemMaxHeight) { | ||||
yStep = itemMaxHeight; | ||||
}*/ | ||||
qreal xStep = itemMaxWidth; | ||||
qreal yStep = itemMaxHeight; | ||||
qreal x = mPos.x() + margin; | ||||
qreal y = mPos.y() + margin; | ||||
int row = 1; | ||||
int column = 0; | ||||
int maxRows = 1; | ||||
int maxColumns = 1; | ||||
sauimone
|
r616 | foreach (LegendMarker* m, mMarkers) { | ||
sauimone
|
r626 | maxRows = row; | ||
m->setPos(x,y); | ||||
sauimone
|
r616 | x += xStep; | ||
sauimone
|
r626 | column++; | ||
if (column > maxColumns) { | ||||
maxColumns = column; | ||||
} | ||||
if ((x + itemMaxWidth + margin*2) > (mPos.x() + mMaximumSize.width())) { | ||||
x = mPos.x() + margin; | ||||
y += yStep; | ||||
row++; | ||||
column = 0; | ||||
} | ||||
sauimone
|
r616 | } | ||
sauimone
|
r626 | totalWidth = maxColumns * itemMaxWidth + margin * 2; | ||
totalHeight = maxRows * itemMaxHeight + margin * 2; | ||||
sauimone
|
r616 | break; | ||
sauimone
|
r529 | } | ||
sauimone
|
r616 | case QLegend::PreferredLayoutVertical: { | ||
sauimone
|
r626 | /* | ||
qreal xStep = mMaximumSize.width() / (mMarkers.count()+1); | ||||
if (xStep > itemMaxWidth) { | ||||
xStep = itemMaxWidth; | ||||
sauimone
|
r616 | } | ||
sauimone
|
r626 | qreal yStep = mMaximumSize.height() / (mMarkers.count()+1); | ||
if (yStep > itemMaxHeight) { | ||||
yStep = itemMaxHeight; | ||||
}*/ | ||||
qreal xStep = itemMaxWidth; | ||||
qreal yStep = itemMaxHeight; | ||||
qreal x = mPos.x() + margin; | ||||
qreal y = mPos.y() + margin; | ||||
int row = 0; | ||||
int column = 1; | ||||
int maxRows = 1; | ||||
int maxColumns = 1; | ||||
sauimone
|
r616 | foreach (LegendMarker* m, mMarkers) { | ||
sauimone
|
r626 | maxColumns = column; | ||
m->setPos(x,y); | ||||
sauimone
|
r616 | y += yStep; | ||
sauimone
|
r626 | row++; | ||
if (row > maxRows) { | ||||
maxRows = row; | ||||
} | ||||
if ((y + itemMaxHeight + margin*2) > (mPos.y() + mMaximumSize.height())) { | ||||
y = mPos.y() + margin; | ||||
x += xStep; | ||||
column++; | ||||
row = 0; | ||||
} | ||||
sauimone
|
r616 | } | ||
sauimone
|
r626 | totalWidth = maxColumns * itemMaxWidth + margin * 2; | ||
totalHeight = maxRows * itemMaxHeight + margin * 2; | ||||
sauimone
|
r616 | break; | ||
} | ||||
default: { | ||||
break; | ||||
} | ||||
} | ||||
sauimone
|
r626 | mSize.setWidth(totalWidth); | ||
mSize.setHeight(totalHeight); | ||||
sauimone
|
r529 | } | ||
sauimone
|
r524 | #include "moc_qlegend.cpp" | ||
QTCOMMERCIALCHART_END_NAMESPACE | ||||