#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 "stackedbarchartseries.h" #include "stackedbargroup.h" #include "xylinechartitem_p.h" #include "plotdomain_p.h" #include "axisitem_p.h" #include #include 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::ChartThemeDefault); PlotDomain domain; m_plotDomainList<type()) { case QChartSeries::SeriesTypeLine: { QXYChartSeries* xyseries = static_cast(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<setPlotDomain(m_plotDomainList.at(m_plotDataIndex)); break; } case QChartSeries::SeriesTypeBar: { qDebug() << "barSeries added"; BarChartSeries* barSeries = static_cast(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<(series); StackedBarGroup* stackedBarGroup = new StackedBarGroup(*stackedBarSeries,this); // Add some fugly colors for 5 fist series... stackedBarGroup->addColor(QColor(255,0,0,128)); stackedBarGroup->addColor(QColor(255,255,0,128)); stackedBarGroup->addColor(QColor(0,255,0,128)); stackedBarGroup->addColor(QColor(0,0,255,128)); stackedBarGroup->addColor(QColor(255,128,0,128)); m_chartItems<(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(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::SeriesTypeStackedBar: { series = new StackedBarChartSeries(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 m_themeColors.append(QColor(QRgb(0xff000000))); m_themeColors.append(QColor(QRgb(0xff707070))); setBackground(QColor(QRgb(0xffffffff)), QColor(QRgb(0xffafafaf)), VerticalGradientOrientation); break; 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)); setBackground(QColor(QRgb(0xff9d844d)), QColor(QRgb(0xffafafaf)), VerticalGradientOrientation); 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)); setBackground(QColor(QRgb(0xffe4ffff)), QColor(QRgb(0xffe4ffff)), VerticalGradientOrientation); 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)); setBackground(QColor(QRgb(0xffffffff)), QColor(QRgb(0xffafafaf)), VerticalGradientOrientation); 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))); setBackground(QColor(QRgb(0xfff3dc9e)), QColor(QRgb(0xffafafaf)), VerticalGradientOrientation); 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(series); lineseries->setPen(nextColor()); } else if (series->type() == QChartSeries::SeriesTypeScatter) { QScatterSeries *scatter = qobject_cast(series); scatter->setMarkerColor(nextColor()); } else if (series->type() == QChartSeries::SeriesTypePie) { QPieSeries *pieSeries = qobject_cast(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<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::zoomReset() { if (m_plotDataIndex > 0) { m_plotDataIndex = 0; 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& axis) { //TODO not implemented } void QChart::setAxis(AxisItem *item, const QChartAxis& axis) { item->setVisible(axis.isAxisVisible()); } #include "moc_qchart.cpp" QTCOMMERCIALCHART_END_NAMESPACE