##// END OF EJS Templates
Added hovered support to scatter series
Added hovered support to scatter series

File last commit:

r2242:2523dbee7254
r2261:f149ef311d71
Show More
chartaxis.cpp
476 lines | 12.8 KiB | text/x-c | CppLexer
/****************************************************************************
**
** 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 "chartaxis_p.h"
#include "qabstractaxis.h"
#include "qabstractaxis_p.h"
#include "chartpresenter_p.h"
#include "chartlayout_p.h"
#include "domain_p.h"
#include <qmath.h>
#include <QDateTime>
#include <QValueAxis>
#include <QGraphicsLayout>
#include <QFontMetrics>
QTCOMMERCIALCHART_BEGIN_NAMESPACE
ChartAxis::ChartAxis(QAbstractAxis *axis, ChartPresenter *presenter, bool intervalAxis)
: ChartElement(presenter),
m_chartAxis(axis),
m_labelsAngle(0),
m_grid(new QGraphicsItemGroup(presenter->rootItem())),
m_shades(new QGraphicsItemGroup(presenter->rootItem())),
m_labels(new QGraphicsItemGroup(presenter->rootItem())),
m_arrow(new QGraphicsItemGroup(presenter->rootItem())),
m_title(new QGraphicsSimpleTextItem(presenter->rootItem())),
m_min(0),
m_max(0),
m_animation(0),
m_labelPadding(5),
m_intervalAxis(intervalAxis)
{
//initial initialization
m_arrow->setZValue(ChartPresenter::AxisZValue);
m_arrow->setHandlesChildEvents(false);
m_labels->setZValue(ChartPresenter::AxisZValue);
m_shades->setZValue(ChartPresenter::ShadesZValue);
m_grid->setZValue(ChartPresenter::GridZValue);
m_title->setZValue(ChartPresenter::GridZValue);
QObject::connect(m_chartAxis->d_ptr.data(), SIGNAL(updated()), this, SLOT(handleAxisUpdated()));
QGraphicsSimpleTextItem item;
m_font = item.font();
}
ChartAxis::~ChartAxis()
{
}
void ChartAxis::setAnimation(AxisAnimation *animation)
{
m_animation = animation;
}
void ChartAxis::setLayout(QVector<qreal> &layout)
{
m_layoutVector = layout;
}
void ChartAxis::createItems(int count)
{
if (m_arrow->childItems().size() == 0)
m_arrow->addToGroup(new AxisItem(this, presenter()->rootItem()));
if (m_intervalAxis && m_grid->childItems().size() == 0) {
for (int i = 0 ; i < 2 ; i ++)
m_grid->addToGroup(new QGraphicsLineItem(presenter()->rootItem()));
}
for (int i = 0; i < count; ++i) {
m_grid->addToGroup(new QGraphicsLineItem(presenter()->rootItem()));
QGraphicsSimpleTextItem *label = new QGraphicsSimpleTextItem(presenter()->rootItem());
label->setFont(m_font);
m_labels->addToGroup(label);
m_arrow->addToGroup(new QGraphicsLineItem(presenter()->rootItem()));
if ((m_grid->childItems().size()) % 2 && m_grid->childItems().size() > 2)
m_shades->addToGroup(new QGraphicsRectItem(presenter()->rootItem()));
}
}
void ChartAxis::deleteItems(int count)
{
QList<QGraphicsItem *> lines = m_grid->childItems();
QList<QGraphicsItem *> labels = m_labels->childItems();
QList<QGraphicsItem *> shades = m_shades->childItems();
QList<QGraphicsItem *> axis = m_arrow->childItems();
for (int i = 0; i < count; ++i) {
if (lines.size() % 2 && lines.size() > 1)
delete(shades.takeLast());
delete(lines.takeLast());
delete(labels.takeLast());
delete(axis.takeLast());
}
}
void ChartAxis::updateLayout(QVector<qreal> &layout)
{
int diff = m_layoutVector.size() - layout.size();
if (diff > 0)
deleteItems(diff);
else if (diff < 0)
createItems(-diff);
if (diff < 0) handleAxisUpdated();
if (m_animation) {
switch (presenter()->state()) {
case ChartPresenter::ZoomInState:
m_animation->setAnimationType(AxisAnimation::ZoomInAnimation);
m_animation->setAnimationPoint(presenter()->statePoint());
break;
case ChartPresenter::ZoomOutState:
m_animation->setAnimationType(AxisAnimation::ZoomOutAnimation);
m_animation->setAnimationPoint(presenter()->statePoint());
break;
case ChartPresenter::ScrollUpState:
case ChartPresenter::ScrollLeftState:
m_animation->setAnimationType(AxisAnimation::MoveBackwordAnimation);
break;
case ChartPresenter::ScrollDownState:
case ChartPresenter::ScrollRightState:
m_animation->setAnimationType(AxisAnimation::MoveForwardAnimation);
break;
case ChartPresenter::ShowState:
m_animation->setAnimationType(AxisAnimation::DefaultAnimation);
break;
}
m_animation->setValues(m_layoutVector, layout);
presenter()->startAnimation(m_animation);
} else {
setLayout(layout);
updateGeometry();
}
}
void ChartAxis::setArrowOpacity(qreal opacity)
{
m_arrow->setOpacity(opacity);
}
qreal ChartAxis::arrowOpacity() const
{
return m_arrow->opacity();
}
void ChartAxis::setArrowVisibility(bool visible)
{
m_arrow->setOpacity(visible);
}
void ChartAxis::setGridOpacity(qreal opacity)
{
m_grid->setOpacity(opacity);
}
qreal ChartAxis::gridOpacity() const
{
return m_grid->opacity();
}
void ChartAxis::setGridVisibility(bool visible)
{
m_grid->setOpacity(visible);
}
void ChartAxis::setLabelsOpacity(qreal opacity)
{
m_labels->setOpacity(opacity);
}
qreal ChartAxis::labelsOpacity() const
{
return m_labels->opacity();
}
void ChartAxis::setLabelsVisibility(bool visible)
{
m_labels->setOpacity(visible);
}
void ChartAxis::setShadesOpacity(qreal opacity)
{
m_shades->setOpacity(opacity);
}
qreal ChartAxis::shadesOpacity() const
{
return m_shades->opacity();
}
void ChartAxis::setShadesVisibility(bool visible)
{
m_shades->setVisible(visible);
}
void ChartAxis::setLabelsAngle(int angle)
{
foreach (QGraphicsItem *item, m_labels->childItems())
item->setRotation(angle);
m_labelsAngle = angle;
}
void ChartAxis::setLabelsPen(const QPen &pen)
{
foreach (QGraphicsItem *item , m_labels->childItems())
static_cast<QGraphicsSimpleTextItem *>(item)->setPen(pen);
}
void ChartAxis::setLabelsBrush(const QBrush &brush)
{
foreach (QGraphicsItem *item , m_labels->childItems())
static_cast<QGraphicsSimpleTextItem *>(item)->setBrush(brush);
}
void ChartAxis::setLabelsFont(const QFont &font)
{
if (m_font != font) {
m_font = font;
foreach (QGraphicsItem *item , m_labels->childItems())
static_cast<QGraphicsSimpleTextItem *>(item)->setFont(font);
QGraphicsLayoutItem::updateGeometry();
presenter()->layout()->invalidate();
}
}
void ChartAxis::setShadesBrush(const QBrush &brush)
{
foreach (QGraphicsItem *item , m_shades->childItems())
static_cast<QGraphicsRectItem *>(item)->setBrush(brush);
}
void ChartAxis::setShadesPen(const QPen &pen)
{
foreach (QGraphicsItem *item , m_shades->childItems())
static_cast<QGraphicsRectItem *>(item)->setPen(pen);
}
void ChartAxis::setArrowPen(const QPen &pen)
{
foreach (QGraphicsItem *item , m_arrow->childItems())
static_cast<QGraphicsLineItem *>(item)->setPen(pen);
}
void ChartAxis::setGridPen(const QPen &pen)
{
foreach (QGraphicsItem *item , m_grid->childItems())
static_cast<QGraphicsLineItem *>(item)->setPen(pen);
}
void ChartAxis::setLabelPadding(int padding)
{
m_labelPadding = padding;
}
bool ChartAxis::isEmpty()
{
return m_axisRect.isEmpty() || m_gridRect.isEmpty() || qFuzzyCompare(m_min, m_max);
}
void ChartAxis::handleDomainUpdated()
{
Domain *domain = qobject_cast<Domain *>(sender());
qreal min(0);
qreal max(0);
if (m_chartAxis->orientation() == Qt::Horizontal) {
min = domain->minX();
max = domain->maxX();
} else if (m_chartAxis->orientation() == Qt::Vertical) {
min = domain->minY();
max = domain->maxY();
}
if (!qFuzzyCompare(m_min, min) || !qFuzzyCompare(m_max, max)) {
m_min = min;
m_max = max;
if (!isEmpty()) {
QVector<qreal> layout = calculateLayout();
updateLayout(layout);
QSizeF before = effectiveSizeHint(Qt::PreferredSize);
QSizeF after = sizeHint(Qt::PreferredSize);
if (before != after) {
QGraphicsLayoutItem::updateGeometry();
//we don't want to call invalidate on layout, since it will change minimum size of component,
//which we would like to avoid since it causes nasty flips when scrolling or zooming,
//instead recalculate layout and use plotArea for extra space.
presenter()->layout()->setGeometry(presenter()->layout()->geometry());
}
}
}
}
void ChartAxis::handleAxisUpdated()
{
if (isEmpty())
return;
bool visible = m_chartAxis->isVisible();
//TODO: split this into separate signal/slots ?
setArrowVisibility(visible && m_chartAxis->isLineVisible());
setGridVisibility(visible && m_chartAxis->isGridLineVisible());
setLabelsVisibility(visible && m_chartAxis->labelsVisible());
setShadesVisibility(visible && m_chartAxis->shadesVisible());
setLabelsAngle(m_chartAxis->labelsAngle());
setArrowPen(m_chartAxis->linePen());
setLabelsPen(m_chartAxis->labelsPen());
setLabelsBrush(m_chartAxis->labelsBrush());
setLabelsFont(m_chartAxis->labelsFont());
setGridPen(m_chartAxis->gridLinePen());
setShadesPen(m_chartAxis->shadesPen());
setShadesBrush(m_chartAxis->shadesBrush());
setTitleText(m_chartAxis->title());
setTitleFont(m_chartAxis->titleFont());
setTitlePen(m_chartAxis->titlePen());
setTitleBrush(m_chartAxis->titleBrush());
}
void ChartAxis::setTitleText(const QString &title)
{
if (m_titleText != title) {
m_titleText = title;
QGraphicsLayoutItem::updateGeometry();
presenter()->layout()->invalidate();
}
}
void ChartAxis::setTitlePen(const QPen &pen)
{
m_title->setPen(pen);
}
void ChartAxis::setTitleBrush(const QBrush &brush)
{
m_title->setBrush(brush);
}
void ChartAxis::setTitleFont(const QFont &font)
{
if(m_title->font() != font){
m_title->setFont(font);
QGraphicsLayoutItem::updateGeometry();
presenter()->layout()->invalidate();
}
}
QFont ChartAxis::titleFont() const
{
return m_title->font();
}
void ChartAxis::hide()
{
setArrowVisibility(false);
setGridVisibility(false);
setLabelsVisibility(false);
setShadesVisibility(false);
}
void ChartAxis::setGeometry(const QRectF &axis, const QRectF &grid)
{
m_gridRect = grid;
m_axisRect = axis;
if (isEmpty())
return;
QVector<qreal> layout = calculateLayout();
updateLayout(layout);
}
void ChartAxis::axisSelected()
{
//TODO: axis clicked;
}
Qt::Orientation ChartAxis::orientation() const
{
return m_chartAxis->orientation();
}
Qt::Alignment ChartAxis::alignment() const
{
return m_chartAxis->alignment();
}
bool ChartAxis::isVisible()
{
return m_chartAxis->isVisible();
}
void ChartAxis::setLabels(const QStringList &labels)
{
m_labelsList = labels;
}
QStringList ChartAxis::createValueLabels(int ticks) const
{
Q_ASSERT(m_max > m_min);
Q_ASSERT(ticks > 1);
QStringList labels;
int n = qMax(int(-qFloor(log10((m_max - m_min) / (ticks - 1)))), 0);
n++;
QValueAxis *axis = qobject_cast<QValueAxis *>(m_chartAxis);
QString format = axis->labelFormat();
if (format.isNull()) {
for (int i = 0; i < ticks; i++) {
qreal value = m_min + (i * (m_max - m_min) / (ticks - 1));
labels << QString::number(value, 'f', n);
}
} else {
QByteArray array = format.toLatin1();
for (int i = 0; i < ticks; i++) {
qreal value = m_min + (i * (m_max - m_min) / (ticks - 1));
labels << QString().sprintf(array, value);
}
}
return labels;
}
QStringList ChartAxis::createDateTimeLabels(const QString &format, int ticks) const
{
Q_ASSERT(m_max > m_min);
Q_ASSERT(ticks > 1);
QStringList labels;
int n = qMax(int(-floor(log10((m_max - m_min) / (ticks - 1)))), 0);
n++;
for (int i = 0; i < ticks; i++) {
qreal value = m_min + (i * (m_max - m_min) / (ticks - 1));
labels << QDateTime::fromMSecsSinceEpoch(value).toString(format);
}
return labels;
}
QSizeF ChartAxis::sizeHint(Qt::SizeHint which, const QSizeF &constraint) const
{
Q_UNUSED(which);
Q_UNUSED(constraint);
return QSizeF();
}
#include "moc_chartaxis_p.cpp"
QTCOMMERCIALCHART_END_NAMESPACE