/**************************************************************************** ** ** Copyright (C) 2016 The Qt Company Ltd. ** Contact: https://www.qt.io/licensing/ ** ** This file is part of the Qt Charts module of the Qt Toolkit. ** ** $QT_BEGIN_LICENSE:GPL$ ** Commercial License Usage ** Licensees holding valid commercial Qt licenses may use this file in ** accordance with the commercial license agreement provided with the ** Software or, alternatively, in accordance with the terms contained in ** a written agreement between you and The Qt Company. For licensing terms ** and conditions see https://www.qt.io/terms-conditions. For further ** information use the contact form at https://www.qt.io/contact-us. ** ** GNU General Public License Usage ** Alternatively, this file may be used under the terms of the GNU ** General Public License version 3 or (at your option) any later version ** approved by the KDE Free Qt Foundation. The licenses are as published by ** the Free Software Foundation and appearing in the file LICENSE.GPL3 ** included in the packaging of this file. Please review the following ** information to ensure the GNU General Public License requirements will ** be met: https://www.gnu.org/licenses/gpl-3.0.html. ** ** $QT_END_LICENSE$ ** ****************************************************************************/ #include #include #include #include #include #include #include #include #include #include #include #include QT_CHARTS_BEGIN_NAMESPACE CartesianChartAxis::CartesianChartAxis(QAbstractAxis *axis, QGraphicsItem *item , bool intervalAxis) : ChartAxisElement(axis, item, intervalAxis) { Q_ASSERT(item); } CartesianChartAxis::~CartesianChartAxis() { } void CartesianChartAxis::createItems(int count) { if (arrowItems().size() == 0) { QGraphicsLineItem *arrow = new LineArrowItem(this, this); arrow->setPen(axis()->linePen()); arrowGroup()->addToGroup(arrow); } if (intervalAxis() && gridItems().size() == 0) { for (int i = 0 ; i < 2 ; i ++){ QGraphicsLineItem *item = new QGraphicsLineItem(this); item->setPen(axis()->gridLinePen()); gridGroup()->addToGroup(item); QGraphicsRectItem *shades = new QGraphicsRectItem(this); shades->setPen(axis()->shadesPen()); shades->setBrush(axis()->shadesBrush()); shadeGroup()->addToGroup(shades); } } QGraphicsTextItem *title = titleItem(); title->setFont(axis()->titleFont()); title->setDefaultTextColor(axis()->titleBrush().color()); title->setHtml(axis()->titleText()); for (int i = 0; i < count; ++i) { QGraphicsLineItem *arrow = new QGraphicsLineItem(this); QGraphicsLineItem *grid = new QGraphicsLineItem(this); QGraphicsTextItem *label = new QGraphicsTextItem(this); label->document()->setDocumentMargin(ChartPresenter::textMargin()); arrow->setPen(axis()->linePen()); grid->setPen(axis()->gridLinePen()); label->setFont(axis()->labelsFont()); label->setDefaultTextColor(axis()->labelsBrush().color()); label->setRotation(axis()->labelsAngle()); arrowGroup()->addToGroup(arrow); gridGroup()->addToGroup(grid); labelGroup()->addToGroup(label); if (gridItems().size() == 1 || (((gridItems().size() + 1) % 2) && gridItems().size() > 0)) { QGraphicsRectItem *shades = new QGraphicsRectItem(this); shades->setPen(axis()->shadesPen()); shades->setBrush(axis()->shadesBrush()); shadeGroup()->addToGroup(shades); } } } void CartesianChartAxis::updateMinorTickItems() { int currentCount = minorArrowItems().size(); int expectedCount = 0; if (axis()->type() == QAbstractAxis::AxisTypeValue) { QValueAxis *valueAxis = qobject_cast(axis()); if (valueAxis->tickType() == QValueAxis::TicksFixed) { expectedCount = valueAxis->minorTickCount() * (valueAxis->tickCount() - 1); expectedCount = qMax(expectedCount, 0); } else { const qreal interval = valueAxis->tickInterval(); qreal firstMajorTick = valueAxis->tickAnchor(); const qreal max = valueAxis->max(); const qreal min = valueAxis->min(); const int _minorTickCount = valueAxis->minorTickCount(); if (min < firstMajorTick) firstMajorTick = firstMajorTick - qCeil((firstMajorTick - min) / interval) * interval; else firstMajorTick = firstMajorTick + int((min - firstMajorTick) / interval) * interval; const qreal deltaMinor = interval / qreal(_minorTickCount + 1); qreal minorTick = firstMajorTick + deltaMinor; int minorCounter = 0; while (minorTick < min) { minorTick += deltaMinor; minorCounter++; } QVector points; // Calculate the points on axis value space. Conversion to graphical points // will be done on axis specific geometry update function while (minorTick <= max || qFuzzyCompare(minorTick, max)) { if (minorCounter < _minorTickCount) { expectedCount++; minorCounter++; points << (minorTick - min); } else { minorCounter = 0; } minorTick += deltaMinor; } setDynamicMinorTickLayout(points); } } else if (axis()->type() == QAbstractAxis::AxisTypeLogValue) { QLogValueAxis *logValueAxis = qobject_cast(axis()); int minorTickCount = logValueAxis->minorTickCount(); if (minorTickCount < 0) minorTickCount = qMax(int(qFloor(logValueAxis->base()) - 2.0), 0); expectedCount = minorTickCount * (logValueAxis->tickCount() + 1); expectedCount = qMax(expectedCount, logValueAxis->minorTickCount()); } else { // minor ticks are not supported return; } int diff = expectedCount - currentCount; if (diff > 0) { for (int i = 0; i < diff; ++i) { QGraphicsLineItem *minorGridLineItem = new QGraphicsLineItem(this); minorGridLineItem->setPen(axis()->minorGridLinePen()); minorGridGroup()->addToGroup(minorGridLineItem); QGraphicsLineItem *minorArrowLineItem = new QGraphicsLineItem(this); minorArrowLineItem->setPen(axis()->linePen()); minorArrowGroup()->addToGroup(minorArrowLineItem); } } else { QList minorGridItemsList = minorGridItems(); QList minorArrowItemsList = minorArrowItems(); for (int i = 0; i > diff; --i) { if (!minorGridItemsList.isEmpty()) delete minorGridItemsList.takeLast(); if (!minorArrowItemsList.isEmpty()) delete minorArrowItemsList.takeLast(); } } } void CartesianChartAxis::deleteItems(int count) { QList lines = gridItems(); QList labels = labelItems(); QList shades = shadeItems(); QList axis = arrowItems(); for (int i = 0; i < count; ++i) { if (lines.size() == 1 || (((lines.size() + 1) % 2) && lines.size() > 0)) delete(shades.takeLast()); delete(lines.takeLast()); delete(labels.takeLast()); delete(axis.takeLast()); } } void CartesianChartAxis::updateLayout(QVector &layout) { int diff = ChartAxisElement::layout().size() - layout.size(); if (diff > 0) deleteItems(diff); else if (diff <= 0) createItems(-diff); updateMinorTickItems(); if (animation()) { switch (presenter()->state()) { case ChartPresenter::ZoomInState: animation()->setAnimationType(AxisAnimation::ZoomInAnimation); animation()->setAnimationPoint(presenter()->statePoint()); break; case ChartPresenter::ZoomOutState: animation()->setAnimationType(AxisAnimation::ZoomOutAnimation); animation()->setAnimationPoint(presenter()->statePoint()); break; case ChartPresenter::ScrollUpState: case ChartPresenter::ScrollLeftState: animation()->setAnimationType(AxisAnimation::MoveBackwordAnimation); break; case ChartPresenter::ScrollDownState: case ChartPresenter::ScrollRightState: animation()->setAnimationType(AxisAnimation::MoveForwardAnimation); break; case ChartPresenter::ShowState: animation()->setAnimationType(AxisAnimation::DefaultAnimation); break; } animation()->setValues(ChartAxisElement::layout(), layout); presenter()->startAnimation(animation()); } else { setLayout(layout); updateGeometry(); } } bool CartesianChartAxis::isEmpty() { return axisGeometry().isEmpty() || gridGeometry().isEmpty() || qFuzzyCompare(min(), max()); } void CartesianChartAxis::setGeometry(const QRectF &axis, const QRectF &grid) { m_gridRect = grid; setAxisGeometry(axis); if (isEmpty()) { prepareGeometryChange(); return; } QVector layout = calculateLayout(); updateLayout(layout); } QSizeF CartesianChartAxis::sizeHint(Qt::SizeHint which, const QSizeF &constraint) const { Q_UNUSED(which); Q_UNUSED(constraint); return QSizeF(); } void CartesianChartAxis::handleArrowPenChanged(const QPen &pen) { foreach (QGraphicsItem *item, arrowItems()) static_cast(item)->setPen(pen); } void CartesianChartAxis::handleGridPenChanged(const QPen &pen) { foreach (QGraphicsItem *item, gridItems()) static_cast(item)->setPen(pen); } void CartesianChartAxis::handleMinorArrowPenChanged(const QPen &pen) { foreach (QGraphicsItem *item, minorArrowItems()) static_cast(item)->setPen(pen); } void CartesianChartAxis::handleMinorGridPenChanged(const QPen &pen) { foreach (QGraphicsItem *item, minorGridItems()) static_cast(item)->setPen(pen); } void CartesianChartAxis::handleGridLineColorChanged(const QColor &color) { foreach (QGraphicsItem *item, gridItems()) { QGraphicsLineItem *lineItem = static_cast(item); QPen pen = lineItem->pen(); pen.setColor(color); lineItem->setPen(pen); } } void CartesianChartAxis::handleMinorGridLineColorChanged(const QColor &color) { foreach (QGraphicsItem *item, minorGridItems()) { QGraphicsLineItem *lineItem = static_cast(item); QPen pen = lineItem->pen(); pen.setColor(color); lineItem->setPen(pen); } } void CartesianChartAxis::handleShadesBrushChanged(const QBrush &brush) { foreach (QGraphicsItem *item, shadeItems()) static_cast(item)->setBrush(brush); } void CartesianChartAxis::handleShadesPenChanged(const QPen &pen) { foreach (QGraphicsItem *item, shadeItems()) static_cast(item)->setPen(pen); } #include "moc_cartesianchartaxis_p.cpp" QT_CHARTS_END_NAMESPACE