##// END OF EJS Templates
Added support for data from model to QBarSeries. Various fixes and small modifications to data from model support to other series
Added support for data from model to QBarSeries. Various fixes and small modifications to data from model support to other series

File last commit:

r611:e622b3a8d8a0
r630:dd8db9a3a988
Show More
axisitem.cpp
406 lines | 10.1 KiB | text/x-c | CppLexer
#include "axisitem_p.h"
#include "qchartaxis.h"
#include "chartpresenter_p.h"
#include "chartanimator_p.h"
#include <QPainter>
#include <QDebug>
static int label_padding = 5;
QTCOMMERCIALCHART_BEGIN_NAMESPACE
AxisItem::AxisItem(QChartAxis* axis,ChartPresenter* presenter,AxisType type,QGraphicsItem* parent) :
ChartItem(parent),
m_presenter(presenter),
m_chartAxis(axis),
m_type(type),
m_labelsAngle(0),
m_grid(parent),
m_shades(parent),
m_labels(parent),
m_axis(parent),
m_min(0),
m_max(0),
m_ticksCount(0)
{
//initial initialization
m_axis.setZValue(ChartPresenter::AxisZValue);
m_shades.setZValue(ChartPresenter::ShadesZValue);
m_grid.setZValue(ChartPresenter::GridZValue);
setFlags(QGraphicsItem::ItemHasNoContents);
QObject::connect(m_chartAxis,SIGNAL(updated()),this,SLOT(handleAxisUpdated()));
QObject::connect(m_chartAxis->categories(),SIGNAL(updated()),this,SLOT(handleAxisCategoriesUpdated()));
handleAxisUpdated();
}
AxisItem::~AxisItem()
{
}
QRectF AxisItem::boundingRect() const
{
return QRectF();
}
void AxisItem::createItems(int count)
{
if(m_axis.children().size()==0)
m_axis.addToGroup(new QGraphicsLineItem());
for (int i = 0; i < count; ++i) {
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());
}
}
void AxisItem::deleteItems(int count)
{
QList<QGraphicsItem *> lines = m_grid.childItems();
QList<QGraphicsItem *> labels = m_labels.childItems();
QList<QGraphicsItem *> shades = m_shades.childItems();
QList<QGraphicsItem *> axis = m_axis.childItems();
for (int i = 0; i < count; ++i) {
if(lines.size()%2 && lines.size()>1) delete(shades.takeLast());
delete(lines.takeLast());
delete(labels.takeLast());
delete(axis.takeLast());
}
}
void AxisItem::updateLayout(QVector<qreal>& layout)
{
if(m_animator){
m_animator->applyLayout(this,layout);
}
else setLayout(layout);
}
QStringList AxisItem::createLabels(int ticks, qreal min, qreal max) const
{
Q_ASSERT(max>=min);
Q_ASSERT(ticks>0);
QStringList labels;
QChartAxisCategories* categories = m_chartAxis->categories();
for(int i=0; i< ticks; i++) {
qreal value = min + (i * (max - min)/ (ticks-1));
if(categories->count()==0) {
labels << QString::number(value);
}
else {
QString label = categories->label(value);
labels << label;
}
}
return labels;
}
void AxisItem::setAxisOpacity(qreal opacity)
{
m_axis.setOpacity(opacity);
}
qreal AxisItem::axisOpacity() const
{
return m_axis.opacity();
}
void AxisItem::setGridOpacity(qreal opacity)
{
m_grid.setOpacity(opacity);
}
qreal AxisItem::gridOpacity() const
{
return m_grid.opacity();
}
void AxisItem::setLabelsOpacity(qreal opacity)
{
m_labels.setOpacity(opacity);
}
qreal AxisItem::labelsOpacity() const
{
return m_labels.opacity();
}
void AxisItem::setShadesOpacity(qreal opacity)
{
m_shades.setOpacity(opacity);
}
qreal AxisItem::shadesOpacity() const
{
return m_shades.opacity();
}
void AxisItem::setLabelsAngle(int angle)
{
foreach(QGraphicsItem* item , m_labels.childItems()) {
QPointF center = item->boundingRect().center();
item->setRotation(angle);
}
m_labelsAngle=angle;
}
void AxisItem::setLabelsPen(const QPen& pen)
{
foreach(QGraphicsItem* item , m_labels.childItems()) {
static_cast<QGraphicsSimpleTextItem*>(item)->setPen(pen);
}
}
void AxisItem::setLabelsBrush(const QBrush& brush)
{
foreach(QGraphicsItem* item , m_labels.childItems()) {
static_cast<QGraphicsSimpleTextItem*>(item)->setBrush(brush);
}
}
void AxisItem::setLabelsFont(const QFont& font)
{
foreach(QGraphicsItem* item , m_labels.childItems()) {
static_cast<QGraphicsSimpleTextItem*>(item)->setFont(font);
}
}
void AxisItem::setShadesBrush(const QBrush& brush)
{
foreach(QGraphicsItem* item , m_shades.childItems()) {
static_cast<QGraphicsRectItem*>(item)->setBrush(brush);
}
}
void AxisItem::setShadesPen(const QPen& pen)
{
foreach(QGraphicsItem* item , m_shades.childItems()) {
static_cast<QGraphicsRectItem*>(item)->setPen(pen);
}
}
void AxisItem::setAxisPen(const QPen& pen)
{
foreach(QGraphicsItem* item , m_axis.childItems()) {
static_cast<QGraphicsLineItem*>(item)->setPen(pen);
}
}
void AxisItem::setGridPen(const QPen& pen)
{
foreach(QGraphicsItem* item , m_grid.childItems()) {
static_cast<QGraphicsLineItem*>(item)->setPen(pen);
}
}
QVector<qreal> AxisItem::calculateLayout() const
{
Q_ASSERT(m_ticksCount>=2);
QVector<qreal> points;
points.resize(m_ticksCount);
switch (m_type)
{
case X_AXIS:
{
const qreal deltaX = m_rect.width()/(m_ticksCount-1);
for (int i = 0; i < m_ticksCount; ++i) {
int x = i * deltaX + m_rect.left();
points[i] = x;
}
}
break;
case Y_AXIS:
{
const qreal deltaY = m_rect.height()/(m_ticksCount-1);
for (int i = 0; i < m_ticksCount; ++i) {
int y = i * -deltaY + m_rect.bottom();
points[i] = y;
}
}
break;
}
return points;
}
void AxisItem::setLayout(QVector<qreal>& layout)
{
int diff = m_layoutVector.size() - layout.size();
if(diff>0) {
deleteItems(diff);
}
else if(diff<0) {
createItems(-diff);
}
if(diff!=0) handleAxisUpdated();
QStringList ticksList = createLabels(layout.size(),m_min,m_max);
QList<QGraphicsItem *> lines = m_grid.childItems();
QList<QGraphicsItem *> labels = m_labels.childItems();
QList<QGraphicsItem *> shades = m_shades.childItems();
QList<QGraphicsItem *> axis = m_axis.childItems();
Q_ASSERT(labels.size() == ticksList.size());
Q_ASSERT(layout.size() == ticksList.size());
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));
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);
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());
}
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));
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());
if((i+1)%2 && i>1) {
QGraphicsRectItem *rectItem = static_cast<QGraphicsRectItem*>(shades.at(i/2-1));
rectItem->setRect(m_rect.left(),layout[i],m_rect.width(),layout[i-1]-layout[i]);
}
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;
}
bool AxisItem::isEmpty()
{
return m_rect.isEmpty() || m_min==m_max || m_ticksCount==0;
}
//handlers
void AxisItem::handleAxisCategoriesUpdated()
{
if(isEmpty()) return;
updateLayout(m_layoutVector);
}
void AxisItem::handleAxisUpdated()
{
if(isEmpty()) return;
if(m_chartAxis->isAxisVisible()) {
setAxisOpacity(100);
}
else {
setAxisOpacity(0);
}
if(m_chartAxis->isGridLineVisible()) {
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());
setGridPen(m_chartAxis->gridLinePen());
setShadesPen(m_chartAxis->shadesPen());
setShadesBrush(m_chartAxis->shadesBrush());
}
void AxisItem::handleRangeChanged(qreal min, qreal max,int tickCount)
{
if(min==max || tickCount<2) return;
m_min = min;
m_max = max;
m_ticksCount= tickCount;
if(isEmpty()) return;
QVector<qreal> layout = calculateLayout();
updateLayout(layout);
}
void AxisItem::handleGeometryChanged(const QRectF& rect)
{
m_rect = rect;
if(isEmpty()) return;
QVector<qreal> layout = calculateLayout();
updateLayout(layout);
}
//painter
void AxisItem::paint(QPainter *painter, const QStyleOptionGraphicsItem *option, QWidget *widget)
{
Q_UNUSED(painter)
Q_UNUSED(option)
Q_UNUSED(widget)
}
//TODO "nice numbers algorithm"
#include "moc_axisitem_p.cpp"
QTCOMMERCIALCHART_END_NAMESPACE