##// END OF EJS Templates
Adds title support
Adds title support

File last commit:

r87:efb0f73ff869
r87:efb0f73ff869
Show More
qchart.cpp
395 lines | 11.9 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 "qchartaxis.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_backgroundItem(0),
m_titleItem(0),
m_axisXItem(new AxisItem(AxisItem::X_AXIS,this)),
m_plotDataIndex(0),
m_marginSize(0)
{
// TODO: the default theme?
//setTheme(QChart::ChartThemeVanilla);
PlotDomain domain;
m_plotDomainList<<domain;
m_axisYItem << new AxisItem(AxisItem::Y_AXIS,this);
m_chartItems<<m_axisXItem;
m_chartItems<<m_axisYItem.at(0);
}
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->pen().color().isValid() && m_themeColors.count()) //TODO: wtf
xyseries->setPen(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());
//recaculate title
if(m_titleItem){
QPointF center = m_rect.center() -m_titleItem->boundingRect().center();
m_titleItem->setPos(center.x(),m_rect.top()/2 + margin()/2);
}
//recalculate background gradient
if(m_backgroundItem){
m_backgroundItem->setRect(rect);
if(m_bacgroundOrinetation==HorizonatlGradientOrientation)
m_backgroundGradient.setFinalStop(m_backgroundItem->rect().width(),0);
else
m_backgroundGradient.setFinalStop(0,m_backgroundItem->rect().height());
m_backgroundItem->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::setBackground(const QColor& startColor, const QColor& endColor, GradientOrientation orientation)
{
if(!m_backgroundItem){
m_backgroundItem = new QGraphicsRectItem(this);
m_backgroundItem->setZValue(-1);
}
m_bacgroundOrinetation = orientation;
m_backgroundGradient.setColorAt( 0.0, startColor);
m_backgroundGradient.setColorAt( 1.0, endColor);
m_backgroundGradient.setStart(0,0);
if(orientation == VerticalGradientOrientation){
m_backgroundGradient.setFinalStop(0,m_rect.height());
}else{
m_backgroundGradient.setFinalStop(m_rect.width(),0);
}
m_backgroundItem->setBrush(m_backgroundGradient);
m_backgroundItem->setPen(Qt::NoPen);
m_backgroundItem->update();
}
void QChart::setTitle(const QString& title,const QFont& font)
{
if(!m_titleItem) m_titleItem = new QGraphicsTextItem(this);
m_titleItem->setPlainText(title);
m_titleItem->setFont(font);
}
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;
}
if(m_backgroundItem){
m_backgroundItem->setBrush(m_backgroundGradient);
m_backgroundItem->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->setPen(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();
}
}
void QChart::setAxisX(const QChartAxis& axis)
{
setAxis(m_axisXItem,axis);
}
void QChart::setAxisY(const QChartAxis& axis)
{
setAxis(m_axisYItem.at(0),axis);
}
void QChart::setAxisY(const QList<QChartAxis>& axis)
{
//TODO not implemented
}
void QChart::setAxis(AxisItem *item, const QChartAxis& axis)
{
item->setVisible(axis.isAxisVisible());
}
#include "moc_qchart.cpp"
QTCOMMERCIALCHART_END_NAMESPACE