##// END OF EJS Templates
git acrobatics
git acrobatics

File last commit:

r82:144fd3c1acc4
r84:64dd93b1af28
Show More
qchart.cpp
345 lines | 10.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_background(new QGraphicsRectItem(this)),
m_title(new QGraphicsTextItem(this)),
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_chartSeries << 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(nextColor());
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);
BarGroup* barGroup = new BarGroup(*barSeries,this);
// Add some fugly colors for 5 fist series...
barGroup->addColor(QColor(255,0,0,128));
barGroup->addColor(QColor(255,255,0,128));
barGroup->addColor(QColor(0,255,0,128));
barGroup->addColor(QColor(0,0,255,128));
barGroup->addColor(QColor(255,128,0,128));
m_chartItems<<barGroup;
childItems().append(barGroup);
break;
}
case QChartSeries::SeriesTypeScatter: {
QScatterSeries *scatterSeries = qobject_cast<QScatterSeries *>(series);
scatterSeries->d->setParentItem(this);
// Set pre-defined colors in case the series has no colors defined
if (!scatterSeries->markerColor().isValid())
scatterSeries->setMarkerColor(nextColor());
connect(this, SIGNAL(sizeChanged(QRectF)),
scatterSeries, SLOT(chartSizeChanged(QRectF)));
// QColor nextColor = m_themeColors.takeFirst();
// nextColor.setAlpha(150); // TODO: default opacity?
// scatterSeries->setMarkerColor(nextColor);
break;
}
case QChartSeries::SeriesTypePie: {
QPieSeries *pieSeries = qobject_cast<QPieSeries *>(series);
for (int i(0); i < pieSeries->sliceCount(); i++) {
if (!pieSeries->sliceColor(i).isValid())
pieSeries->setSliceColor(i, nextColor());
}
connect(this, SIGNAL(sizeChanged(QRectF)),
pieSeries, SLOT(chartSizeChanged(QRectF)));
// Set pre-defined colors in case the series has no colors defined
// TODO: how to define the color for all the slices of a pie?
// for (int (i); i < pieSeries.sliceCount(); i++)
break;
}
}
}
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());
//recalculate background gradient
m_background->setRect(rect);
m_backgroundGradient.setFinalStop(0,m_background->rect().height());
m_background->setBrush(m_backgroundGradient);
//resize elements
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();
}
void QChart::setBackgroundColor(const QColor& color)
{
m_backgroundGradient.setColorAt( 0.0, Qt::white);
m_backgroundGradient.setColorAt( 1.0, color);
m_background->setBrush(m_backgroundGradient);
m_background->setPen(Qt::NoPen);
m_background->update();
}
void QChart::setTitle(const QString& title)
{
m_title->setPlainText(title);
}
int QChart::margin() const
{
return m_marginSize;
}
void QChart::setMargin(int margin)
{
m_marginSize = margin;
}
void QChart::setTheme(QChart::ChartThemeId theme)
{
// if (theme != m_currentTheme) {
m_themeColors.clear();
// TODO: define color themes
switch (theme) {
case QChart::ChartThemeDefault:
// TODO: define the default theme based on the OS
// For now we just fallthrough to "vanilla"
case QChart::ChartThemeVanilla:
m_themeColors.append(QColor(217, 197, 116));
m_themeColors.append(QColor(214, 168, 150));
m_themeColors.append(QColor(160, 160, 113));
m_themeColors.append(QColor(210, 210, 52));
m_themeColors.append(QColor(136, 114, 58));
m_backgroundGradient.setColorAt(0.0, QColor(QRgb(0xff9d844d)));
m_backgroundGradient.setColorAt(1.0, QColor(QRgb(0xffafafaf)));
break;
case QChart::ChartThemeIcy:
m_themeColors.append(QColor(0, 3, 165));
m_themeColors.append(QColor(49, 52, 123));
m_themeColors.append(QColor(71, 114, 187));
m_themeColors.append(QColor(48, 97, 87));
m_themeColors.append(QColor(19, 71, 90));
m_themeColors.append(QColor(110, 70, 228));
m_backgroundGradient.setColorAt(0.0, QColor(QRgb(0xffe4ffff)));
m_backgroundGradient.setColorAt(1.0, QColor(QRgb(0xffe4ffff)));
break;
case QChart::ChartThemeGrayscale:
m_themeColors.append(QColor(0, 0, 0));
m_themeColors.append(QColor(50, 50, 50));
m_themeColors.append(QColor(100, 100, 100));
m_themeColors.append(QColor(140, 140, 140));
m_themeColors.append(QColor(180, 180, 180));
m_backgroundGradient.setColorAt(0.0, QColor(QRgb(0xffffffff)));
m_backgroundGradient.setColorAt(1.0, QColor(QRgb(0xffafafaf)));
break;
case QChart::ChartThemeUnnamed1:
m_themeColors.append(QColor(QRgb(0xff3fa9f5)));
m_themeColors.append(QColor(QRgb(0xff7AC943)));
m_themeColors.append(QColor(QRgb(0xffFF931E)));
m_themeColors.append(QColor(QRgb(0xffFF1D25)));
m_themeColors.append(QColor(QRgb(0xffFF7BAC)));
m_backgroundGradient.setColorAt(0.0, QColor(QRgb(0xfff3dc9e)));
m_backgroundGradient.setColorAt(1.0, QColor(QRgb(0xffafafaf)));
break;
default:
Q_ASSERT(false);
break;
}
m_background->setBrush(m_backgroundGradient);
m_background->setPen(Qt::NoPen);
foreach(QChartSeries* series, m_chartSeries) {
// TODO: other series interested on themes?
if (series->type() == QChartSeries::SeriesTypeLine) {
QXYChartSeries *lineseries = reinterpret_cast<QXYChartSeries *>(series);
lineseries->setColor(nextColor());
} else if (series->type() == QChartSeries::SeriesTypeScatter) {
QScatterSeries *scatter = qobject_cast<QScatterSeries *>(series);
scatter->setMarkerColor(nextColor());
} else if (series->type() == QChartSeries::SeriesTypePie) {
QPieSeries *pieSeries = qobject_cast<QPieSeries *>(series);
for (int i(0); i < pieSeries->sliceCount(); i++)
pieSeries->setSliceColor(i, nextColor());
}
}
update();
}
QColor QChart::nextColor()
{
QColor nextColor = m_themeColors.first();
m_themeColors.move(0, m_themeColors.size() - 1);
return nextColor;
}
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 domain = oldDomain.subDomain(rect,m_rect.width() - 2 * margin,m_rect.height() - 2 * margin);
m_plotDomainList.resize(m_plotDataIndex + 1);
m_plotDomainList<<domain;
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