axisitem.cpp
428 lines
| 11.0 KiB
| text/x-c
|
CppLexer
Michal Klocek
|
r67 | #include "axisitem_p.h" | ||
Michal Klocek
|
r140 | #include "qchartaxis.h" | ||
Michal Klocek
|
r262 | #include "chartpresenter_p.h" | ||
Michal Klocek
|
r530 | #include "chartanimator_p.h" | ||
Michal Klocek
|
r67 | #include <QPainter> | ||
#include <QDebug> | ||||
Michal Klocek
|
r678 | #include <cmath> | ||
Michal Klocek
|
r67 | |||
Michal Klocek
|
r176 | static int label_padding = 5; | ||
Michal Klocek
|
r85 | |||
Michal Klocek
|
r67 | QTCOMMERCIALCHART_BEGIN_NAMESPACE | ||
Michal Klocek
|
r677 | Axis::Axis(QChartAxis* axis,ChartPresenter* presenter,AxisType type) : | ||
Chart(presenter), | ||||
Michal Klocek
|
r439 | m_chartAxis(axis), | ||
Michal Klocek
|
r176 | m_type(type), | ||
m_labelsAngle(0), | ||||
Michal Klocek
|
r685 | m_grid(presenter->rootItem()), | ||
m_shades(presenter->rootItem()), | ||||
m_labels(presenter->rootItem()), | ||||
m_axis(presenter->rootItem()), | ||||
Michal Klocek
|
r502 | m_min(0), | ||
m_max(0), | ||||
Michal Klocek
|
r531 | m_ticksCount(0) | ||
Michal Klocek
|
r67 | { | ||
Michal Klocek
|
r145 | //initial initialization | ||
Michal Klocek
|
r685 | m_axis.setZValue(ChartPresenter::AxisZValue); | ||
m_axis.setHandlesChildEvents(false); | ||||
Michal Klocek
|
r677 | |||
Michal Klocek
|
r685 | m_shades.setZValue(ChartPresenter::ShadesZValue); | ||
m_grid.setZValue(ChartPresenter::GridZValue); | ||||
Michal Klocek
|
r439 | |||
QObject::connect(m_chartAxis,SIGNAL(updated()),this,SLOT(handleAxisUpdated())); | ||||
Michal Klocek
|
r497 | QObject::connect(m_chartAxis->categories(),SIGNAL(updated()),this,SLOT(handleAxisCategoriesUpdated())); | ||
Michal Klocek
|
r502 | |||
handleAxisUpdated(); | ||||
Michal Klocek
|
r67 | } | ||
Michal Klocek
|
r677 | Axis::~Axis() | ||
Michal Klocek
|
r67 | { | ||
} | ||||
Michal Klocek
|
r677 | void Axis::createItems(int count) | ||
Michal Klocek
|
r176 | { | ||
Michal Klocek
|
r530 | |||
Michal Klocek
|
r685 | if(m_axis.children().size()==0) | ||
m_axis.addToGroup(new AxisItem(this)); | ||||
Michal Klocek
|
r223 | for (int i = 0; i < count; ++i) { | ||
Michal Klocek
|
r685 | m_grid.addToGroup(new QGraphicsLineItem()); | ||
m_labels.addToGroup(new QGraphicsSimpleTextItem()); | ||||
m_axis.addToGroup(new QGraphicsLineItem()); | ||||
if((m_grid.childItems().size())%2 && m_grid.childItems().size()>2) m_shades.addToGroup(new QGraphicsRectItem()); | ||||
Michal Klocek
|
r176 | } | ||
} | ||||
Michal Klocek
|
r677 | void Axis::deleteItems(int count) | ||
Michal Klocek
|
r176 | { | ||
Michal Klocek
|
r685 | QList<QGraphicsItem *> lines = m_grid.childItems(); | ||
QList<QGraphicsItem *> labels = m_labels.childItems(); | ||||
QList<QGraphicsItem *> shades = m_shades.childItems(); | ||||
QList<QGraphicsItem *> axis = m_axis.childItems(); | ||||
Michal Klocek
|
r176 | |||
Michal Klocek
|
r291 | for (int i = 0; i < count; ++i) { | ||
Michal Klocek
|
r530 | if(lines.size()%2 && lines.size()>1) delete(shades.takeLast()); | ||
Michal Klocek
|
r291 | delete(lines.takeLast()); | ||
delete(labels.takeLast()); | ||||
delete(axis.takeLast()); | ||||
Michal Klocek
|
r272 | } | ||
Michal Klocek
|
r176 | } | ||
Michal Klocek
|
r677 | void Axis::updateLayout(QVector<qreal>& layout) | ||
Michal Klocek
|
r176 | { | ||
Michal Klocek
|
r677 | if(animator()){ | ||
animator()->updateLayout(this,layout); | ||||
Michal Klocek
|
r530 | } | ||
else setLayout(layout); | ||||
Michal Klocek
|
r85 | } | ||
Michal Klocek
|
r176 | |||
Michal Klocek
|
r701 | bool Axis::createLabels(QStringList& labels,qreal min, qreal max,int ticks) const | ||
Michal Klocek
|
r85 | { | ||
Michal Klocek
|
r439 | Q_ASSERT(max>=min); | ||
Michal Klocek
|
r678 | Q_ASSERT(ticks>1); | ||
Michal Klocek
|
r85 | |||
Michal Klocek
|
r497 | QChartAxisCategories* categories = m_chartAxis->categories(); | ||
Michal Klocek
|
r291 | |||
Michal Klocek
|
r701 | bool category = categories->count()>0; | ||
if(!category) { | ||||
int n = qMax(int(-floor(log10((max-min)/(ticks-1)))),0); | ||||
for(int i=0; i< ticks; i++) { | ||||
qreal value = min + (i * (max - min)/ (ticks-1)); | ||||
Michal Klocek
|
r678 | labels << QString::number(value,'f',n); | ||
Michal Klocek
|
r439 | } | ||
Michal Klocek
|
r701 | } | ||
else { | ||||
Michal Klocek
|
r706 | QList<qreal> values = categories->values(); | ||
Michal Klocek
|
r701 | for(int i=0; i< ticks; i++) { | ||
Michal Klocek
|
r706 | qreal value = (min + (i * (max - min)/ (ticks-1))); | ||
int j=0; | ||||
for(; j<values.count(); j++){ | ||||
if (values.at(j) > value) break; | ||||
} | ||||
if(j!=0) value=values.at(j-1); | ||||
Michal Klocek
|
r497 | QString label = categories->label(value); | ||
Michal Klocek
|
r439 | labels << label; | ||
} | ||||
Michal Klocek
|
r223 | } | ||
Michal Klocek
|
r701 | |||
return category; | ||||
Michal Klocek
|
r176 | } | ||
Michal Klocek
|
r85 | |||
Michal Klocek
|
r677 | void Axis::setAxisOpacity(qreal opacity) | ||
Michal Klocek
|
r184 | { | ||
Michal Klocek
|
r685 | m_axis.setOpacity(opacity); | ||
Michal Klocek
|
r184 | } | ||
Michal Klocek
|
r677 | qreal Axis::axisOpacity() const | ||
Michal Klocek
|
r184 | { | ||
Michal Klocek
|
r685 | return m_axis.opacity(); | ||
Michal Klocek
|
r184 | } | ||
Michal Klocek
|
r677 | void Axis::setGridOpacity(qreal opacity) | ||
Michal Klocek
|
r176 | { | ||
Michal Klocek
|
r685 | m_grid.setOpacity(opacity); | ||
Michal Klocek
|
r176 | } | ||
Michal Klocek
|
r85 | |||
Michal Klocek
|
r677 | qreal Axis::gridOpacity() const | ||
Michal Klocek
|
r176 | { | ||
Michal Klocek
|
r685 | return m_grid.opacity(); | ||
Michal Klocek
|
r176 | } | ||
Michal Klocek
|
r85 | |||
Michal Klocek
|
r677 | void Axis::setLabelsOpacity(qreal opacity) | ||
Michal Klocek
|
r176 | { | ||
Michal Klocek
|
r685 | m_labels.setOpacity(opacity); | ||
Michal Klocek
|
r176 | } | ||
Michal Klocek
|
r85 | |||
Michal Klocek
|
r677 | qreal Axis::labelsOpacity() const | ||
Michal Klocek
|
r176 | { | ||
Michal Klocek
|
r685 | return m_labels.opacity(); | ||
Michal Klocek
|
r176 | } | ||
Michal Klocek
|
r85 | |||
Michal Klocek
|
r677 | void Axis::setShadesOpacity(qreal opacity) | ||
Michal Klocek
|
r176 | { | ||
Michal Klocek
|
r685 | m_shades.setOpacity(opacity); | ||
Michal Klocek
|
r176 | } | ||
Michal Klocek
|
r85 | |||
Michal Klocek
|
r677 | qreal Axis::shadesOpacity() const | ||
Michal Klocek
|
r176 | { | ||
Michal Klocek
|
r685 | return m_shades.opacity(); | ||
Michal Klocek
|
r176 | } | ||
Michal Klocek
|
r67 | |||
Michal Klocek
|
r677 | void Axis::setLabelsAngle(int angle) | ||
Michal Klocek
|
r176 | { | ||
Michal Klocek
|
r685 | foreach(QGraphicsItem* item , m_labels.childItems()) { | ||
Michal Klocek
|
r176 | QPointF center = item->boundingRect().center(); | ||
item->setRotation(angle); | ||||
Michal Klocek
|
r145 | } | ||
Michal Klocek
|
r176 | |||
m_labelsAngle=angle; | ||||
Michal Klocek
|
r67 | } | ||
Michal Klocek
|
r677 | void Axis::setLabelsPen(const QPen& pen) | ||
Michal Klocek
|
r140 | { | ||
Michal Klocek
|
r685 | foreach(QGraphicsItem* item , m_labels.childItems()) { | ||
Michal Klocek
|
r176 | static_cast<QGraphicsSimpleTextItem*>(item)->setPen(pen); | ||
} | ||||
Michal Klocek
|
r140 | } | ||
Michal Klocek
|
r677 | void Axis::setLabelsBrush(const QBrush& brush) | ||
Michal Klocek
|
r140 | { | ||
Michal Klocek
|
r685 | foreach(QGraphicsItem* item , m_labels.childItems()) { | ||
Michal Klocek
|
r176 | static_cast<QGraphicsSimpleTextItem*>(item)->setBrush(brush); | ||
} | ||||
Michal Klocek
|
r140 | } | ||
Michal Klocek
|
r677 | void Axis::setLabelsFont(const QFont& font) | ||
Michal Klocek
|
r140 | { | ||
Michal Klocek
|
r685 | foreach(QGraphicsItem* item , m_labels.childItems()) { | ||
Michal Klocek
|
r176 | static_cast<QGraphicsSimpleTextItem*>(item)->setFont(font); | ||
} | ||||
Michal Klocek
|
r140 | } | ||
Michal Klocek
|
r677 | void Axis::setShadesBrush(const QBrush& brush) | ||
Michal Klocek
|
r140 | { | ||
Michal Klocek
|
r685 | foreach(QGraphicsItem* item , m_shades.childItems()) { | ||
Michal Klocek
|
r176 | static_cast<QGraphicsRectItem*>(item)->setBrush(brush); | ||
} | ||||
Michal Klocek
|
r140 | } | ||
Michal Klocek
|
r677 | void Axis::setShadesPen(const QPen& pen) | ||
Michal Klocek
|
r140 | { | ||
Michal Klocek
|
r685 | foreach(QGraphicsItem* item , m_shades.childItems()) { | ||
Michal Klocek
|
r176 | static_cast<QGraphicsRectItem*>(item)->setPen(pen); | ||
} | ||||
} | ||||
Michal Klocek
|
r677 | void Axis::setAxisPen(const QPen& pen) | ||
Michal Klocek
|
r184 | { | ||
Michal Klocek
|
r685 | foreach(QGraphicsItem* item , m_axis.childItems()) { | ||
Michal Klocek
|
r272 | static_cast<QGraphicsLineItem*>(item)->setPen(pen); | ||
} | ||||
Michal Klocek
|
r184 | } | ||
Michal Klocek
|
r677 | void Axis::setGridPen(const QPen& pen) | ||
Michal Klocek
|
r176 | { | ||
Michal Klocek
|
r685 | foreach(QGraphicsItem* item , m_grid.childItems()) { | ||
Michal Klocek
|
r176 | static_cast<QGraphicsLineItem*>(item)->setPen(pen); | ||
} | ||||
} | ||||
Michal Klocek
|
r677 | QVector<qreal> Axis::calculateLayout() const | ||
Michal Klocek
|
r291 | { | ||
Michal Klocek
|
r502 | Q_ASSERT(m_ticksCount>=2); | ||
Michal Klocek
|
r393 | QVector<qreal> points; | ||
Michal Klocek
|
r502 | points.resize(m_ticksCount); | ||
Michal Klocek
|
r393 | |||
Michal Klocek
|
r291 | switch (m_type) | ||
{ | ||||
case X_AXIS: | ||||
{ | ||||
Michal Klocek
|
r502 | const qreal deltaX = m_rect.width()/(m_ticksCount-1); | ||
for (int i = 0; i < m_ticksCount; ++i) { | ||||
Michal Klocek
|
r291 | int x = i * deltaX + m_rect.left(); | ||
Michal Klocek
|
r393 | points[i] = x; | ||
Michal Klocek
|
r291 | } | ||
} | ||||
break; | ||||
case Y_AXIS: | ||||
{ | ||||
Michal Klocek
|
r502 | const qreal deltaY = m_rect.height()/(m_ticksCount-1); | ||
for (int i = 0; i < m_ticksCount; ++i) { | ||||
Michal Klocek
|
r291 | int y = i * -deltaY + m_rect.bottom(); | ||
Michal Klocek
|
r393 | points[i] = y; | ||
Michal Klocek
|
r291 | } | ||
} | ||||
break; | ||||
} | ||||
Michal Klocek
|
r393 | return points; | ||
Michal Klocek
|
r291 | } | ||
Michal Klocek
|
r677 | void Axis::setLayout(QVector<qreal>& layout) | ||
Michal Klocek
|
r291 | { | ||
Michal Klocek
|
r502 | int diff = m_layoutVector.size() - layout.size(); | ||
if(diff>0) { | ||||
deleteItems(diff); | ||||
} | ||||
else if(diff<0) { | ||||
createItems(-diff); | ||||
} | ||||
Michal Klocek
|
r513 | if(diff!=0) handleAxisUpdated(); | ||
Michal Klocek
|
r701 | QStringList ticksList; | ||
bool categories = createLabels(ticksList,m_min,m_max,layout.size()); | ||||
Michal Klocek
|
r513 | |||
Michal Klocek
|
r685 | QList<QGraphicsItem *> lines = m_grid.childItems(); | ||
QList<QGraphicsItem *> labels = m_labels.childItems(); | ||||
QList<QGraphicsItem *> shades = m_shades.childItems(); | ||||
QList<QGraphicsItem *> axis = m_axis.childItems(); | ||||
Michal Klocek
|
r502 | |||
Michal Klocek
|
r513 | Q_ASSERT(labels.size() == ticksList.size()); | ||
Q_ASSERT(layout.size() == ticksList.size()); | ||||
Michal Klocek
|
r502 | |||
switch (m_type) | ||||
{ | ||||
case X_AXIS: | ||||
{ | ||||
QGraphicsLineItem *lineItem = static_cast<QGraphicsLineItem*>(axis.at(0)); | ||||
lineItem->setLine(m_rect.left(), m_rect.bottom(), m_rect.right(), m_rect.bottom()); | ||||
for (int i = 0; i < layout.size(); ++i) { | ||||
QGraphicsLineItem *lineItem = static_cast<QGraphicsLineItem*>(lines.at(i)); | ||||
lineItem->setLine(layout[i], m_rect.top(), layout[i], m_rect.bottom()); | ||||
QGraphicsSimpleTextItem *labelItem = static_cast<QGraphicsSimpleTextItem*>(labels.at(i)); | ||||
Michal Klocek
|
r701 | if(!categories){ | ||
labelItem->setText(ticksList.at(i)); | ||||
QPointF center = labelItem->boundingRect().center(); | ||||
labelItem->setTransformOriginPoint(center.x(), center.y()); | ||||
labelItem->setPos(layout[i] - center.x(), m_rect.bottom() + label_padding); | ||||
}else if(i>0){ | ||||
labelItem->setText(ticksList.at(i)); | ||||
QPointF center = labelItem->boundingRect().center(); | ||||
labelItem->setTransformOriginPoint(center.x(), center.y()); | ||||
labelItem->setPos(layout[i] - (layout[i] - layout[i-1])/2 - center.x(), m_rect.bottom() + label_padding); | ||||
} | ||||
Michal Klocek
|
r551 | if((i+1)%2 && i>1) { | ||
QGraphicsRectItem *rectItem = static_cast<QGraphicsRectItem*>(shades.at(i/2-1)); | ||||
rectItem->setRect(layout[i-1],m_rect.top(),layout[i]-layout[i-1],m_rect.height()); | ||||
Michal Klocek
|
r502 | } | ||
lineItem = static_cast<QGraphicsLineItem*>(axis.at(i+1)); | ||||
lineItem->setLine(layout[i],m_rect.bottom(),layout[i],m_rect.bottom()+5); | ||||
} | ||||
} | ||||
break; | ||||
case Y_AXIS: | ||||
{ | ||||
QGraphicsLineItem *lineItem = static_cast<QGraphicsLineItem*>(axis.at(0)); | ||||
lineItem->setLine(m_rect.left() , m_rect.top(), m_rect.left(), m_rect.bottom()); | ||||
for (int i = 0; i < layout.size(); ++i) { | ||||
QGraphicsLineItem *lineItem = static_cast<QGraphicsLineItem*>(lines.at(i)); | ||||
lineItem->setLine(m_rect.left() , layout[i], m_rect.right(), layout[i]); | ||||
QGraphicsSimpleTextItem *labelItem = static_cast<QGraphicsSimpleTextItem*>(labels.at(i)); | ||||
Michal Klocek
|
r706 | |||
if(!categories){ | ||||
labelItem->setText(ticksList.at(i)); | ||||
QPointF center = labelItem->boundingRect().center(); | ||||
labelItem->setTransformOriginPoint(center.x(), center.y()); | ||||
labelItem->setPos(m_rect.left() - labelItem->boundingRect().width() - label_padding , layout[i]-center.y()); | ||||
} else if(i>0){ | ||||
labelItem->setText(ticksList.at(i)); | ||||
QPointF center = labelItem->boundingRect().center(); | ||||
labelItem->setTransformOriginPoint(center.x(), center.y()); | ||||
labelItem->setPos(m_rect.left() - labelItem->boundingRect().width() - label_padding , layout[i] - (layout[i] - layout[i-1])/2 -center.y()); | ||||
} | ||||
Michal Klocek
|
r551 | if((i+1)%2 && i>1) { | ||
QGraphicsRectItem *rectItem = static_cast<QGraphicsRectItem*>(shades.at(i/2-1)); | ||||
Michal Klocek
|
r578 | rectItem->setRect(m_rect.left(),layout[i],m_rect.width(),layout[i-1]-layout[i]); | ||
Michal Klocek
|
r502 | } | ||
lineItem = static_cast<QGraphicsLineItem*>(axis.at(i+1)); | ||||
lineItem->setLine(m_rect.left()-5,layout[i],m_rect.left(),layout[i]); | ||||
} | ||||
} | ||||
break; | ||||
default: | ||||
qDebug()<<"Unknown axis type"; | ||||
break; | ||||
} | ||||
m_layoutVector=layout; | ||||
} | ||||
Michal Klocek
|
r291 | |||
Michal Klocek
|
r677 | bool Axis::isEmpty() | ||
Michal Klocek
|
r502 | { | ||
return m_rect.isEmpty() || m_min==m_max || m_ticksCount==0; | ||||
Michal Klocek
|
r291 | } | ||
Michal Klocek
|
r140 | |||
Michal Klocek
|
r439 | //handlers | ||
Michal Klocek
|
r677 | void Axis::handleAxisCategoriesUpdated() | ||
Michal Klocek
|
r497 | { | ||
Michal Klocek
|
r502 | if(isEmpty()) return; | ||
updateLayout(m_layoutVector); | ||||
Michal Klocek
|
r497 | } | ||
Michal Klocek
|
r677 | void Axis::handleAxisUpdated() | ||
Michal Klocek
|
r439 | { | ||
Michal Klocek
|
r497 | |||
Michal Klocek
|
r452 | if(isEmpty()) return; | ||
Michal Klocek
|
r439 | |||
if(m_chartAxis->isAxisVisible()) { | ||||
setAxisOpacity(100); | ||||
} | ||||
else { | ||||
setAxisOpacity(0); | ||||
} | ||||
Michal Klocek
|
r535 | if(m_chartAxis->isGridLineVisible()) { | ||
Michal Klocek
|
r439 | setGridOpacity(100); | ||
} | ||||
else { | ||||
setGridOpacity(0); | ||||
} | ||||
if(m_chartAxis->labelsVisible()) | ||||
{ | ||||
setLabelsOpacity(100); | ||||
} | ||||
else { | ||||
setLabelsOpacity(0); | ||||
} | ||||
if(m_chartAxis->shadesVisible()) { | ||||
setShadesOpacity(m_chartAxis->shadesOpacity()); | ||||
} | ||||
else { | ||||
setShadesOpacity(0); | ||||
} | ||||
setLabelsAngle(m_chartAxis->labelsAngle()); | ||||
setAxisPen(m_chartAxis->axisPen()); | ||||
setLabelsPen(m_chartAxis->labelsPen()); | ||||
setLabelsBrush(m_chartAxis->labelsBrush()); | ||||
setLabelsFont(m_chartAxis->labelsFont()); | ||||
Michal Klocek
|
r535 | setGridPen(m_chartAxis->gridLinePen()); | ||
Michal Klocek
|
r439 | setShadesPen(m_chartAxis->shadesPen()); | ||
setShadesBrush(m_chartAxis->shadesBrush()); | ||||
Michal Klocek
|
r452 | |||
Michal Klocek
|
r439 | } | ||
Michal Klocek
|
r677 | void Axis::handleRangeChanged(qreal min, qreal max,int tickCount) | ||
Michal Klocek
|
r439 | { | ||
Michal Klocek
|
r554 | if(min==max || tickCount<2) return; | ||
Michal Klocek
|
r452 | m_min = min; | ||
m_max = max; | ||||
Michal Klocek
|
r554 | m_ticksCount= tickCount; | ||
Michal Klocek
|
r513 | |||
Michal Klocek
|
r452 | if(isEmpty()) return; | ||
Michal Klocek
|
r502 | QVector<qreal> layout = calculateLayout(); | ||
updateLayout(layout); | ||||
Michal Klocek
|
r439 | |||
Michal Klocek
|
r452 | } | ||
Michal Klocek
|
r439 | |||
Michal Klocek
|
r677 | void Axis::handleGeometryChanged(const QRectF& rect) | ||
Michal Klocek
|
r439 | { | ||
m_rect = rect; | ||||
Michal Klocek
|
r452 | if(isEmpty()) return; | ||
Michal Klocek
|
r502 | QVector<qreal> layout = calculateLayout(); | ||
updateLayout(layout); | ||||
Michal Klocek
|
r452 | } | ||
Michal Klocek
|
r439 | |||
Michal Klocek
|
r677 | void Axis::axisSelected() | ||
Michal Klocek
|
r452 | { | ||
Michal Klocek
|
r677 | qDebug()<<"TODO axis clicked"; | ||
Michal Klocek
|
r439 | } | ||
Michal Klocek
|
r67 | //TODO "nice numbers algorithm" | ||
Michal Klocek
|
r140 | #include "moc_axisitem_p.cpp" | ||
Michal Klocek
|
r67 | |||
QTCOMMERCIALCHART_END_NAMESPACE | ||||