##// END OF EJS Templates
Bugfix precision error when calculating layout in valueaxis
Bugfix precision error when calculating layout in valueaxis

File last commit:

r2131:730996457e67
r2144:c7788799f12f
Show More
chartpresenter.cpp
438 lines | 10.6 KiB | text/x-c | CppLexer
/ src / chartpresenter.cpp
/****************************************************************************
**
** 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$
**
****************************************************************************/
#include "chartpresenter_p.h"
#include "qchart.h"
#include "qchart_p.h"
#include "qabstractaxis.h"
#include "qabstractaxis_p.h"
#include "chartdataset_p.h"
#include "charttheme_p.h"
#include "chartanimation_p.h"
#include "qabstractseries_p.h"
#include "qareaseries.h"
#include "chartaxis_p.h"
#include "chartbackground_p.h"
#include "chartlayout_p.h"
#include "charttitle_p.h"
#include <QTimer>
QTCOMMERCIALCHART_BEGIN_NAMESPACE
ChartPresenter::ChartPresenter(QChart *chart, ChartDataSet *dataset)
: QObject(chart),
m_chart(chart),
m_dataset(dataset),
m_chartTheme(0),
m_options(QChart::NoAnimation),
m_state(ShowState),
m_layout(new ChartLayout(this)),
m_background(0),
m_title(0)
{
}
ChartPresenter::~ChartPresenter()
{
delete m_chartTheme;
}
void ChartPresenter::handleAxisAdded(QAbstractAxis *axis, Domain *domain)
{
ChartAxis *item = axis->d_ptr->createGraphics(this);
item->setDomain(domain);
if (m_options.testFlag(QChart::GridAxisAnimations))
item->setAnimation(new AxisAnimation(item));
QObject::connect(domain, SIGNAL(updated()), item, SLOT(handleDomainUpdated()));
QObject::connect(axis, SIGNAL(visibleChanged(bool)), this, SLOT(handleAxisVisibleChanged(bool)));
//initialize
domain->emitUpdated();
m_chartTheme->decorate(axis);
axis->d_ptr->setDirty(false);
axis->d_ptr->emitUpdated();
m_axisItems.insert(axis, item);
selectVisibleAxis();
m_layout->invalidate();
}
void ChartPresenter::handleAxisRemoved(QAbstractAxis *axis)
{
ChartAxis *item = m_axisItems.take(axis);
Q_ASSERT(item);
selectVisibleAxis();
item->hide();
item->disconnect();
QObject::disconnect(this, 0, item, 0);
item->deleteLater();
}
void ChartPresenter::handleSeriesAdded(QAbstractSeries *series, Domain *domain)
{
ChartElement *item = series->d_ptr->createGraphics(this);
Q_ASSERT(item);
item->setDomain(domain);
QObject::connect(domain, SIGNAL(updated()), item, SLOT(handleDomainUpdated()));
//initialize
item->handleDomainUpdated();
m_chartItems.insert(series, item);
m_layout->invalidate();
}
void ChartPresenter::handleSeriesRemoved(QAbstractSeries *series)
{
ChartElement *item = m_chartItems.take(series);
Q_ASSERT(item);
item->deleteLater();
}
void ChartPresenter::selectVisibleAxis()
{
QMapIterator<QAbstractAxis *, ChartAxis *> i(m_axisItems);
while (i.hasNext()) {
i.next();
// i.key()->hide();
i.key()->show();
}
// i.toFront();
// bool axisX=false;
// bool axisY=false;
// while (i.hasNext()) {
// i.next();
// if(i.key()->orientation()==Qt::Vertical && !axisY) {
// axisY=true;
// i.key()->show();
// }
// if(i.key()->orientation()==Qt::Horizontal && !axisX) {
// axisX=true;
// i.key()->show();
// }
// }
}
void ChartPresenter::handleAxisVisibleChanged(bool visible)
{
QAbstractAxis *axis = static_cast<QAbstractAxis *>(sender());
Q_ASSERT(axis);
if (visible) {
QMapIterator<QAbstractAxis *, ChartAxis *> i(m_axisItems);
while (i.hasNext()) {
i.next();
if (i.key() == axis)
continue;
if (i.key()->orientation() == axis->orientation())
i.key()->setVisible(false);
}
}
}
void ChartPresenter::setTheme(QChart::ChartTheme theme, bool force)
{
if (m_chartTheme && m_chartTheme->id() == theme)
return;
delete m_chartTheme;
m_chartTheme = ChartTheme::createTheme(theme);
m_chartTheme->setForced(force);
m_chartTheme->decorate(m_chart);
m_chartTheme->decorate(m_chart->legend());
resetAllElements();
// We do not want "force" to stay on.
// Bar/pie are calling decorate when adding/removing slices/bars which means
// that to preserve users colors "force" must not be on.
m_chartTheme->setForced(false);
}
QChart::ChartTheme ChartPresenter::theme()
{
return m_chartTheme->id();
}
void ChartPresenter::setAnimationOptions(QChart::AnimationOptions options)
{
if (m_options != options) {
m_options = options;
resetAllElements();
}
}
void ChartPresenter::resetAllElements()
{
QMapIterator<QAbstractAxis *, ChartAxis *> i(m_axisItems);
while (i.hasNext()) {
i.next();
Domain *domain = i.value()->domain();
QAbstractAxis *axis = i.key();
handleAxisRemoved(axis);
handleAxisAdded(axis, domain);
}
QMapIterator<QAbstractSeries *, ChartElement *> j(m_chartItems);
while (j.hasNext()) {
j.next();
Domain *domain = j.value()->domain();
QAbstractSeries *series = j.key();
handleSeriesRemoved(series);
handleSeriesAdded(series, domain);
}
layout()->invalidate();
}
void ChartPresenter::zoomIn(qreal factor)
{
QRectF rect = m_layout->chartsGeometry();
rect.setWidth(rect.width() / factor);
rect.setHeight(rect.height() / factor);
rect.moveCenter(m_layout->chartsGeometry().center());
zoomIn(rect);
}
void ChartPresenter::zoomIn(const QRectF &rect)
{
QRectF r = rect.normalized();
r.translate(-m_layout->chartsGeometry().topLeft());
if (!r.isValid())
return;
m_state = ZoomInState;
m_statePoint = QPointF(r.center().x() / m_layout->chartsGeometry().width(), r.center().y() / m_layout->chartsGeometry().height());
m_dataset->zoomInDomain(r, m_layout->chartsGeometry().size());
m_state = ShowState;
}
void ChartPresenter::zoomOut(qreal factor)
{
m_state = ZoomOutState;
QRectF chartRect;
chartRect.setSize(m_layout->chartsGeometry().size());
QRectF rect;
rect.setSize(chartRect.size() / factor);
rect.moveCenter(chartRect.center());
if (!rect.isValid())
return;
m_statePoint = QPointF(rect.center().x() / m_layout->chartsGeometry().width(), rect.center().y() / m_layout->chartsGeometry().height());
m_dataset->zoomOutDomain(rect, chartRect.size());
m_state = ShowState;
}
void ChartPresenter::scroll(qreal dx, qreal dy)
{
if (dx < 0) m_state = ScrollLeftState;
if (dx > 0) m_state = ScrollRightState;
if (dy < 0) m_state = ScrollUpState;
if (dy > 0) m_state = ScrollDownState;
m_dataset->scrollDomain(dx, dy, m_layout->chartsGeometry().size());
m_state = ShowState;
}
QChart::AnimationOptions ChartPresenter::animationOptions() const
{
return m_options;
}
void ChartPresenter::createBackgroundItem()
{
if (!m_background) {
m_background = new ChartBackground(rootItem());
m_background->setPen(Qt::NoPen);
m_background->setZValue(ChartPresenter::BackgroundZValue);
}
}
void ChartPresenter::createTitleItem()
{
if (!m_title) {
m_title = new ChartTitle(rootItem());
m_title->setZValue(ChartPresenter::BackgroundZValue);
}
}
void ChartPresenter::handleAnimationFinished()
{
m_animations.removeAll(qobject_cast<ChartAnimation *>(sender()));
if (m_animations.empty())
emit animationsFinished();
}
void ChartPresenter::startAnimation(ChartAnimation *animation)
{
if (animation->state() != QAbstractAnimation::Stopped) animation->stop();
QObject::connect(animation, SIGNAL(finished()), this, SLOT(handleAnimationFinished()), Qt::UniqueConnection);
if (!m_animations.isEmpty())
m_animations.append(animation);
QTimer::singleShot(0, animation, SLOT(start()));
}
void ChartPresenter::setBackgroundBrush(const QBrush &brush)
{
createBackgroundItem();
m_background->setBrush(brush);
m_layout->invalidate();
}
QBrush ChartPresenter::backgroundBrush() const
{
if (!m_background)
return QBrush();
return m_background->brush();
}
void ChartPresenter::setBackgroundPen(const QPen &pen)
{
createBackgroundItem();
m_background->setPen(pen);
m_layout->invalidate();
}
QPen ChartPresenter::backgroundPen() const
{
if (!m_background)
return QPen();
return m_background->pen();
}
void ChartPresenter::setTitle(const QString &title)
{
createTitleItem();
m_title->setText(title);
m_layout->invalidate();
}
QString ChartPresenter::title() const
{
if (!m_title)
return QString();
return m_title->text();
}
void ChartPresenter::setTitleFont(const QFont &font)
{
createTitleItem();
m_title->setFont(font);
m_layout->invalidate();
}
QFont ChartPresenter::titleFont() const
{
if (!m_title)
return QFont();
return m_title->font();
}
void ChartPresenter::setTitleBrush(const QBrush &brush)
{
createTitleItem();
m_title->setBrush(brush);
m_layout->invalidate();
}
QBrush ChartPresenter::titleBrush() const
{
if (!m_title)
return QBrush();
return m_title->brush();
}
void ChartPresenter::setBackgroundVisible(bool visible)
{
createBackgroundItem();
m_background->setVisible(visible);
}
bool ChartPresenter::isBackgroundVisible() const
{
if (!m_background)
return false;
return m_background->isVisible();
}
void ChartPresenter::setBackgroundDropShadowEnabled(bool enabled)
{
createBackgroundItem();
m_background->setDropShadowEnabled(enabled);
}
bool ChartPresenter::isBackgroundDropShadowEnabled() const
{
if (!m_background)
return false;
return m_background->isDropShadowEnabled();
}
ChartLayout *ChartPresenter::layout()
{
return m_layout;
}
QLegend *ChartPresenter::legend()
{
return m_chart->legend();
}
void ChartPresenter::setVisible(bool visible)
{
m_chart->setVisible(visible);
}
ChartBackground *ChartPresenter::backgroundElement()
{
return m_background;
}
QList<ChartAxis *> ChartPresenter::axisItems() const
{
return m_axisItems.values();
}
QList<ChartElement *> ChartPresenter::chartItems() const
{
return m_chartItems.values();
}
ChartTitle *ChartPresenter::titleElement()
{
return m_title;
}
#include "moc_chartpresenter_p.cpp"
QTCOMMERCIALCHART_END_NAMESPACE