##// END OF EJS Templates
Add zoom support...
Add zoom support * adds zoom in zoom out * adds zoomlinechart example

File last commit:

r67:8474a34cb818
r67:8474a34cb818
Show More
qchart.cpp
264 lines | 7.5 KiB | text/x-c | CppLexer
#include "qchart.h"
#include "qchartseries.h"
#include "qscatterseries.h"
#include "qscatterseries_p.h"
#include "qpieseries.h"
#include "qxychartseries.h"
#include "barchartseries.h"
#include "bargroup.h"
#include "xylinechartitem_p.h"
#include "plotdomain_p.h"
#include "axisitem_p.h"
#include <QGraphicsScene>
#include <QDebug>
QTCOMMERCIALCHART_BEGIN_NAMESPACE
QChart::QChart(QGraphicsObject* parent) : QGraphicsObject(parent),
m_axisX(new AxisItem(AxisItem::X_AXIS,this)),
m_axisY(new AxisItem(AxisItem::Y_AXIS,this)),
m_plotDataIndex(0),
m_marginSize(0)
{
// TODO: the default theme?
setTheme(QChart::ChartThemeVanilla);
// setFlags(QGraphicsItem::ItemClipsChildrenToShape);
PlotDomain domain;
m_plotDomainList<<domain;
m_chartItems<<m_axisX;
m_chartItems<<m_axisY;
}
QChart::~QChart(){}
QRectF QChart::boundingRect() const
{
return m_rect;
}
void QChart::addSeries(QChartSeries* series)
{
// TODO: we should check the series not already added
m_series<<series;
switch(series->type())
{
case QChartSeries::SeriesTypeLine: {
QXYChartSeries* xyseries = static_cast<QXYChartSeries*>(series);
// Use color defined by theme in case the series does not define a custom color
if (!xyseries->color().isValid() && m_themeColors.count())
xyseries->setColor(m_themeColors.takeFirst());
m_plotDataIndex = 0 ;
m_plotDomainList.resize(1);
PlotDomain& domain = m_plotDomainList[m_plotDataIndex];
for (int i = 0 ; i < xyseries->count() ; i++)
{
qreal x = xyseries->x(i);
qreal y = xyseries->y(i);
domain.m_minX = qMin(domain.m_minX,x);
domain.m_minY = qMin(domain.m_minY,y);
domain.m_maxX = qMax(domain.m_maxX,x);
domain.m_maxY = qMax(domain.m_maxY,y);
}
XYLineChartItem* item = new XYLineChartItem(xyseries,this);
m_chartItems<<item;
foreach(ChartItem* i ,m_chartItems)
i->setPlotDomain(m_plotDomainList.at(m_plotDataIndex));
break;
}
case QChartSeries::SeriesTypeBar: {
qDebug() << "barSeries added";
BarChartSeries* barSeries = static_cast<BarChartSeries*>(series);
// Who owns the series?
BarGroup* group = new BarGroup(*barSeries, this);
scene()->addItem(group);
m_BarGroupItems.append(group); // If we need to access group later
break;
}
case QChartSeries::SeriesTypeScatter: {
QScatterSeries *scatterSeries = qobject_cast<QScatterSeries *>(series);
connect(this, SIGNAL(sizeChanged(QRectF)),
scatterSeries, SLOT(chartSizeChanged(QRectF)));
scatterSeries->d->setParentItem(this);
QColor nextColor = m_themeColors.takeFirst();
nextColor.setAlpha(150); // TODO: default opacity?
scatterSeries->setMarkerColor(nextColor);
}
case QChartSeries::SeriesTypePie: {
// TODO: we now have also a list of y values as a parameter, it is ignored
// we should use a generic data class instead of list of x and y values
QPieSeries *pieSeries = qobject_cast<QPieSeries *>(series);
connect(this, SIGNAL(sizeChanged(QRectF)),
pieSeries, SLOT(chartSizeChanged(QRectF)));
// TODO: how to define the color for all the slices of a pie?
}
}
}
QChartSeries* QChart::createSeries(QChartSeries::QChartSeriesType type)
{
// TODO: support also other types; not only scatter and pie
QChartSeries *series(0);
switch (type) {
case QChartSeries::SeriesTypeLine: {
series = QXYChartSeries::create();
break;
}
case QChartSeries::SeriesTypeBar: {
series = new BarChartSeries(this);
break;
}
case QChartSeries::SeriesTypeScatter: {
series = new QScatterSeries(this);
break;
}
case QChartSeries::SeriesTypePie: {
series = new QPieSeries(this);
break;
}
default:
Q_ASSERT(false);
break;
}
addSeries(series);
return series;
}
void QChart::setSize(const QSize& size)
{
m_rect = QRect(QPoint(0,0),size);
QRect rect = m_rect.adjusted(margin(),margin(), -margin(), -margin());
foreach (ChartItem* item ,m_chartItems) {
item->setPos(rect.topLeft());
item->setSize(rect.size());
}
// TODO: TTD for setting scale
//emit scaleChanged(100, 100);
// TODO: calculate the origo
// TODO: not sure if emitting a signal here is the best from performance point of view
emit sizeChanged(QRectF(0, 0, size.width(), size.height()));
update();
}
int QChart::margin() const
{
return m_marginSize;
}
void QChart::setMargin(int margin)
{
m_marginSize = margin;
}
void QChart::setTheme(QChart::ChartTheme theme)
{
// TODO: define color themes
switch (theme) {
case ChartThemeVanilla:
m_themeColors.append(QColor(255, 238, 174));
m_themeColors.append(QColor(228, 228, 160));
m_themeColors.append(QColor(228, 179, 160));
m_themeColors.append(QColor(180, 151, 18));
m_themeColors.append(QColor(252, 252, 37));
break;
case ChartThemeIcy:
m_themeColors.append(QColor(255, 238, 174));
m_themeColors.append(QColor(228, 228, 160));
m_themeColors.append(QColor(228, 179, 160));
m_themeColors.append(QColor(180, 151, 18));
m_themeColors.append(QColor(252, 252, 37));
break;
case ChartThemeGrayscale:
m_themeColors.append(QColor(255, 238, 174));
m_themeColors.append(QColor(228, 228, 160));
m_themeColors.append(QColor(228, 179, 160));
m_themeColors.append(QColor(180, 151, 18));
m_themeColors.append(QColor(252, 252, 37));
break;
default:
Q_ASSERT(false);
break;
}
// TODO: update coloring of different elements to match the selected theme
}
void QChart::zoomInToRect(const QRect& rectangle)
{
if(!rectangle.isValid()) return;
qreal margin = this->margin();
QRect rect = rectangle.normalized();
rect.translate(-margin, -margin);
PlotDomain& oldDomain = m_plotDomainList[m_plotDataIndex];
PlotDomain newDomain;
qreal dx = oldDomain.spanX() / (m_rect.width() - 2 * margin);
qreal dy = oldDomain.spanY() / (m_rect.height() - 2 * margin);
newDomain.m_minX = oldDomain.m_minX + dx * rect.left();
newDomain.m_maxX = oldDomain.m_minX + dx * rect.right();
newDomain.m_minY = oldDomain.m_maxY - dy * rect.bottom();
newDomain.m_maxY = oldDomain.m_maxY - dy * rect.top();
m_plotDomainList.resize(m_plotDataIndex + 1);
m_plotDomainList<<newDomain;
m_plotDataIndex++;
foreach (ChartItem* item ,m_chartItems)
item->setPlotDomain(m_plotDomainList[m_plotDataIndex]);
update();
}
void QChart::zoomIn()
{
if (m_plotDataIndex < m_plotDomainList.count() - 1) {
m_plotDataIndex++;
foreach (ChartItem* item ,m_chartItems)
item->setPlotDomain(m_plotDomainList[m_plotDataIndex]);
update();
}else{
QRect rect = m_rect.adjusted(margin(),margin(), -margin(), -margin());
rect.setWidth(rect.width()/2);
rect.setHeight(rect.height()/2);
rect.moveCenter(m_rect.center());
zoomInToRect(rect);
}
}
void QChart::zoomOut()
{
if (m_plotDataIndex > 0) {
m_plotDataIndex--;
foreach (ChartItem* item ,m_chartItems)
item->setPlotDomain(m_plotDomainList[m_plotDataIndex]);
update();
}
}
#include "moc_qchart.cpp"
QTCOMMERCIALCHART_END_NAMESPACE