#include "qchart.h" #include "qchartseries.h" #include "qscatterseries.h" #include "qscatterseries_p.h" #include "qxychartseries.h" #include "xylinechartitem_p.h" #include "xyplotdomain_p.h" #include "axis_p.h" #include "xygrid_p.h" #include #include QTCOMMERCIALCHART_BEGIN_NAMESPACE class QChartPrivate { public: QChartPrivate(QChart* parent): m_axisX(new Axis(parent)), m_axisY(new Axis(parent)), m_grid(new XYGrid(parent)), m_plotDataIndex(0), m_marginSize(0){} Axis* m_axisX; Axis* m_axisY; XYGrid* m_grid; QRect m_rect; QList m_series; QList m_plotDomainList; QList m_xyLineChartItems; QList m_items; int m_plotDataIndex; int m_marginSize; }; //////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// QChart::QChart(QGraphicsObject* parent) : QGraphicsObject(parent), d(new QChartPrivate(this)) { // setFlags(QGraphicsItem::ItemClipsChildrenToShape); // set axis d->m_axisY->rotate(90); } QChart::~QChart(){} QRectF QChart::boundingRect() const { return d->m_rect; } void QChart::addSeries(QChartSeries* series) { // TODO: we should check the series not already added d->m_series<type()) { case QChartSeries::SeriesTypeLine: { QXYChartSeries* xyseries = static_cast(series); XYPlotDomain domain; //TODO "nice numbers algorithm" domain.m_ticksX=4; domain.m_ticksY=4; 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); item->updateXYPlotDomain(domain); d->m_plotDomainList<m_xyLineChartItems<(series); if (scatter) { scatter->d->setParentItem(this); scene()->addItem(scatter->d); } break; } } } QChartSeries* QChart::createSeries(QList x, QList y, QChartSeries::QChartSeriesType type) { // TODO: support also other types in addition to scatter Q_ASSERT(type == QChartSeries::SeriesTypeScatter); QChartSeries *series = 0; switch (type) { case QChartSeries::SeriesTypeScatter: { QScatterSeries *scatterSeries = new QScatterSeries(x, y, this); scatterSeries->d->setParentItem(this); break; } default: break; } return series; } void QChart::setSize(const QSizeF& size) { d->m_rect = QRect(QPoint(0,0),size.toSize()); d->m_rect.adjust(margin(),margin(), -margin(), -margin()); d->m_grid->setPos(d->m_rect.topLeft()); d->m_grid->setSize(d->m_rect.size()); // TODO: calculate the scale // TODO: calculate the origo // TODO: not sure if emitting a signal here is the best from performance point of view const qreal xscale = size.width() / 100; const qreal yscale = size.height() / 100; emit sizeChanged(QRectF(0, 0, size.width(), size.height()), xscale, yscale); for (int i(0); i < d->m_plotDomainList.size(); i++) d->m_plotDomainList[i].m_viewportRect = d->m_rect; // TODO: line chart items are updated separately as they don't support update // via sizeChanged signal foreach(XYLineChartItem* item , d->m_xyLineChartItems) item->updateXYPlotDomain(d->m_plotDomainList.at(d->m_plotDataIndex)); if (d->m_plotDomainList.count()) d->m_grid->setXYPlotData(d->m_plotDomainList.at(d->m_plotDataIndex)); update(); } int QChart::margin() const { return d->m_marginSize; } void QChart::setMargin(int margin) { d->m_marginSize = margin; } #include "moc_qchart.cpp" QTCOMMERCIALCHART_END_NAMESPACE