chartdataset.cpp
484 lines
| 12.2 KiB
| text/x-c
|
CppLexer
/ src / chartdataset.cpp
Michal Klocek
|
r771 | /**************************************************************************** | ||
** | ||||
** Copyright (C) 2012 Digia Plc | ||||
** All rights reserved. | ||||
** For any questions to Digia, please use contact form at http://qt.digia.com | ||||
** | ||||
** This file is part of the Qt Commercial Charts Add-on. | ||||
** | ||||
** $QT_BEGIN_LICENSE$ | ||||
** Licensees holding valid Qt Commercial licenses may use this file in | ||||
** accordance with the Qt Commercial License Agreement provided with the | ||||
** 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$ | ||||
** | ||||
****************************************************************************/ | ||||
Michal Klocek
|
r131 | #include "chartdataset_p.h" | ||
Jani Honkonen
|
r1314 | #include "qchart.h" | ||
Michal Klocek
|
r1541 | #include "qvaluesaxis.h" | ||
Michal Klocek
|
r1613 | #include "qbarcategoriesaxis.h" | ||
Michal Klocek
|
r1541 | #include "qvaluesaxis_p.h" | ||
Marek Rosa
|
r1717 | #include "qintervalsaxis.h" | ||
#include "qdatetimeaxis.h" | ||||
Tero Ahola
|
r988 | #include "qabstractseries_p.h" | ||
sauimone
|
r1586 | #include "qabstractbarseries.h" | ||
sauimone
|
r338 | #include "qstackedbarseries.h" | ||
#include "qpercentbarseries.h" | ||||
Jani Honkonen
|
r146 | #include "qpieseries.h" | ||
Michal Klocek
|
r131 | |||
QTCOMMERCIALCHART_BEGIN_NAMESPACE | ||||
Michal Klocek
|
r1695 | ChartDataSet::ChartDataSet(QChart *parent):QObject(parent) | ||
Michal Klocek
|
r131 | { | ||
Michal Klocek
|
r1556 | |||
Michal Klocek
|
r131 | } | ||
ChartDataSet::~ChartDataSet() | ||||
{ | ||||
Michal Klocek
|
r1062 | removeAllSeries(); | ||
Michal Klocek
|
r131 | } | ||
Michal Klocek
|
r1541 | void ChartDataSet::addSeries(QAbstractSeries* series) | ||
Michal Klocek
|
r131 | { | ||
Michal Klocek
|
r1577 | Domain* domain = m_seriesDomainMap.value(series); | ||
Michal Klocek
|
r910 | |||
Michal Klocek
|
r1577 | if(domain) { | ||
Michal Klocek
|
r439 | qWarning() << "Can not add series. Series already on the chart"; | ||
return; | ||||
} | ||||
Michal Klocek
|
r223 | |||
Michal Klocek
|
r1695 | domain = new Domain(series); | ||
m_seriesDomainMap.insert(series,domain); | ||||
series->d_ptr->scaleDomain(*domain); | ||||
createSeriesIndex(series); | ||||
Michal Klocek
|
r974 | series->setParent(this); // take ownership | ||
Michal Klocek
|
r1695 | series->d_ptr->m_chart = qobject_cast<QChart*>(parent()); | ||
series->d_ptr->m_dataset = this; | ||||
Michal Klocek
|
r223 | |||
Michal Klocek
|
r1695 | emit seriesAdded(series,domain); | ||
Michal Klocek
|
r421 | |||
Michal Klocek
|
r1695 | } | ||
void ChartDataSet::removeSeries(QAbstractSeries* series) | ||||
{ | ||||
if(!m_seriesDomainMap.contains(series)) { | ||||
qWarning()<<"Can not remove series. Series not found on the chart."; | ||||
return; | ||||
} | ||||
emit seriesRemoved(series); | ||||
Domain* domain = m_seriesDomainMap.take(series); | ||||
delete domain; | ||||
domain = 0; | ||||
removeSeriesIndex(series); | ||||
series->setParent(0); | ||||
series->d_ptr->m_chart = 0; | ||||
series->d_ptr->m_dataset = 0; | ||||
removeAxes(series); | ||||
} | ||||
Michal Klocek
|
r421 | |||
Michal Klocek
|
r943 | |||
Michal Klocek
|
r1695 | |||
void ChartDataSet::createSeriesIndex(QAbstractSeries* series) | ||||
{ | ||||
Tero Ahola
|
r988 | QMapIterator<int, QAbstractSeries*> i(m_indexSeriesMap); | ||
Michal Klocek
|
r910 | |||
int key=0; | ||||
while (i.hasNext()) { | ||||
i.next(); | ||||
if(i.key()!=key) { | ||||
break; | ||||
} | ||||
key++; | ||||
} | ||||
m_indexSeriesMap.insert(key,series); | ||||
Michal Klocek
|
r1695 | } | ||
Michal Klocek
|
r910 | |||
Michal Klocek
|
r1695 | void ChartDataSet::removeSeriesIndex(QAbstractSeries* series) | ||
{ | ||||
int key = seriesIndex(series); | ||||
Q_ASSERT(key!=-1); | ||||
m_indexSeriesMap.remove(key); | ||||
Michal Klocek
|
r439 | } | ||
Jani Honkonen
|
r142 | |||
Michal Klocek
|
r1577 | void ChartDataSet::createDefaultAxes() | ||
{ | ||||
Michal Klocek
|
r1588 | if(m_seriesDomainMap.isEmpty()) return; | ||
QAbstractAxis::AxisTypes typeX(0); | ||||
QAbstractAxis::AxisTypes typeY(0); | ||||
QMapIterator<QAbstractSeries*, Domain*> i(m_seriesDomainMap); | ||||
Michal Klocek
|
r1577 | while (i.hasNext()) { | ||
Marek Rosa
|
r1717 | i.next(); | ||
removeAxes(i.key()); | ||||
Michal Klocek
|
r1588 | } | ||
Michal Klocek
|
r1577 | |||
Michal Klocek
|
r1588 | i.toFront(); | ||
Michal Klocek
|
r1577 | |||
Michal Klocek
|
r1588 | while (i.hasNext()) { | ||
Marek Rosa
|
r1717 | i.next(); | ||
QAbstractAxis* axisX = m_seriesAxisXMap.value(i.key()); | ||||
QAbstractAxis* axisY = m_seriesAxisYMap.value(i.key()); | ||||
if(axisX) typeX&=axisX->type(); | ||||
else typeX|=i.key()->d_ptr->defaultAxisType(Qt::Horizontal); | ||||
if(axisY) typeY&=axisY->type(); | ||||
else typeY|=i.key()->d_ptr->defaultAxisType(Qt::Vertical); | ||||
Michal Klocek
|
r1588 | } | ||
Michal Klocek
|
r1577 | |||
Michal Klocek
|
r1695 | createAxes(typeX,Qt::Horizontal); | ||
createAxes(typeY,Qt::Vertical); | ||||
} | ||||
Michal Klocek
|
r1577 | |||
Michal Klocek
|
r1695 | void ChartDataSet::createAxes(QAbstractAxis::AxisTypes type,Qt::Orientation orientation) | ||
{ | ||||
QMapIterator<QAbstractSeries*, Domain*> i(m_seriesDomainMap); | ||||
if(type.testFlag(QAbstractAxis::AxisTypeValues) && type.testFlag(QAbstractAxis::AxisTypeCategories)) | ||||
Michal Klocek
|
r1588 | { | ||
while (i.hasNext()) { | ||||
Michal Klocek
|
r1695 | i.next(); | ||
QAbstractAxis* axis = createAxis(i.key()->d_ptr->defaultAxisType(orientation),orientation); | ||||
if(!axis) continue; | ||||
initializeAxis(axis,i.key()); | ||||
emit axisAdded(axis,i.value()); | ||||
Michal Klocek
|
r1577 | } | ||
Michal Klocek
|
r1588 | } | ||
Michal Klocek
|
r1695 | else if(!type.testFlag(QAbstractAxis::AxisTypeNoAxis)) { | ||
QAbstractAxis* axis = createAxis(QAbstractAxis::AxisType(int(type)),orientation); | ||||
Michal Klocek
|
r1588 | i.toFront(); | ||
while (i.hasNext()) { | ||||
Michal Klocek
|
r1695 | i.next(); | ||
initializeAxis(axis,i.key()); | ||||
Michal Klocek
|
r1577 | } | ||
Michal Klocek
|
r1695 | emit axisAdded(axis,i.value()); | ||
Michal Klocek
|
r1577 | } | ||
} | ||||
Michal Klocek
|
r1695 | QAbstractAxis* ChartDataSet::createAxis(QAbstractAxis::AxisType type,Qt::Orientation orientation) | ||
Michal Klocek
|
r1588 | { | ||
QAbstractAxis* axis =0; | ||||
switch(type) { | ||||
Marek Rosa
|
r1717 | case QAbstractAxis::AxisTypeValues: | ||
Michal Klocek
|
r1588 | axis = new QValuesAxis(this); | ||
break; | ||||
Marek Rosa
|
r1717 | case QAbstractAxis::AxisTypeCategories: | ||
Michal Klocek
|
r1612 | axis = new QBarCategoriesAxis(this); | ||
Michal Klocek
|
r1588 | break; | ||
Marek Rosa
|
r1717 | case QAbstractAxis::AxisTypeIntervals: | ||
axis = new QIntervalsAxis(this); | ||||
break; | ||||
case QAbstractAxis::AxisTypeDateTime: | ||||
axis = new QDateTimeAxis(this); | ||||
break; | ||||
default: | ||||
Michal Klocek
|
r1588 | axis = 0; | ||
break; | ||||
} | ||||
Michal Klocek
|
r1695 | if(axis) | ||
Michal Klocek
|
r1698 | axis->d_ptr->setOrientation(orientation); | ||
Michal Klocek
|
r1588 | |||
Michal Klocek
|
r1695 | return axis; | ||
Michal Klocek
|
r1588 | } | ||
Michal Klocek
|
r1695 | void ChartDataSet::initializeAxis(QAbstractAxis* axis,QAbstractSeries* series) | ||
Michal Klocek
|
r439 | { | ||
Michal Klocek
|
r1695 | Domain* domain = m_seriesDomainMap.value(series); | ||
Michal Klocek
|
r1725 | axis->d_ptr->m_dataset = this; | ||
Michal Klocek
|
r1695 | series->d_ptr->initializeAxis(axis); | ||
Michal Klocek
|
r1764 | axis->d_ptr->intializeDomain(domain); | ||
Michal Klocek
|
r1695 | if(axis->orientation()==Qt::Horizontal) { | ||
Michal Klocek
|
r1698 | QObject::connect(axis->d_ptr.data(),SIGNAL(updated()),domain,SLOT(handleAxisUpdated())); | ||
QObject::connect(domain,SIGNAL(updated()),axis->d_ptr.data(),SLOT(handleDomainUpdated())); | ||||
Michal Klocek
|
r1695 | m_seriesAxisXMap.insert(series,axis); | ||
Michal Klocek
|
r1556 | } | ||
Michal Klocek
|
r1695 | else { | ||
Michal Klocek
|
r1698 | QObject::connect(axis->d_ptr.data(),SIGNAL(updated()),domain,SLOT(handleAxisUpdated())); | ||
QObject::connect(domain,SIGNAL(updated()),axis->d_ptr.data(),SLOT(handleDomainUpdated())); | ||||
Michal Klocek
|
r1695 | m_seriesAxisYMap.insert(series,axis); | ||
} | ||||
Michal Klocek
|
r1698 | axis->d_ptr->emitUpdated(); | ||
Michal Klocek
|
r1588 | } | ||
void ChartDataSet::removeAxes(QAbstractSeries* series) | ||||
{ | ||||
Michal Klocek
|
r1564 | QAbstractAxis* axisX = m_seriesAxisXMap.take(series); | ||
Michal Klocek
|
r910 | |||
Michal Klocek
|
r1564 | if(axisX) { | ||
QList<QAbstractAxis*> axesX = m_seriesAxisXMap.values(); | ||||
int x = axesX.indexOf(axisX); | ||||
Michal Klocek
|
r223 | |||
Michal Klocek
|
r1564 | if(x==-1) { | ||
emit axisRemoved(axisX); | ||||
Michal Klocek
|
r1725 | axisX->d_ptr->m_dataset=0; | ||
Michal Klocek
|
r1569 | axisX->deleteLater(); | ||
Michal Klocek
|
r1564 | } | ||
Michal Klocek
|
r439 | } | ||
Michal Klocek
|
r223 | |||
Michal Klocek
|
r1564 | QAbstractAxis* axisY = m_seriesAxisYMap.take(series); | ||
if(axisY) { | ||||
QList<QAbstractAxis*> axesY = m_seriesAxisYMap.values(); | ||||
int y = axesY.indexOf(axisY); | ||||
if(y==-1) { | ||||
emit axisRemoved(axisY); | ||||
Michal Klocek
|
r1725 | axisY->d_ptr->m_dataset=0; | ||
Michal Klocek
|
r1569 | axisY->deleteLater(); | ||
Michal Klocek
|
r1564 | } | ||
Michal Klocek
|
r223 | } | ||
Michal Klocek
|
r139 | } | ||
Michal Klocek
|
r1725 | void ChartDataSet::removeAxis(QAbstractAxis* axis) | ||
{ | ||||
if(!axis->d_ptr->m_dataset) { | ||||
qWarning()<<"UnBound axis found !"; | ||||
return; | ||||
} | ||||
QMap<QAbstractSeries*, QAbstractAxis*> *seriesAxisMap; | ||||
if(axis->orientation()==Qt::Vertical) { | ||||
seriesAxisMap= &m_seriesAxisYMap; | ||||
} | ||||
else { | ||||
seriesAxisMap= &m_seriesAxisXMap; | ||||
} | ||||
QMapIterator<QAbstractSeries*, QAbstractAxis*> i(*seriesAxisMap); | ||||
while (i.hasNext()) { | ||||
i.next(); | ||||
if(i.value()==axis) { | ||||
removeSeries(i.key()); | ||||
} | ||||
} | ||||
} | ||||
Michal Klocek
|
r258 | void ChartDataSet::removeAllSeries() | ||
{ | ||||
Michal Klocek
|
r1556 | QList<QAbstractSeries*> series = m_seriesDomainMap.keys(); | ||
Tero Ahola
|
r988 | foreach(QAbstractSeries *s , series) { | ||
Michal Klocek
|
r1556 | removeSeries(s); | ||
Michal Klocek
|
r139 | } | ||
sauimone
|
r1554 | Q_ASSERT(m_seriesAxisXMap.count()==0); | ||
Michal Klocek
|
r1556 | Q_ASSERT(m_seriesAxisXMap.count()==0); | ||
Q_ASSERT(m_seriesDomainMap.count()==0); | ||||
Michal Klocek
|
r910 | |||
Michal Klocek
|
r974 | qDeleteAll(series); | ||
Michal Klocek
|
r139 | } | ||
Michal Klocek
|
r439 | void ChartDataSet::zoomInDomain(const QRectF& rect, const QSizeF& size) | ||
Michal Klocek
|
r139 | { | ||
Michal Klocek
|
r1593 | //for performance reasons block, signals and scale "full" domain one by one. Gives twice less screen updates | ||
Michal Klocek
|
r1652 | |||
Michal Klocek
|
r1593 | blockAxisSignals(true); | ||
Michal Klocek
|
r1556 | QMapIterator<QAbstractSeries*, Domain*> i(m_seriesDomainMap); | ||
Michal Klocek
|
r1593 | |||
Michal Klocek
|
r439 | while (i.hasNext()) { | ||
i.next(); | ||||
i.value()->zoomIn(rect,size); | ||||
Michal Klocek
|
r223 | } | ||
Michal Klocek
|
r1593 | |||
blockAxisSignals(false); | ||||
Michal Klocek
|
r439 | } | ||
Michal Klocek
|
r223 | |||
Michal Klocek
|
r439 | void ChartDataSet::zoomOutDomain(const QRectF& rect, const QSizeF& size) | ||
{ | ||||
Michal Klocek
|
r1593 | //for performance reasons block, signals and scale "full" domain one by one. Gives twice less screen updates | ||
blockAxisSignals(true); | ||||
Michal Klocek
|
r1556 | QMapIterator<QAbstractSeries*, Domain*> i(m_seriesDomainMap); | ||
Michal Klocek
|
r1593 | |||
Michal Klocek
|
r439 | while (i.hasNext()) { | ||
i.next(); | ||||
i.value()->zoomOut(rect,size); | ||||
} | ||||
Michal Klocek
|
r1593 | |||
blockAxisSignals(false); | ||||
} | ||||
void ChartDataSet::blockAxisSignals(bool enabled) | ||||
{ | ||||
QMapIterator<QAbstractSeries*, Domain*> i(m_seriesDomainMap); | ||||
while (i.hasNext()) { | ||||
Marek Rosa
|
r1717 | i.next(); | ||
QAbstractAxis* axisX = m_seriesAxisXMap.value(i.key()); | ||||
QAbstractAxis* axisY = m_seriesAxisYMap.value(i.key()); | ||||
Michal Klocek
|
r1738 | if(axisX) { | ||
axisX->d_ptr->blockSignals(enabled); | ||||
if(!enabled) { | ||||
axisX->d_ptr->setDirty(false); | ||||
axisX->d_ptr->emitUpdated(); | ||||
} | ||||
} | ||||
if(axisY) { | ||||
axisY->d_ptr->blockSignals(enabled); | ||||
if(!enabled) { | ||||
axisY->d_ptr->setDirty(false); | ||||
axisY->d_ptr->emitUpdated(); | ||||
} | ||||
} | ||||
Marek Rosa
|
r1717 | } | ||
Michal Klocek
|
r223 | } | ||
Michal Klocek
|
r1107 | int ChartDataSet::seriesCount(QAbstractSeries::SeriesType type) | ||
Michal Klocek
|
r910 | { | ||
int count=0; | ||||
Michal Klocek
|
r1577 | QMapIterator<QAbstractSeries*, Domain*> i(m_seriesDomainMap); | ||
Michal Klocek
|
r910 | while (i.hasNext()) { | ||
Marek Rosa
|
r1717 | i.next(); | ||
if(i.key()->type()==type) count++; | ||||
Michal Klocek
|
r910 | } | ||
return count; | ||||
} | ||||
Tero Ahola
|
r988 | int ChartDataSet::seriesIndex(QAbstractSeries *series) | ||
Tero Ahola
|
r538 | { | ||
Tero Ahola
|
r988 | QMapIterator<int, QAbstractSeries*> i(m_indexSeriesMap); | ||
Michal Klocek
|
r910 | while (i.hasNext()) { | ||
i.next(); | ||||
if (i.value() == series) | ||||
return i.key(); | ||||
Tero Ahola
|
r538 | } | ||
Tero Ahola
|
r825 | return -1; | ||
Tero Ahola
|
r538 | } | ||
sauimone
|
r1554 | QAbstractAxis* ChartDataSet::axisX(QAbstractSeries *series) const | ||
{ | ||||
Michal Klocek
|
r1641 | if(series == 0) { | ||
Marek Rosa
|
r1717 | QMapIterator<QAbstractSeries*, QAbstractAxis *> i(m_seriesAxisXMap); | ||
Michal Klocek
|
r1641 | |||
Marek Rosa
|
r1717 | while (i.hasNext()) { | ||
i.next(); | ||||
if(i.value()->isVisible()) return i.value(); | ||||
} | ||||
return 0; | ||||
Michal Klocek
|
r1641 | } | ||
sauimone
|
r1554 | return m_seriesAxisXMap.value(series); | ||
} | ||||
Michal Klocek
|
r1541 | QAbstractAxis* ChartDataSet::axisY(QAbstractSeries *series) const | ||
Michal Klocek
|
r223 | { | ||
Michal Klocek
|
r1641 | if(series == 0) { | ||
QMapIterator<QAbstractSeries*, QAbstractAxis *> i(m_seriesAxisYMap); | ||||
while (i.hasNext()) { | ||||
i.next(); | ||||
if(i.value()->isVisible()) return i.value(); | ||||
} | ||||
return 0; | ||||
} | ||||
sauimone
|
r1554 | return m_seriesAxisYMap.value(series); | ||
Michal Klocek
|
r223 | } | ||
Michal Klocek
|
r1695 | void ChartDataSet::setAxis(QAbstractSeries *series, QAbstractAxis *axis, Qt::Orientation orientation) | ||
sauimone
|
r1554 | { | ||
Michal Klocek
|
r1564 | Q_ASSERT(axis); | ||
Michal Klocek
|
r1695 | |||
Michal Klocek
|
r1699 | if(!series) { | ||
qWarning() << "Series not found on the chart."; | ||||
return; | ||||
} | ||||
Michal Klocek
|
r1577 | Domain* domain = m_seriesDomainMap.value(series); | ||
Michal Klocek
|
r1564 | |||
Michal Klocek
|
r1577 | if(!domain) { | ||
qWarning() << "Series not found on the chart."; | ||||
return; | ||||
Michal Klocek
|
r1564 | } | ||
Michal Klocek
|
r1695 | if(orientation==Qt::Horizontal && axis->orientation()==Qt::Vertical) { | ||
Michal Klocek
|
r1577 | qWarning()<<"Axis already defined as axis Y"; | ||
return; | ||||
Michal Klocek
|
r1564 | } | ||
Michal Klocek
|
r1695 | if(orientation==Qt::Vertical && axis->orientation()==Qt::Horizontal) { | ||
qWarning()<<"Axis already defined as axis X"; | ||||
return; | ||||
Michal Klocek
|
r1564 | } | ||
Michal Klocek
|
r1698 | axis->d_ptr->setOrientation(orientation); | ||
sauimone
|
r1554 | |||
Michal Klocek
|
r1695 | QMap<QAbstractSeries*, QAbstractAxis*> *seriesAxisMap; | ||
Michal Klocek
|
r1564 | |||
Michal Klocek
|
r1695 | if(orientation==Qt::Vertical) { | ||
seriesAxisMap= &m_seriesAxisYMap; | ||||
}else{ | ||||
seriesAxisMap= &m_seriesAxisXMap; | ||||
Michal Klocek
|
r1564 | } | ||
Michal Klocek
|
r1695 | QAbstractAxis *oldAxis = seriesAxisMap->take(series); | ||
QList<QAbstractAxis*> axes = seriesAxisMap->values(); | ||||
Michal Klocek
|
r1577 | if(oldAxis) { | ||
Michal Klocek
|
r1695 | if(axes.indexOf(oldAxis)==-1) { | ||
Michal Klocek
|
r1577 | emit axisRemoved(oldAxis); | ||
Michal Klocek
|
r1729 | oldAxis->disconnect(); | ||
QObject::disconnect(domain,0,oldAxis,0); | ||||
Michal Klocek
|
r1725 | oldAxis->d_ptr->m_dataset=0; | ||
Michal Klocek
|
r1577 | oldAxis->deleteLater(); | ||
} | ||||
} | ||||
Michal Klocek
|
r1564 | |||
Michal Klocek
|
r1695 | if(axes.indexOf(axis)==-1) { | ||
initializeAxis(axis,series); | ||||
Michal Klocek
|
r1564 | emit axisAdded(axis,domain); | ||
Michal Klocek
|
r1695 | }else{ | ||
initializeAxis(axis,series); | ||||
Michal Klocek
|
r1564 | } | ||
} | ||||
Domain* ChartDataSet::domain(QAbstractSeries *series) const | ||||
{ | ||||
return m_seriesDomainMap.value(series); | ||||
Michal Klocek
|
r1542 | } | ||
Michal Klocek
|
r1267 | void ChartDataSet::scrollDomain(qreal dx,qreal dy,const QSizeF& size) | ||
Michal Klocek
|
r531 | { | ||
Michal Klocek
|
r1593 | blockAxisSignals(true); | ||
Michal Klocek
|
r1556 | QMapIterator<QAbstractSeries*, Domain*> i(m_seriesDomainMap); | ||
Michal Klocek
|
r1078 | while (i.hasNext()) { | ||
i.next(); | ||||
i.value()->move(dx,dy,size); | ||||
} | ||||
Michal Klocek
|
r1593 | blockAxisSignals(false); | ||
Michal Klocek
|
r1078 | } | ||
QList<QAbstractSeries*> ChartDataSet::series() const | ||||
{ | ||||
Michal Klocek
|
r1661 | return m_seriesDomainMap.keys(); | ||
Michal Klocek
|
r531 | } | ||
sauimone
|
r1263 | void ChartDataSet::updateSeries(QAbstractSeries *series) | ||
{ | ||||
emit seriesUpdated(series); | ||||
} | ||||
Michal Klocek
|
r131 | #include "moc_chartdataset_p.cpp" | ||
QTCOMMERCIALCHART_END_NAMESPACE | ||||