declarativechart.cpp
1015 lines
| 32.2 KiB
| text/x-c
|
CppLexer
Jani Honkonen
|
r830 | /**************************************************************************** | ||
** | ||||
Miikka Heikkinen
|
r2433 | ** Copyright (C) 2013 Digia Plc | ||
Jani Honkonen
|
r830 | ** All rights reserved. | ||
** For any questions to Digia, please use contact form at http://qt.digia.com | ||||
** | ||||
Miikka Heikkinen
|
r2574 | ** This file is part of the Qt Enterprise Charts Add-on. | ||
Jani Honkonen
|
r830 | ** | ||
** $QT_BEGIN_LICENSE$ | ||||
Miikka Heikkinen
|
r2574 | ** Licensees holding valid Qt Enterprise licenses may use this file in | ||
** accordance with the Qt Enterprise License Agreement provided with the | ||||
Jani Honkonen
|
r830 | ** Software or, alternatively, in accordance with the terms contained in | ||
** a written agreement between you and Digia. | ||||
** | ||||
** If you have questions regarding the use of this file, please use | ||||
** contact form at http://qt.digia.com | ||||
** $QT_END_LICENSE$ | ||||
** | ||||
****************************************************************************/ | ||||
Jani Honkonen
|
r1 | #include "declarativechart.h" | ||
Jani Honkonen
|
r660 | #include <QPainter> | ||
Tero Ahola
|
r1240 | #include "declarativelineseries.h" | ||
#include "declarativeareaseries.h" | ||||
#include "declarativebarseries.h" | ||||
#include "declarativepieseries.h" | ||||
#include "declarativesplineseries.h" | ||||
Mika Salmela
|
r2548 | #include "declarativeboxplotseries.h" | ||
Tero Ahola
|
r1240 | #include "declarativescatterseries.h" | ||
Marek Rosa
|
r1810 | #include "qbarcategoryaxis.h" | ||
Tero Ahola
|
r1813 | #include "qvalueaxis.h" | ||
Miikka Heikkinen
|
r2493 | #include "qlogvalueaxis.h" | ||
Marek Rosa
|
r1818 | #include "qcategoryaxis.h" | ||
Tero Ahola
|
r1813 | #include "qabstractseries_p.h" | ||
Tero Ahola
|
r1928 | #include "declarativemargins.h" | ||
Michal Klocek
|
r2273 | #include "chartdataset_p.h" | ||
Tero Ahola
|
r2296 | #include "declarativeaxes.h" | ||
Tero Ahola
|
r2068 | #include "qchart_p.h" | ||
Miikka Heikkinen
|
r2483 | #include "qpolarchart.h" | ||
Jani Honkonen
|
r1 | |||
Marek Rosa
|
r1867 | #ifndef QT_ON_ARM | ||
#include "qdatetimeaxis.h" | ||||
#endif | ||||
Miikka Heikkinen
|
r2495 | #ifdef CHARTS_FOR_QUICK2 | ||
#include <QGraphicsSceneMouseEvent> | ||||
#include <QGraphicsSceneHoverEvent> | ||||
#include <QApplication> | ||||
Heikkinen Miikka
|
r2505 | #include <QTimer> | ||
Miikka Heikkinen
|
r2506 | #include <QThread> | ||
Miikka Heikkinen
|
r2495 | #endif | ||
Tero Ahola
|
r120 | QTCOMMERCIALCHART_BEGIN_NAMESPACE | ||
Tero Ahola
|
r1443 | /*! | ||
\qmlclass ChartView DeclarativeChart | ||||
ChartView element is the parent that is responsible for showing different chart series types. | ||||
Tero Ahola
|
r1491 | The following QML shows how to create a simple chart with one pie series: | ||
\snippet ../examples/qmlpiechart/qml/qmlpiechart/main.qml 1 | ||||
\snippet ../examples/qmlpiechart/qml/qmlpiechart/main.qml 2 | ||||
\snippet ../examples/qmlpiechart/qml/qmlpiechart/main.qml 3 | ||||
Tero Ahola
|
r1443 | |||
\beginfloatleft | ||||
Tero Ahola
|
r1491 | \image examples_qmlpiechart.png | ||
Tero Ahola
|
r1443 | \endfloat | ||
\clearfloat | ||||
*/ | ||||
Tero Ahola
|
r1462 | /*! | ||
\qmlproperty Theme ChartView::theme | ||||
Theme defines the visual appearance of the chart, including for example colors, fonts, line | ||||
widths and chart background. | ||||
*/ | ||||
/*! | ||||
Tero Ahola
|
r2113 | \qmlproperty Animation ChartView::animationOptions | ||
Tero Ahola
|
r1462 | Animation configuration of the chart. One of ChartView.NoAnimation, ChartView.GridAxisAnimations, | ||
ChartView.SeriesAnimations or ChartView.AllAnimations. | ||||
*/ | ||||
Jani Honkonen
|
r1517 | /*! | ||
\qmlproperty Font ChartView::titleFont | ||||
The title font of the chart | ||||
See the \l {Font} {QML Font Element} for detailed documentation. | ||||
*/ | ||||
Tero Ahola
|
r1462 | /*! | ||
\qmlproperty string ChartView::title | ||||
The title of the chart, shown on top of the chart. | ||||
Tero Ahola
|
r1473 | \sa ChartView::titleColor | ||
Tero Ahola
|
r1462 | */ | ||
/*! | ||||
Tero Ahola
|
r2113 | \qmlproperty color ChartView::titleColor | ||
Tero Ahola
|
r1462 | The color of the title text. | ||
*/ | ||||
Tero Ahola
|
r1473 | /*! | ||
\qmlproperty Legend ChartView::legend | ||||
The legend of the chart. Legend lists all the series, pie slices and bar sets added on the chart. | ||||
*/ | ||||
/*! | ||||
\qmlproperty int ChartView::count | ||||
The count of series added to the chart. | ||||
*/ | ||||
/*! | ||||
\qmlproperty color ChartView::backgroundColor | ||||
The color of the chart's background. By default background color is defined by chart theme. | ||||
\sa ChartView::theme | ||||
*/ | ||||
Miikka Heikkinen
|
r2549 | /*! | ||
\qmlproperty real ChartView::backgroundRoundness | ||||
The diameter of the rounding cirle at the corners of the chart background. | ||||
*/ | ||||
Miikka Heikkinen
|
r2498 | /*! | ||
\qmlproperty color ChartView::plotAreaColor | ||||
The color of the background of the chart's plot area. By default plot area background uses chart's | ||||
background color. | ||||
\sa ChartView::backgroundColor | ||||
*/ | ||||
Tero Ahola
|
r1473 | /*! | ||
\qmlproperty bool ChartView::dropShadowEnabled | ||||
The chart's border drop shadow. Set to true to enable drop shadow. | ||||
*/ | ||||
Tero Ahola
|
r1524 | /*! | ||
\qmlproperty real ChartView::topMargin | ||||
*/ | ||||
/*! | ||||
\qmlproperty real ChartView::bottomMargin | ||||
*/ | ||||
/*! | ||||
\qmlproperty real ChartView::leftMargin | ||||
*/ | ||||
/*! | ||||
\qmlproperty real ChartView::rightMargin | ||||
Tero Ahola
|
r1928 | */ | ||
/*! | ||||
\qmlproperty Margins ChartView::minimumMargins | ||||
Tero Ahola
|
r2369 | Deprecated; use margins instead. | ||
Tero Ahola
|
r1929 | The minimum margins allowed between the outer bounds and the plotArea of the ChartView. Margins | ||
area of ChartView is used for drawing title, axes and legend. Please note that setting the | ||||
properties of minimumMargins may be bigger than the defined value, depending on other ChartView | ||||
properties that affect it's layout. If you need to know the actual plotting area used at any | ||||
given time, you can check ChartView::plotArea instead. | ||||
*/ | ||||
/*! | ||||
\qmlproperty rect ChartView::plotArea | ||||
The area on the ChartView that is used for drawing series. This is the ChartView rect without the | ||||
margins. | ||||
\sa ChartView::minimumMargins | ||||
Tero Ahola
|
r1524 | */ | ||
Tero Ahola
|
r2369 | /*! | ||
\qmlproperty Margins ChartView::margins | ||||
The minimum margins allowed between the outer bounds and the plotArea of the ChartView. Margins | ||||
area of ChartView is used for drawing title, axes and legend. | ||||
*/ | ||||
Tero Ahola
|
r1521 | /*! | ||
\qmlmethod AbstractSeries ChartView::series(int index) | ||||
Returns the series with \a index on the chart. This allows you to loop through the series of a chart together with | ||||
the count property of the chart. | ||||
*/ | ||||
/*! | ||||
\qmlmethod AbstractSeries ChartView::series(string name) | ||||
Returns the first series on the chart with \a name. If there is no series with that name, returns null. | ||||
*/ | ||||
/*! | ||||
Tero Ahola
|
r1960 | \qmlmethod AbstractSeries ChartView::createSeries(SeriesType type, string name, AbstractAxis axisX, AbstractAxis axisY) | ||
Creates a series object of \a type to the chart with name \a name, optional axis \a axisX and | ||||
optional axis \a axisY. For example: | ||||
Tero Ahola
|
r1521 | \code | ||
Tero Ahola
|
r1960 | // lineSeries is a LineSeries object that has already been added to the ChartView; re-use it's axes | ||
var myAxisX = chartView.axisX(lineSeries); | ||||
var myAxisY = chartView.axisY(lineSeries); | ||||
var scatter = chartView.createSeries(ChartView.SeriesTypeScatter, "scatter series", myAxisX, myAxisY); | ||||
Tero Ahola
|
r1521 | \endcode | ||
*/ | ||||
Tero Ahola
|
r1948 | /*! | ||
\qmlmethod ChartView::removeSeries(AbstractSeries series) | ||||
Removes the \a series from the chart. The series object is also destroyed. | ||||
*/ | ||||
/*! | ||||
\qmlmethod ChartView::removeAllSeries() | ||||
Removes all series from the chart. All the series objects are also destroyed. | ||||
*/ | ||||
Tero Ahola
|
r1960 | /*! | ||
Tero Ahola
|
r2068 | \qmlmethod Axis ChartView::axisX(AbstractSeries series) | ||
Tero Ahola
|
r1960 | The x-axis of the series. | ||
*/ | ||||
Tero Ahola
|
r1462 | /*! | ||
Tero Ahola
|
r2068 | \qmlmethod Axis ChartView::axisY(AbstractSeries series) | ||
Tero Ahola
|
r1960 | The y-axis of the series. | ||
Tero Ahola
|
r1521 | */ | ||
/*! | ||||
\qmlmethod ChartView::zoomY(real factor) | ||||
Zooms in by \a factor on the center of the chart. | ||||
*/ | ||||
/*! | ||||
\qmlmethod ChartView::scrollLeft(real pixels) | ||||
Scrolls to left by \a pixels. This is a convenience function that suits for example for key navigation. | ||||
*/ | ||||
/*! | ||||
\qmlmethod ChartView::scrollRight(real pixels) | ||||
Scrolls to right by \a pixels. This is a convenience function that suits for example for key navigation. | ||||
*/ | ||||
/*! | ||||
\qmlmethod ChartView::scrollUp(real pixels) | ||||
Scrolls up by \a pixels. This is a convenience function that suits for example for key navigation. | ||||
*/ | ||||
/*! | ||||
\qmlmethod ChartView::scrollDown(real pixels) | ||||
Scrolls down by \a pixels. This is a convenience function that suits for example for key navigation. | ||||
Tero Ahola
|
r1462 | */ | ||
Tero Ahola
|
r1531 | /*! | ||
Tero Ahola
|
r1946 | \qmlsignal ChartView::onPlotAreaChanged(rect plotArea) | ||
The plot area of the chart has changed. This may happen for example, if you modify minimumMargins | ||||
or if you resize the chart, or if you modify font size related properties of the legend or chart | ||||
title. | ||||
Tero Ahola
|
r1531 | */ | ||
Tero Ahola
|
r2068 | /*! | ||
\qmlsignal ChartView::seriesAdded(AbstractSeries series) | ||||
The \a series has been added to the chart. | ||||
*/ | ||||
/*! | ||||
\qmlsignal ChartView::seriesRemoved(AbstractSeries series) | ||||
The \a series has been removed from the chart. Please note that \a series is no longer a valid | ||||
object after the signal handler has completed. | ||||
*/ | ||||
Miikka Heikkinen
|
r2488 | DeclarativeChart::DeclarativeChart(QDECLARATIVE_ITEM *parent) | ||
: QDECLARATIVE_PAINTED_ITEM(parent) | ||||
Jani Honkonen
|
r1 | { | ||
Miikka Heikkinen
|
r2483 | initChart(QChart::ChartTypeCartesian); | ||
} | ||||
Miikka Heikkinen
|
r2488 | DeclarativeChart::DeclarativeChart(QChart::ChartType type, QDECLARATIVE_ITEM *parent) | ||
: QDECLARATIVE_PAINTED_ITEM(parent) | ||||
Miikka Heikkinen
|
r2483 | { | ||
initChart(type); | ||||
} | ||||
void DeclarativeChart::initChart(QChart::ChartType type) | ||||
{ | ||||
Miikka Heikkinen
|
r2488 | #ifdef CHARTS_FOR_QUICK2 | ||
Heikkinen Miikka
|
r2505 | m_currentSceneImage = 0; | ||
Miikka Heikkinen
|
r2506 | m_guiThreadId = QThread::currentThreadId(); | ||
m_paintThreadId = 0; | ||||
Heikkinen Miikka
|
r2505 | |||
Miikka Heikkinen
|
r2488 | if (type == QChart::ChartTypePolar) | ||
m_chart = new QPolarChart(); | ||||
else | ||||
m_chart = new QChart(); | ||||
m_scene = new QGraphicsScene(this); | ||||
m_scene->addItem(m_chart); | ||||
setAntialiasing(QQuickItem::antialiasing()); | ||||
Heikkinen Miikka
|
r2505 | connect(m_scene, SIGNAL(changed(QList<QRectF>)), this, SLOT(sceneChanged(QList<QRectF>))); | ||
Miikka Heikkinen
|
r2488 | connect(this, SIGNAL(antialiasingChanged(bool)), this, SLOT(handleAntialiasingChanged(bool))); | ||
Miikka Heikkinen
|
r2495 | |||
setAcceptedMouseButtons(Qt::AllButtons); | ||||
setAcceptHoverEvents(true); | ||||
Miikka Heikkinen
|
r2488 | #else | ||
Miikka Heikkinen
|
r2483 | if (type == QChart::ChartTypePolar) | ||
m_chart = new QPolarChart(this); | ||||
else | ||||
m_chart = new QChart(this); | ||||
Jani Honkonen
|
r1 | setFlag(QGraphicsItem::ItemHasNoContents, false); | ||
Miikka Heikkinen
|
r2488 | #endif | ||
Michal Klocek
|
r2090 | m_margins = new DeclarativeMargins(this); | ||
m_margins->setTop(m_chart->margins().top()); | ||||
m_margins->setLeft(m_chart->margins().left()); | ||||
m_margins->setRight(m_chart->margins().right()); | ||||
m_margins->setBottom(m_chart->margins().bottom()); | ||||
Jani Honkonen
|
r2110 | connect(m_margins, SIGNAL(topChanged(int,int,int,int)), this, SLOT(changeMinimumMargins(int,int,int,int))); | ||
connect(m_margins, SIGNAL(bottomChanged(int,int,int,int)), this, SLOT(changeMinimumMargins(int,int,int,int))); | ||||
connect(m_margins, SIGNAL(leftChanged(int,int,int,int)), this, SLOT(changeMinimumMargins(int,int,int,int))); | ||||
connect(m_margins, SIGNAL(rightChanged(int,int,int,int)), this, SLOT(changeMinimumMargins(int,int,int,int))); | ||||
Jani Honkonen
|
r2277 | connect(m_chart->d_ptr->m_dataset, SIGNAL(seriesAdded(QAbstractSeries*)), this, SLOT(handleSeriesAdded(QAbstractSeries*))); | ||
Jani Honkonen
|
r2110 | connect(m_chart->d_ptr->m_dataset, SIGNAL(seriesRemoved(QAbstractSeries*)), this, SIGNAL(seriesRemoved(QAbstractSeries*))); | ||
Tero Ahola
|
r2068 | } | ||
Jani Honkonen
|
r2277 | void DeclarativeChart::handleSeriesAdded(QAbstractSeries *series) | ||
Tero Ahola
|
r2068 | { | ||
emit seriesAdded(series); | ||||
Tero Ahola
|
r1524 | } | ||
Tero Ahola
|
r1928 | void DeclarativeChart::changeMinimumMargins(int top, int bottom, int left, int right) | ||
Tero Ahola
|
r1524 | { | ||
Michal Klocek
|
r1965 | m_chart->setMargins(QMargins(left, top, right, bottom)); | ||
Tero Ahola
|
r1946 | emit minimumMarginsChanged(); | ||
emit plotAreaChanged(m_chart->plotArea()); | ||||
Jani Honkonen
|
r1 | } | ||
Tero Ahola
|
r722 | DeclarativeChart::~DeclarativeChart() | ||
{ | ||||
delete m_chart; | ||||
Heikkinen Miikka
|
r2505 | #ifdef CHARTS_FOR_QUICK2 | ||
m_sceneImageLock.lock(); | ||||
delete m_currentSceneImage; | ||||
m_currentSceneImage = 0; | ||||
m_sceneImageLock.unlock(); | ||||
#endif | ||||
Tero Ahola
|
r722 | } | ||
Tero Ahola
|
r1117 | void DeclarativeChart::childEvent(QChildEvent *event) | ||
{ | ||||
if (event->type() == QEvent::ChildAdded) { | ||||
if (qobject_cast<QAbstractSeries *>(event->child())) { | ||||
m_chart->addSeries(qobject_cast<QAbstractSeries *>(event->child())); | ||||
} | ||||
} | ||||
} | ||||
void DeclarativeChart::componentComplete() | ||||
{ | ||||
Jani Honkonen
|
r2100 | foreach (QObject *child, children()) { | ||
Tero Ahola
|
r1117 | if (qobject_cast<QAbstractSeries *>(child)) { | ||
Tero Ahola
|
r1903 | // Add series to the chart | ||
QAbstractSeries *series = qobject_cast<QAbstractSeries *>(child); | ||||
m_chart->addSeries(series); | ||||
Tero Ahola
|
r2296 | // Connect to axis changed signals (unless this is a pie series) | ||
if (!qobject_cast<DeclarativePieSeries *>(series)) { | ||||
connect(series, SIGNAL(axisXChanged(QAbstractAxis*)), this, SLOT(handleAxisXSet(QAbstractAxis*))); | ||||
connect(series, SIGNAL(axisXTopChanged(QAbstractAxis*)), this, SLOT(handleAxisXSet(QAbstractAxis*))); | ||||
connect(series, SIGNAL(axisYChanged(QAbstractAxis*)), this, SLOT(handleAxisYSet(QAbstractAxis*))); | ||||
connect(series, SIGNAL(axisYRightChanged(QAbstractAxis*)), this, SLOT(handleAxisYRightSet(QAbstractAxis*))); | ||||
Tero Ahola
|
r1813 | } | ||
Tero Ahola
|
r2296 | |||
initializeAxes(series); | ||||
Tero Ahola
|
r1117 | } | ||
} | ||||
Tero Ahola
|
r1813 | |||
Miikka Heikkinen
|
r2488 | QDECLARATIVE_ITEM::componentComplete(); | ||
Tero Ahola
|
r1117 | } | ||
Jani Honkonen
|
r2101 | void DeclarativeChart::handleAxisXSet(QAbstractAxis *axis) | ||
Tero Ahola
|
r1813 | { | ||
Tero Ahola
|
r2296 | QAbstractSeries *s = qobject_cast<QAbstractSeries *>(sender()); | ||
if (axis && s) { | ||||
if (!m_chart->axes(Qt::Horizontal).contains(axis)) | ||||
m_chart->addAxis(axis, Qt::AlignBottom); | ||||
if (!s->attachedAxes().contains(axis)) | ||||
s->attachAxis(axis); | ||||
} else { | ||||
Tero Ahola
|
r1947 | qWarning() << "Trying to set axisX to null."; | ||
Tero Ahola
|
r2296 | } | ||
} | ||||
void DeclarativeChart::handleAxisXTopSet(QAbstractAxis *axis) | ||||
{ | ||||
QAbstractSeries *s = qobject_cast<QAbstractSeries *>(sender()); | ||||
if (axis && s) { | ||||
if (!m_chart->axes(Qt::Horizontal).contains(axis)) | ||||
m_chart->addAxis(axis, Qt::AlignTop); | ||||
if (!s->attachedAxes().contains(axis)) | ||||
s->attachAxis(axis); | ||||
} else { | ||||
qWarning() << "Trying to set axisXTop to null."; | ||||
} | ||||
Tero Ahola
|
r1813 | } | ||
Jani Honkonen
|
r2101 | void DeclarativeChart::handleAxisYSet(QAbstractAxis *axis) | ||
Tero Ahola
|
r1813 | { | ||
Tero Ahola
|
r2296 | QAbstractSeries *s = qobject_cast<QAbstractSeries *>(sender()); | ||
if (axis && s) { | ||||
if (!m_chart->axes(Qt::Vertical).contains(axis)) | ||||
m_chart->addAxis(axis, Qt::AlignLeft); | ||||
if (!s->attachedAxes().contains(axis)) | ||||
s->attachAxis(axis); | ||||
} else { | ||||
Tero Ahola
|
r1947 | qWarning() << "Trying to set axisY to null."; | ||
Tero Ahola
|
r2296 | } | ||
} | ||||
void DeclarativeChart::handleAxisYRightSet(QAbstractAxis *axis) | ||||
{ | ||||
QAbstractSeries *s = qobject_cast<QAbstractSeries *>(sender()); | ||||
if (axis && s) { | ||||
if (!m_chart->axes(Qt::Vertical).contains(axis)) | ||||
m_chart->addAxis(axis, Qt::AlignRight); | ||||
if (!s->attachedAxes().contains(axis)) | ||||
s->attachAxis(axis); | ||||
} else { | ||||
qWarning() << "Trying to set axisYRight to null."; | ||||
} | ||||
Tero Ahola
|
r1813 | } | ||
Tero Ahola
|
r120 | void DeclarativeChart::geometryChanged(const QRectF &newGeometry, const QRectF &oldGeometry) | ||
Jani Honkonen
|
r1 | { | ||
Tero Ahola
|
r200 | if (newGeometry.isValid()) { | ||
if (newGeometry.width() > 0 && newGeometry.height() > 0) { | ||||
Tero Ahola
|
r642 | m_chart->resize(newGeometry.width(), newGeometry.height()); | ||
Tero Ahola
|
r200 | } | ||
} | ||||
Miikka Heikkinen
|
r2488 | QDECLARATIVE_ITEM::geometryChanged(newGeometry, oldGeometry); | ||
Tero Ahola
|
r1946 | |||
// It would be better to trigger the plotAreaChanged signal from QChart::plotAreaChanged or | ||||
// similar. Since that kind of a signal is not clearly needed in the C++ API the work-around is | ||||
// to implement it here for the QML API purposes. | ||||
emit plotAreaChanged(m_chart->plotArea()); | ||||
Jani Honkonen
|
r1 | } | ||
Miikka Heikkinen
|
r2488 | #ifdef CHARTS_FOR_QUICK2 | ||
Heikkinen Miikka
|
r2505 | void DeclarativeChart::sceneChanged(QList<QRectF> region) | ||
{ | ||||
Q_UNUSED(region); | ||||
Miikka Heikkinen
|
r2506 | if (m_guiThreadId == m_paintThreadId) { | ||
// Rendering in gui thread, no need for shenannigans, just update | ||||
update(); | ||||
} else { | ||||
// Multi-threaded rendering, need to ensure scene is actually rendered in gui thread | ||||
if (!m_updatePending) { | ||||
m_updatePending = true; | ||||
// Do async render to avoid some unnecessary renders. | ||||
QTimer::singleShot(0, this, SLOT(renderScene())); | ||||
} | ||||
Heikkinen Miikka
|
r2505 | } | ||
} | ||||
void DeclarativeChart::renderScene() | ||||
{ | ||||
m_updatePending = false; | ||||
m_sceneImageLock.lock(); | ||||
delete m_currentSceneImage; | ||||
m_currentSceneImage = new QImage(m_chart->size().toSize(), QImage::Format_ARGB32); | ||||
m_currentSceneImage->fill(Qt::transparent); | ||||
QPainter painter(m_currentSceneImage); | ||||
Miikka Heikkinen
|
r2506 | if (antialiasing()) | ||
painter.setRenderHints(QPainter::Antialiasing | QPainter::TextAntialiasing | QPainter::SmoothPixmapTransform); | ||||
Heikkinen Miikka
|
r2505 | QRect renderRect(QPoint(0, 0), m_chart->size().toSize()); | ||
m_scene->render(&painter, renderRect, renderRect); | ||||
m_sceneImageLock.unlock(); | ||||
update(); | ||||
} | ||||
Miikka Heikkinen
|
r2488 | void DeclarativeChart::paint(QPainter *painter) | ||
{ | ||||
Miikka Heikkinen
|
r2506 | if (!m_paintThreadId) { | ||
m_paintThreadId = QThread::currentThreadId(); | ||||
if (m_guiThreadId == m_paintThreadId) { | ||||
// No need for scene image in single threaded rendering, so delete | ||||
// the one that got made by default before the rendering type was | ||||
// detected. | ||||
delete m_currentSceneImage; | ||||
m_currentSceneImage = 0; | ||||
} | ||||
} | ||||
if (m_guiThreadId == m_paintThreadId) { | ||||
QRectF renderRect(QPointF(0, 0), m_chart->size()); | ||||
m_scene->render(painter, renderRect, renderRect); | ||||
} else { | ||||
m_sceneImageLock.lock(); | ||||
if (m_currentSceneImage) { | ||||
QRect imageRect(QPoint(0, 0), m_currentSceneImage->size()); | ||||
QRect itemRect(QPoint(0, 0), QSize(width(), height())); | ||||
painter->drawImage(itemRect, *m_currentSceneImage, imageRect); | ||||
} | ||||
m_sceneImageLock.unlock(); | ||||
Heikkinen Miikka
|
r2505 | } | ||
Miikka Heikkinen
|
r2488 | } | ||
Miikka Heikkinen
|
r2495 | void DeclarativeChart::mousePressEvent(QMouseEvent *event) | ||
{ | ||||
m_mousePressScenePoint = event->pos(); | ||||
m_mousePressScreenPoint = event->globalPos(); | ||||
m_lastMouseMoveScenePoint = m_mousePressScenePoint; | ||||
m_lastMouseMoveScreenPoint = m_mousePressScreenPoint; | ||||
m_mousePressButton = event->button(); | ||||
m_mousePressButtons = event->buttons(); | ||||
QGraphicsSceneMouseEvent mouseEvent(QEvent::GraphicsSceneMousePress); | ||||
mouseEvent.setWidget(0); | ||||
mouseEvent.setButtonDownScenePos(m_mousePressButton, m_mousePressScenePoint); | ||||
mouseEvent.setButtonDownScreenPos(m_mousePressButton, m_mousePressScreenPoint); | ||||
mouseEvent.setScenePos(m_mousePressScenePoint); | ||||
mouseEvent.setScreenPos(m_mousePressScreenPoint); | ||||
mouseEvent.setLastScenePos(m_lastMouseMoveScenePoint); | ||||
mouseEvent.setLastScreenPos(m_lastMouseMoveScreenPoint); | ||||
mouseEvent.setButtons(m_mousePressButtons); | ||||
mouseEvent.setButton(m_mousePressButton); | ||||
mouseEvent.setModifiers(event->modifiers()); | ||||
mouseEvent.setAccepted(false); | ||||
QApplication::sendEvent(m_scene, &mouseEvent); | ||||
} | ||||
void DeclarativeChart::mouseReleaseEvent(QMouseEvent *event) | ||||
{ | ||||
QGraphicsSceneMouseEvent mouseEvent(QEvent::GraphicsSceneMouseRelease); | ||||
mouseEvent.setWidget(0); | ||||
mouseEvent.setButtonDownScenePos(m_mousePressButton, m_mousePressScenePoint); | ||||
mouseEvent.setButtonDownScreenPos(m_mousePressButton, m_mousePressScreenPoint); | ||||
mouseEvent.setScenePos(event->pos()); | ||||
mouseEvent.setScreenPos(event->globalPos()); | ||||
mouseEvent.setLastScenePos(m_lastMouseMoveScenePoint); | ||||
mouseEvent.setLastScreenPos(m_lastMouseMoveScreenPoint); | ||||
mouseEvent.setButtons(event->buttons()); | ||||
mouseEvent.setButton(event->button()); | ||||
mouseEvent.setModifiers(event->modifiers()); | ||||
mouseEvent.setAccepted(false); | ||||
QApplication::sendEvent(m_scene, &mouseEvent); | ||||
m_mousePressButtons = event->buttons(); | ||||
m_mousePressButton = Qt::NoButton; | ||||
} | ||||
void DeclarativeChart::hoverMoveEvent(QHoverEvent *event) | ||||
{ | ||||
// Convert hover move to mouse move, since we don't seem to get actual mouse move events. | ||||
// QGraphicsScene generates hover events from mouse move events, so we don't need | ||||
// to pass hover events there. | ||||
QGraphicsSceneMouseEvent mouseEvent(QEvent::GraphicsSceneMouseMove); | ||||
mouseEvent.setWidget(0); | ||||
mouseEvent.setButtonDownScenePos(m_mousePressButton, m_mousePressScenePoint); | ||||
mouseEvent.setButtonDownScreenPos(m_mousePressButton, m_mousePressScreenPoint); | ||||
mouseEvent.setScenePos(event->pos()); | ||||
// Hover events do not have global pos in them, and the screen position doesn't seem to | ||||
// matter anyway in this use case, so just pass event pos instead of trying to | ||||
// calculate the real screen position. | ||||
mouseEvent.setScreenPos(event->pos()); | ||||
mouseEvent.setLastScenePos(m_lastMouseMoveScenePoint); | ||||
mouseEvent.setLastScreenPos(m_lastMouseMoveScreenPoint); | ||||
mouseEvent.setButtons(m_mousePressButtons); | ||||
mouseEvent.setButton(m_mousePressButton); | ||||
mouseEvent.setModifiers(event->modifiers()); | ||||
m_lastMouseMoveScenePoint = mouseEvent.scenePos(); | ||||
m_lastMouseMoveScreenPoint = mouseEvent.screenPos(); | ||||
mouseEvent.setAccepted(false); | ||||
QApplication::sendEvent(m_scene, &mouseEvent); | ||||
} | ||||
Miikka Heikkinen
|
r2488 | void DeclarativeChart::handleAntialiasingChanged(bool enable) | ||
{ | ||||
setAntialiasing(enable); | ||||
} | ||||
#endif | ||||
Tero Ahola
|
r1240 | void DeclarativeChart::setTheme(DeclarativeChart::Theme theme) | ||
Tero Ahola
|
r1094 | { | ||
Tero Ahola
|
r1357 | QChart::ChartTheme chartTheme = (QChart::ChartTheme) theme; | ||
Tero Ahola
|
r1481 | if (chartTheme != m_chart->theme()) | ||
Tero Ahola
|
r1357 | m_chart->setTheme(chartTheme); | ||
Tero Ahola
|
r1094 | } | ||
Tero Ahola
|
r1240 | DeclarativeChart::Theme DeclarativeChart::theme() | ||
{ | ||||
return (DeclarativeChart::Theme) m_chart->theme(); | ||||
} | ||||
void DeclarativeChart::setAnimationOptions(DeclarativeChart::Animation animations) | ||||
{ | ||||
Tero Ahola
|
r1357 | QChart::AnimationOption animationOptions = (QChart::AnimationOption) animations; | ||
Tero Ahola
|
r1481 | if (animationOptions != m_chart->animationOptions()) | ||
Tero Ahola
|
r1357 | m_chart->setAnimationOptions(animationOptions); | ||
Tero Ahola
|
r1240 | } | ||
DeclarativeChart::Animation DeclarativeChart::animationOptions() | ||||
Tero Ahola
|
r1094 | { | ||
if (m_chart->animationOptions().testFlag(QChart::AllAnimations)) | ||||
Tero Ahola
|
r1240 | return DeclarativeChart::AllAnimations; | ||
Tero Ahola
|
r1094 | else if (m_chart->animationOptions().testFlag(QChart::GridAxisAnimations)) | ||
Tero Ahola
|
r1240 | return DeclarativeChart::GridAxisAnimations; | ||
Tero Ahola
|
r1094 | else if (m_chart->animationOptions().testFlag(QChart::SeriesAnimations)) | ||
Tero Ahola
|
r1240 | return DeclarativeChart::SeriesAnimations; | ||
Tero Ahola
|
r1094 | else | ||
Tero Ahola
|
r1240 | return DeclarativeChart::NoAnimation; | ||
Tero Ahola
|
r1094 | } | ||
Tero Ahola
|
r1357 | void DeclarativeChart::setTitle(QString title) | ||
{ | ||||
Tero Ahola
|
r1481 | if (title != m_chart->title()) | ||
Tero Ahola
|
r1357 | m_chart->setTitle(title); | ||
Tero Ahola
|
r1095 | } | ||
Tero Ahola
|
r1357 | QString DeclarativeChart::title() | ||
Tero Ahola
|
r1095 | { | ||
Tero Ahola
|
r1357 | return m_chart->title(); | ||
Tero Ahola
|
r1095 | } | ||
Tero Ahola
|
r1550 | QAbstractAxis *DeclarativeChart::axisX(QAbstractSeries *series) | ||
Tero Ahola
|
r1139 | { | ||
Tero Ahola
|
r2296 | QList<QAbstractAxis *> axes = m_chart->axes(Qt::Horizontal, series); | ||
if (axes.count()) | ||||
return axes[0]; | ||||
return 0; | ||||
Tero Ahola
|
r1139 | } | ||
Michal Klocek
|
r1541 | QAbstractAxis *DeclarativeChart::axisY(QAbstractSeries *series) | ||
Tero Ahola
|
r1139 | { | ||
Tero Ahola
|
r2296 | QList<QAbstractAxis *> axes = m_chart->axes(Qt::Vertical, series); | ||
if (axes.count()) | ||||
return axes[0]; | ||||
return 0; | ||||
Tero Ahola
|
r1139 | } | ||
Tero Ahola
|
r1357 | QLegend *DeclarativeChart::legend() | ||
{ | ||||
return m_chart->legend(); | ||||
} | ||||
void DeclarativeChart::setTitleColor(QColor color) | ||||
{ | ||||
QBrush b = m_chart->titleBrush(); | ||||
if (color != b.color()) { | ||||
b.setColor(color); | ||||
m_chart->setTitleBrush(b); | ||||
Tero Ahola
|
r1524 | emit titleColorChanged(color); | ||
Tero Ahola
|
r1357 | } | ||
} | ||||
Jani Honkonen
|
r1517 | QFont DeclarativeChart::titleFont() const | ||
{ | ||||
return m_chart->titleFont(); | ||||
} | ||||
Jani Honkonen
|
r2101 | void DeclarativeChart::setTitleFont(const QFont &font) | ||
Jani Honkonen
|
r1517 | { | ||
m_chart->setTitleFont(font); | ||||
} | ||||
Tero Ahola
|
r1357 | QColor DeclarativeChart::titleColor() | ||
{ | ||||
return m_chart->titleBrush().color(); | ||||
} | ||||
void DeclarativeChart::setBackgroundColor(QColor color) | ||||
{ | ||||
QBrush b = m_chart->backgroundBrush(); | ||||
Tero Ahola
|
r1473 | if (b.style() != Qt::SolidPattern || color != b.color()) { | ||
b.setStyle(Qt::SolidPattern); | ||||
Tero Ahola
|
r1357 | b.setColor(color); | ||
m_chart->setBackgroundBrush(b); | ||||
emit backgroundColorChanged(); | ||||
} | ||||
} | ||||
QColor DeclarativeChart::backgroundColor() | ||||
{ | ||||
return m_chart->backgroundBrush().color(); | ||||
Tero Ahola
|
r1157 | } | ||
Miikka Heikkinen
|
r2498 | void QtCommercialChart::DeclarativeChart::setPlotAreaColor(QColor color) | ||
{ | ||||
QBrush b = m_chart->plotAreaBackgroundBrush(); | ||||
if (b.style() != Qt::SolidPattern || color != b.color()) { | ||||
b.setStyle(Qt::SolidPattern); | ||||
b.setColor(color); | ||||
m_chart->setPlotAreaBackgroundBrush(b); | ||||
m_chart->setPlotAreaBackgroundVisible(true); | ||||
emit plotAreaColorChanged(); | ||||
} | ||||
} | ||||
QColor QtCommercialChart::DeclarativeChart::plotAreaColor() | ||||
{ | ||||
return m_chart->plotAreaBackgroundBrush().color(); | ||||
} | ||||
Tero Ahola
|
r1240 | int DeclarativeChart::count() | ||
{ | ||||
return m_chart->series().count(); | ||||
} | ||||
Tero Ahola
|
r1461 | void DeclarativeChart::setDropShadowEnabled(bool enabled) | ||
{ | ||||
Tero Ahola
|
r1462 | if (enabled != m_chart->isDropShadowEnabled()) { | ||
m_chart->setDropShadowEnabled(enabled); | ||||
Tero Ahola
|
r1461 | dropShadowEnabledChanged(enabled); | ||
} | ||||
} | ||||
bool DeclarativeChart::dropShadowEnabled() | ||||
{ | ||||
Tero Ahola
|
r1462 | return m_chart->isDropShadowEnabled(); | ||
Tero Ahola
|
r1461 | } | ||
Miikka Heikkinen
|
r2549 | qreal DeclarativeChart::backgroundRoundness() const | ||
{ | ||||
return m_chart->backgroundRoundness(); | ||||
} | ||||
void DeclarativeChart::setBackgroundRoundness(qreal diameter) | ||||
{ | ||||
if (m_chart->backgroundRoundness() != diameter) { | ||||
m_chart->setBackgroundRoundness(diameter); | ||||
emit backgroundRoundnessChanged(diameter); | ||||
} | ||||
} | ||||
Tero Ahola
|
r1524 | qreal DeclarativeChart::topMargin() | ||
{ | ||||
Tero Ahola
|
r2369 | qWarning() << "ChartView.topMargin is deprecated. Use margins instead."; | ||
return m_chart->margins().top(); | ||||
Tero Ahola
|
r1524 | } | ||
qreal DeclarativeChart::bottomMargin() | ||||
{ | ||||
Tero Ahola
|
r2369 | qWarning() << "ChartView.bottomMargin is deprecated. Use margins instead."; | ||
return m_chart->margins().bottom(); | ||||
Tero Ahola
|
r1524 | } | ||
qreal DeclarativeChart::leftMargin() | ||||
{ | ||||
Tero Ahola
|
r2369 | qWarning() << "ChartView.leftMargin is deprecated. Use margins instead."; | ||
return m_chart->margins().left(); | ||||
Tero Ahola
|
r1524 | } | ||
qreal DeclarativeChart::rightMargin() | ||||
{ | ||||
Tero Ahola
|
r2369 | qWarning() << "ChartView.rightMargin is deprecated. Use margins instead."; | ||
return m_chart->margins().right(); | ||||
Tero Ahola
|
r1524 | } | ||
Tero Ahola
|
r1461 | void DeclarativeChart::zoom(qreal factor) | ||
{ | ||||
m_chart->zoom(factor); | ||||
} | ||||
void DeclarativeChart::scrollLeft(qreal pixels) | ||||
{ | ||||
Tero Ahola
|
r1955 | m_chart->scroll(-pixels, 0); | ||
Tero Ahola
|
r1461 | } | ||
void DeclarativeChart::scrollRight(qreal pixels) | ||||
{ | ||||
Tero Ahola
|
r1955 | m_chart->scroll(pixels, 0); | ||
Tero Ahola
|
r1461 | } | ||
void DeclarativeChart::scrollUp(qreal pixels) | ||||
{ | ||||
Michal Klocek
|
r1553 | m_chart->scroll(0, pixels); | ||
Tero Ahola
|
r1461 | } | ||
void DeclarativeChart::scrollDown(qreal pixels) | ||||
{ | ||||
Michal Klocek
|
r1553 | m_chart->scroll(0, -pixels); | ||
Tero Ahola
|
r1461 | } | ||
Miikka Heikkinen
|
r2488 | QDECLARATIVE_LIST_PROPERTY<QAbstractAxis> DeclarativeChart::axes() | ||
Tero Ahola
|
r2296 | { | ||
Miikka Heikkinen
|
r2488 | return QDECLARATIVE_LIST_PROPERTY<QAbstractAxis>(this, 0, | ||
Tero Ahola
|
r2296 | &DeclarativeChart::axesAppendFunc, | ||
&DeclarativeChart::axesCountFunc, | ||||
Miikka Heikkinen
|
r2488 | #ifdef CHARTS_FOR_QUICK2 | ||
&DeclarativeChart::axesAtFunc, | ||||
&DeclarativeChart::axesClearFunc); | ||||
#else | ||||
Tero Ahola
|
r2296 | &DeclarativeChart::axesAtFunc); | ||
Miikka Heikkinen
|
r2488 | #endif | ||
Tero Ahola
|
r2296 | } | ||
Miikka Heikkinen
|
r2488 | void DeclarativeChart::axesAppendFunc(QDECLARATIVE_LIST_PROPERTY<QAbstractAxis> *list, QAbstractAxis *element) | ||
Tero Ahola
|
r2296 | { | ||
// Empty implementation | ||||
Q_UNUSED(list); | ||||
Q_UNUSED(element); | ||||
} | ||||
Miikka Heikkinen
|
r2488 | int DeclarativeChart::axesCountFunc(QDECLARATIVE_LIST_PROPERTY<QAbstractAxis> *list) | ||
Tero Ahola
|
r2296 | { | ||
if (qobject_cast<DeclarativeChart *>(list->object)) { | ||||
DeclarativeChart *chart = qobject_cast<DeclarativeChart *>(list->object); | ||||
Tero Ahola
|
r2300 | return chart->m_chart->axes(Qt::Horizontal | Qt::Vertical).count(); | ||
Tero Ahola
|
r2296 | } | ||
return 0; | ||||
} | ||||
Miikka Heikkinen
|
r2488 | QAbstractAxis *DeclarativeChart::axesAtFunc(QDECLARATIVE_LIST_PROPERTY<QAbstractAxis> *list, int index) | ||
Tero Ahola
|
r2296 | { | ||
if (qobject_cast<DeclarativeChart *>(list->object)) { | ||||
DeclarativeChart *chart = qobject_cast<DeclarativeChart *>(list->object); | ||||
QList<QAbstractAxis *> axes = chart->m_chart->axes(Qt::Horizontal | Qt::Vertical, chart->m_chart->series()[0]); | ||||
return axes.at(index); | ||||
} | ||||
return 0; | ||||
} | ||||
Miikka Heikkinen
|
r2488 | void DeclarativeChart::axesClearFunc(QDECLARATIVE_LIST_PROPERTY<QAbstractAxis> *list) | ||
{ | ||||
// Empty implementation | ||||
Q_UNUSED(list); | ||||
} | ||||
Tero Ahola
|
r1240 | QAbstractSeries *DeclarativeChart::series(int index) | ||
{ | ||||
if (index < m_chart->series().count()) { | ||||
return m_chart->series().at(index); | ||||
} | ||||
return 0; | ||||
} | ||||
QAbstractSeries *DeclarativeChart::series(QString seriesName) | ||||
{ | ||||
Jani Honkonen
|
r2100 | foreach (QAbstractSeries *series, m_chart->series()) { | ||
Tero Ahola
|
r1240 | if (series->name() == seriesName) | ||
return series; | ||||
} | ||||
return 0; | ||||
} | ||||
Tero Ahola
|
r2389 | QAbstractSeries *DeclarativeChart::createSeries(int type, QString name, QAbstractAxis *axisX, QAbstractAxis *axisY) | ||
Tero Ahola
|
r1240 | { | ||
QAbstractSeries *series = 0; | ||||
Tero Ahola
|
r1813 | |||
Tero Ahola
|
r1240 | switch (type) { | ||
case DeclarativeChart::SeriesTypeLine: | ||||
series = new DeclarativeLineSeries(); | ||||
break; | ||||
Tero Ahola
|
r2114 | case DeclarativeChart::SeriesTypeArea: { | ||
DeclarativeAreaSeries *area = new DeclarativeAreaSeries(); | ||||
area->setUpperSeries(new DeclarativeLineSeries()); | ||||
series = area; | ||||
Tero Ahola
|
r1240 | break; | ||
Tero Ahola
|
r2114 | } | ||
Tero Ahola
|
r1240 | case DeclarativeChart::SeriesTypeStackedBar: | ||
sauimone
|
r1690 | series = new DeclarativeStackedBarSeries(); | ||
Tero Ahola
|
r1240 | break; | ||
case DeclarativeChart::SeriesTypePercentBar: | ||||
sauimone
|
r1690 | series = new DeclarativePercentBarSeries(); | ||
Tero Ahola
|
r1240 | break; | ||
sauimone
|
r1594 | case DeclarativeChart::SeriesTypeBar: | ||
series = new DeclarativeBarSeries(); | ||||
Tero Ahola
|
r1240 | break; | ||
sauimone
|
r1811 | case DeclarativeChart::SeriesTypeHorizontalBar: | ||
series = new DeclarativeHorizontalBarSeries(); | ||||
break; | ||||
case DeclarativeChart::SeriesTypeHorizontalPercentBar: | ||||
series = new DeclarativeHorizontalPercentBarSeries(); | ||||
break; | ||||
case DeclarativeChart::SeriesTypeHorizontalStackedBar: | ||||
series = new DeclarativeHorizontalStackedBarSeries(); | ||||
break; | ||||
Mika Salmela
|
r2548 | case DeclarativeChart::SeriesTypeBoxPlot: | ||
series = new DeclarativeBoxPlotSeries(); | ||||
break; | ||||
Tero Ahola
|
r1240 | case DeclarativeChart::SeriesTypePie: | ||
series = new DeclarativePieSeries(); | ||||
break; | ||||
case DeclarativeChart::SeriesTypeScatter: | ||||
series = new DeclarativeScatterSeries(); | ||||
break; | ||||
case DeclarativeChart::SeriesTypeSpline: | ||||
series = new DeclarativeSplineSeries(); | ||||
break; | ||||
default: | ||||
qWarning() << "Illegal series type"; | ||||
} | ||||
Tero Ahola
|
r1813 | |||
if (series) { | ||||
Tero Ahola
|
r2296 | // Connect to axis changed signals (unless this is a pie series) | ||
if (!qobject_cast<DeclarativePieSeries *>(series)) { | ||||
connect(series, SIGNAL(axisXChanged(QAbstractAxis*)), this, SLOT(handleAxisXSet(QAbstractAxis*))); | ||||
connect(series, SIGNAL(axisXTopChanged(QAbstractAxis*)), this, SLOT(handleAxisXSet(QAbstractAxis*))); | ||||
connect(series, SIGNAL(axisYChanged(QAbstractAxis*)), this, SLOT(handleAxisYSet(QAbstractAxis*))); | ||||
connect(series, SIGNAL(axisYRightChanged(QAbstractAxis*)), this, SLOT(handleAxisYRightSet(QAbstractAxis*))); | ||||
} | ||||
Tero Ahola
|
r1813 | series->setName(name); | ||
m_chart->addSeries(series); | ||||
Tero Ahola
|
r2296 | |||
if (axisX) | ||||
setAxisX(axisX, series); | ||||
if (axisY) | ||||
setAxisY(axisY, series); | ||||
if (series->attachedAxes().count() < 2) | ||||
initializeAxes(series); | ||||
Tero Ahola
|
r1813 | } | ||
Tero Ahola
|
r1240 | return series; | ||
} | ||||
Tero Ahola
|
r2068 | void DeclarativeChart::removeSeries(QAbstractSeries *series) | ||
{ | ||||
if (series) | ||||
m_chart->removeSeries(series); | ||||
else | ||||
qWarning("removeSeries: cannot remove null"); | ||||
} | ||||
Tero Ahola
|
r1813 | void DeclarativeChart::setAxisX(QAbstractAxis *axis, QAbstractSeries *series) | ||
Michal Klocek
|
r1604 | { | ||
Tero Ahola
|
r1813 | if (axis) | ||
m_chart->setAxisX(axis, series); | ||||
Michal Klocek
|
r1604 | } | ||
Tero Ahola
|
r1813 | void DeclarativeChart::setAxisY(QAbstractAxis *axis, QAbstractSeries *series) | ||
Michal Klocek
|
r1604 | { | ||
Tero Ahola
|
r1813 | if (axis) | ||
m_chart->setAxisY(axis, series); | ||||
Michal Klocek
|
r1604 | } | ||
void DeclarativeChart::createDefaultAxes() | ||||
{ | ||||
Tero Ahola
|
r1813 | qWarning() << "ChartView.createDefaultAxes() is deprecated. Axes are created automatically."; | ||
} | ||||
Tero Ahola
|
r2296 | QAbstractAxis *DeclarativeChart::defaultAxis(Qt::Orientation orientation, QAbstractSeries *series) | ||
Tero Ahola
|
r1813 | { | ||
Tero Ahola
|
r2296 | if (!series) { | ||
qWarning() << "No axis type defined for null series"; | ||||
return 0; | ||||
Tero Ahola
|
r1813 | } | ||
Tero Ahola
|
r2296 | foreach (QAbstractAxis *existingAxis, m_chart->axes(orientation)) { | ||
if (existingAxis->type() == series->d_ptr->defaultAxisType(orientation)) | ||||
return existingAxis; | ||||
Tero Ahola
|
r1813 | } | ||
Tero Ahola
|
r2296 | switch (series->d_ptr->defaultAxisType(orientation)) { | ||
case QAbstractAxis::AxisTypeValue: | ||||
return new QValueAxis(this); | ||||
case QAbstractAxis::AxisTypeBarCategory: | ||||
return new QBarCategoryAxis(this); | ||||
case QAbstractAxis::AxisTypeCategory: | ||||
return new QCategoryAxis(this); | ||||
Marek Rosa
|
r1867 | #ifndef QT_ON_ARM | ||
Tero Ahola
|
r2296 | case QAbstractAxis::AxisTypeDateTime: | ||
return new QDateTimeAxis(this); | ||||
Marek Rosa
|
r1867 | #endif | ||
Miikka Heikkinen
|
r2493 | case QAbstractAxis::AxisTypeLogValue: | ||
return new QLogValueAxis(this); | ||||
Tero Ahola
|
r2296 | default: | ||
// assume AxisTypeNoAxis | ||||
return 0; | ||||
Tero Ahola
|
r1813 | } | ||
Tero Ahola
|
r2296 | } | ||
Tero Ahola
|
r1813 | |||
Tero Ahola
|
r2296 | void DeclarativeChart::initializeAxes(QAbstractSeries *series) | ||
{ | ||||
if (qobject_cast<DeclarativeLineSeries *>(series)) | ||||
doInitializeAxes(series, qobject_cast<DeclarativeLineSeries *>(series)->m_axes); | ||||
else if (qobject_cast<DeclarativeScatterSeries *>(series)) | ||||
doInitializeAxes(series, qobject_cast<DeclarativeScatterSeries *>(series)->m_axes); | ||||
else if (qobject_cast<DeclarativeSplineSeries *>(series)) | ||||
doInitializeAxes(series, qobject_cast<DeclarativeSplineSeries *>(series)->m_axes); | ||||
else if (qobject_cast<DeclarativeAreaSeries *>(series)) | ||||
doInitializeAxes(series, qobject_cast<DeclarativeAreaSeries *>(series)->m_axes); | ||||
else if (qobject_cast<DeclarativeBarSeries *>(series)) | ||||
doInitializeAxes(series, qobject_cast<DeclarativeBarSeries *>(series)->m_axes); | ||||
else if (qobject_cast<DeclarativeStackedBarSeries *>(series)) | ||||
doInitializeAxes(series, qobject_cast<DeclarativeStackedBarSeries *>(series)->m_axes); | ||||
else if (qobject_cast<DeclarativePercentBarSeries *>(series)) | ||||
doInitializeAxes(series, qobject_cast<DeclarativePercentBarSeries *>(series)->m_axes); | ||||
else if (qobject_cast<DeclarativeHorizontalBarSeries *>(series)) | ||||
doInitializeAxes(series, qobject_cast<DeclarativeHorizontalBarSeries *>(series)->m_axes); | ||||
else if (qobject_cast<DeclarativeHorizontalStackedBarSeries *>(series)) | ||||
doInitializeAxes(series, qobject_cast<DeclarativeHorizontalStackedBarSeries *>(series)->m_axes); | ||||
else if (qobject_cast<DeclarativeHorizontalPercentBarSeries *>(series)) | ||||
doInitializeAxes(series, qobject_cast<DeclarativeHorizontalPercentBarSeries *>(series)->m_axes); | ||||
Mika Salmela
|
r2548 | else if (qobject_cast<DeclarativeBoxPlotSeries *>(series)) | ||
doInitializeAxes(series, qobject_cast<DeclarativeBoxPlotSeries *>(series)->m_axes); | ||||
Tero Ahola
|
r2296 | // else: do nothing | ||
} | ||||
void DeclarativeChart::doInitializeAxes(QAbstractSeries *series, DeclarativeAxes *axes) | ||||
{ | ||||
// Initialize axis X | ||||
if (axes->axisX()) | ||||
axes->emitAxisXChanged(); | ||||
else if (axes->axisXTop()) | ||||
axes->emitAxisXTopChanged(); | ||||
else | ||||
axes->setAxisX(defaultAxis(Qt::Horizontal, series)); | ||||
// Initialize axis Y | ||||
if (axes->axisY()) | ||||
axes->emitAxisYChanged(); | ||||
else if (axes->axisYRight()) | ||||
axes->emitAxisYRightChanged(); | ||||
else | ||||
axes->setAxisY(defaultAxis(Qt::Vertical, series)); | ||||
Michal Klocek
|
r1604 | } | ||
Tero Ahola
|
r120 | #include "moc_declarativechart.cpp" | ||
QTCOMMERCIALCHART_END_NAMESPACE | ||||