// Copyright (C) 2023 The Qt Company Ltd. // SPDX-License-Identifier: LicenseRef-Qt-Commercial OR BSD-3-Clause #include "graphmodifier.h" #include "rainfalldata.h" #include #include #include #include #include #include #include #include using namespace Qt::StringLiterals; // TODO: Many of the values do not affect custom proxy series now - should be fixed //! [set up the graph] GraphModifier::GraphModifier(Q3DBarsWidgetItem *bargraph, QObject *parent) : QObject(parent) , m_graph(bargraph) //! [set up the graph] //! [set up axes] , m_temperatureAxis(new QValue3DAxis) , m_yearAxis(new QCategory3DAxis) , m_monthAxis(new QCategory3DAxis) , m_primarySeries(new QBar3DSeries) , m_secondarySeries(new QBar3DSeries) //! [set up axes] , m_celsiusString(u"°C"_s) { //! [adjust visuals] m_graph->setShadowQuality(QtGraphs3D::ShadowQuality::SoftMedium); m_graph->setMultiSeriesUniform(true); // These are set through the active theme m_graph->activeTheme()->setPlotAreaBackgroundVisible(false); m_graph->activeTheme()->setLabelFont(QFont("Times New Roman", m_fontSize)); m_graph->activeTheme()->setLabelBackgroundVisible(true); //! [adjust visuals] m_months = {"January", "February", "March", "April", "May", "June", "July", "August", "September", "October", "November", "December"}; m_years = {"2015", "2016", "2017", "2018", "2019", "2020", "2021", "2022"}; //! [activate axes] m_temperatureAxis->setTitle("Average temperature"); m_temperatureAxis->setSegmentCount(m_segments); m_temperatureAxis->setSubSegmentCount(m_subSegments); m_temperatureAxis->setRange(m_minval, m_maxval); m_temperatureAxis->setLabelFormat(u"%.1f "_s + m_celsiusString); m_temperatureAxis->setLabelAutoAngle(30.0f); m_temperatureAxis->setTitleVisible(true); m_yearAxis->setTitle("Year"); //! [auto angle] m_yearAxis->setLabelAutoAngle(30.0f); //! [auto angle] m_yearAxis->setTitleVisible(true); m_monthAxis->setTitle("Month"); m_monthAxis->setLabelAutoAngle(30.0f); m_monthAxis->setTitleVisible(true); m_graph->setValueAxis(m_temperatureAxis); m_graph->setRowAxis(m_yearAxis); m_graph->setColumnAxis(m_monthAxis); //! [activate axes] //! [initialize series visuals] m_primarySeries->setItemLabelFormat(u"Oulu - @colLabel @rowLabel: @valueLabel"_s); m_primarySeries->setMesh(QAbstract3DSeries::Mesh::BevelBar); m_primarySeries->setMeshSmooth(false); m_secondarySeries->setItemLabelFormat(u"Helsinki - @colLabel @rowLabel: @valueLabel"_s); m_secondarySeries->setMesh(QAbstract3DSeries::Mesh::BevelBar); m_secondarySeries->setMeshSmooth(false); m_secondarySeries->setVisible(false); //! [initialize series visuals] //! [add series] m_graph->addSeries(m_primarySeries); m_graph->addSeries(m_secondarySeries); //! [add series] //! [invoke camera preset] changePresetCamera(); //! [invoke camera preset] //! [invoke data reset] resetTemperatureData(); //! [invoke data reset] // Set up property animations for zooming to the selected bar //! [animation setup] m_defaultAngleX = m_graph->cameraXRotation(); m_defaultAngleY = m_graph->cameraYRotation(); m_defaultZoom = m_graph->cameraZoomLevel(); m_defaultTarget = m_graph->cameraTargetPosition(); m_animationCameraX.setTargetObject(m_graph); m_animationCameraY.setTargetObject(m_graph); m_animationCameraZoom.setTargetObject(m_graph); m_animationCameraTarget.setTargetObject(m_graph); m_animationCameraX.setPropertyName("cameraXRotation"); m_animationCameraY.setPropertyName("cameraYRotation"); m_animationCameraZoom.setPropertyName("cameraZoomLevel"); m_animationCameraTarget.setPropertyName("cameraTargetPosition"); int duration = 1700; m_animationCameraX.setDuration(duration); m_animationCameraY.setDuration(duration); m_animationCameraZoom.setDuration(duration); m_animationCameraTarget.setDuration(duration); // The zoom always first zooms out above the graph and then zooms in qreal zoomOutFraction = 0.3; m_animationCameraX.setKeyValueAt(zoomOutFraction, QVariant::fromValue(0.0f)); m_animationCameraY.setKeyValueAt(zoomOutFraction, QVariant::fromValue(90.0f)); m_animationCameraZoom.setKeyValueAt(zoomOutFraction, QVariant::fromValue(50.0f)); m_animationCameraTarget.setKeyValueAt(zoomOutFraction, QVariant::fromValue(QVector3D(0.0f, 0.0f, 0.0f))); //! [animation setup] m_customData = new RainfallData(); } GraphModifier::~GraphModifier() { delete m_customData; } void GraphModifier::resetTemperatureData() { //! [set up data] // Set up data static const float tempOulu[8][12] = { {-7.4f, -2.4f, 0.0f, 3.0f, 8.2f, 11.6f, 14.7f, 15.4f, 11.4f, 4.2f, 2.1f, -2.3f}, // 2015 {-13.4f, -3.9f, -1.8f, 3.1f, 10.6f, 13.7f, 17.8f, 13.6f, 10.7f, 3.5f, -3.1f, -4.2f}, // 2016 //! [set up data] {-5.7f, -6.7f, -3.0f, -0.1f, 4.7f, 12.4f, 16.1f, 14.1f, 9.4f, 3.0f, -0.3f, -3.2f}, // 2017 {-6.4f, -11.9f, -7.4f, 1.9f, 11.4f, 12.4f, 21.5f, 16.1f, 11.0f, 4.4f, 2.1f, -4.1f}, // 2018 {-11.7f, -6.1f, -2.4f, 3.9f, 7.2f, 14.5f, 15.6f, 14.4f, 8.5f, 2.0f, -3.0f, -1.5f}, // 2019 {-2.1f, -3.4f, -1.8f, 0.6f, 7.0f, 17.1f, 15.6f, 15.4f, 11.1f, 5.6f, 1.9f, -1.7f}, // 2020 {-9.6f, -11.6f, -3.2f, 2.4f, 7.8f, 17.3f, 19.4f, 14.2f, 8.0f, 5.2f, -2.2f, -8.6f}, // 2021 {-7.3f, -6.4f, -1.8f, 1.3f, 8.1f, 15.5f, 17.6f, 17.6f, 9.1f, 5.4f, -1.5f, -4.4f} // 2022 }; static const float tempHelsinki[8][12] = { {-2.0f, -0.1f, 1.8f, 5.1f, 9.7f, 13.7f, 16.3f, 17.3f, 12.7f, 5.4f, 4.6f, 2.1f}, // 2015 {-10.3f, -0.6f, 0.0f, 4.9f, 14.3f, 15.7f, 17.7f, 16.0f, 12.7f, 4.6f, -1.0f, -0.9f}, // 2016 {-2.9f, -3.3f, 0.7f, 2.3f, 9.9f, 13.8f, 16.1f, 15.9f, 11.4f, 5.0f, 2.7f, 0.7f}, // 2017 {-2.2f, -8.4f, -4.7f, 5.0f, 15.3f, 15.8f, 21.2f, 18.2f, 13.3f, 6.7f, 2.8f, -2.0f}, // 2018 {-6.2f, -0.5f, -0.3f, 6.8f, 10.6f, 17.9f, 17.5f, 16.8f, 11.3f, 5.2f, 1.8f, 1.4f}, // 2019 {1.9f, 0.5f, 1.7f, 4.5f, 9.5f, 18.4f, 16.5f, 16.8f, 13.0f, 8.2f, 4.4f, 0.9f}, // 2020 {-4.7f, -8.1f, -0.9f, 4.5f, 10.4f, 19.2f, 20.9f, 15.4f, 9.5f, 8.0f, 1.5f, -6.7f}, // 2021 {-3.3f, -2.2f, -0.2f, 3.3f, 9.6f, 16.9f, 18.1f, 18.9f, 9.2f, 7.6f, 2.3f, -3.4f} // 2022 }; // Create data arrays //! [populate series vie proxies] QBarDataArray dataSet; QBarDataArray dataSet2; dataSet.reserve(m_years.size()); for (qsizetype year = 0; year < m_years.size(); ++year) { // Create a data row QBarDataRow dataRow(m_months.size()); QBarDataRow dataRow2(m_months.size()); for (qsizetype month = 0; month < m_months.size(); ++month) { // Add data to the row dataRow[month].setValue(tempOulu[year][month]); dataRow2[month].setValue(tempHelsinki[year][month]); } // Add the row to the set dataSet.append(dataRow); dataSet2.append(dataRow2); } // Add data to the data proxy (the data proxy assumes ownership of it) m_primarySeries->dataProxy()->resetArray(dataSet, m_years, m_months); m_secondarySeries->dataProxy()->resetArray(dataSet2, m_years, m_months); //! [populate series vie proxies] } void GraphModifier::changeRange(int range) { if (range >= m_years.count()) m_yearAxis->setRange(0, m_years.count() - 1); else m_yearAxis->setRange(range, range); } void GraphModifier::changeStyle(int style) { QComboBox *comboBox = qobject_cast(sender()); if (comboBox) { m_barMesh = comboBox->itemData(style).value(); m_primarySeries->setMesh(m_barMesh); m_secondarySeries->setMesh(m_barMesh); m_customData->customSeries()->setMesh(m_barMesh); if (m_barMesh == QAbstract3DSeries::Mesh::UserDefined) { m_primarySeries->setUserDefinedMesh(":/data/narrowarrow.mesh"); m_secondarySeries->setUserDefinedMesh(":/data/narrowarrow.mesh"); m_customData->customSeries()->setUserDefinedMesh(":/data/narrowarrow.mesh"); } } } void GraphModifier::changePresetCamera() { m_animationCameraX.stop(); m_animationCameraY.stop(); m_animationCameraZoom.stop(); m_animationCameraTarget.stop(); // Restore camera target in case animation has changed it m_graph->setCameraTargetPosition(QVector3D(0.0f, 0.0f, 0.0f)); //! [change the preset] static int preset = int(QtGraphs3D::CameraPreset::Front); m_graph->setCameraPreset((QtGraphs3D::CameraPreset) preset); if (++preset > int(QtGraphs3D::CameraPreset::DirectlyBelow)) preset = int(QtGraphs3D::CameraPreset::FrontLow); //! [change the preset] } void GraphModifier::changeTheme(int theme) { QGraphsTheme *currentTheme = m_graph->activeTheme(); currentTheme->setTheme(QGraphsTheme::Theme(theme)); emit backgroundVisibleChanged(currentTheme->isPlotAreaBackgroundVisible()); emit gridVisibleChanged(currentTheme->isGridVisible()); emit fontChanged(currentTheme->labelFont()); emit fontSizeChanged(currentTheme->labelFont().pointSize()); } void GraphModifier::changeLabelBackground() { m_graph->activeTheme()->setLabelBackgroundVisible( !m_graph->activeTheme()->isLabelBackgroundVisible()); } void GraphModifier::changeSelectionMode(int selectionMode) { const auto *comboBox = qobject_cast(sender()); if (comboBox) { int flags = comboBox->itemData(selectionMode).toInt(); m_graph->setSelectionMode(QtGraphs3D::SelectionFlags(flags)); } } void GraphModifier::changeFont(const QFont &font) { m_graph->activeTheme()->setLabelFont(font); } void GraphModifier::changeFontSize(int fontsize) { m_fontSize = fontsize; QFont font = m_graph->activeTheme()->labelFont(); font.setPointSize(m_fontSize); m_graph->activeTheme()->setLabelFont(font); } void GraphModifier::shadowQualityUpdatedByVisual(QtGraphs3D::ShadowQuality sq) { int quality = int(sq); // Updates the UI component to show correct shadow quality emit shadowQualityChanged(quality); } void GraphModifier::changeLabelRotation(int rotation) { m_temperatureAxis->setLabelAutoAngle(float(rotation)); m_monthAxis->setLabelAutoAngle(float(rotation)); m_yearAxis->setLabelAutoAngle(float(rotation)); } void GraphModifier::setAxisTitleVisibility(bool visible) { m_temperatureAxis->setTitleVisible(visible); m_monthAxis->setTitleVisible(visible); m_yearAxis->setTitleVisible(visible); } void GraphModifier::setAxisTitleFixed(bool enabled) { m_temperatureAxis->setTitleFixed(enabled); m_monthAxis->setTitleFixed(enabled); m_yearAxis->setTitleFixed(enabled); } void GraphModifier::zoomToSelectedBar() { m_animationCameraX.stop(); m_animationCameraY.stop(); m_animationCameraZoom.stop(); m_animationCameraTarget.stop(); float currentX = m_graph->cameraXRotation(); float currentY = m_graph->cameraYRotation(); float currentZoom = m_graph->cameraZoomLevel(); QVector3D currentTarget = m_graph->cameraTargetPosition(); m_animationCameraX.setStartValue(QVariant::fromValue(currentX)); m_animationCameraY.setStartValue(QVariant::fromValue(currentY)); m_animationCameraZoom.setStartValue(QVariant::fromValue(currentZoom)); m_animationCameraTarget.setStartValue(QVariant::fromValue(currentTarget)); QPoint selectedBar = m_graph->selectedSeries() ? m_graph->selectedSeries()->selectedBar() : QBar3DSeries::invalidSelectionPosition(); if (selectedBar != QBar3DSeries::invalidSelectionPosition()) { // Normalize selected bar position within axis range to determine target coordinates //! [calculate target] QVector3D endTarget; float xMin = m_graph->columnAxis()->min(); float xRange = m_graph->columnAxis()->max() - xMin; float zMin = m_graph->rowAxis()->min(); float zRange = m_graph->rowAxis()->max() - zMin; endTarget.setX((selectedBar.y() - xMin) / xRange * 2.0f - 1.0f); endTarget.setZ((selectedBar.x() - zMin) / zRange * 2.0f - 1.0f); //! [calculate target] // Rotate the camera so that it always points approximately to the graph center //! [center camera] qreal endAngleX = 90.0 - qRadiansToDegrees(qAtan(qreal(endTarget.z() / endTarget.x()))); if (endTarget.x() > 0.0f) endAngleX -= 180.0f; float barValue = m_graph->selectedSeries() ->dataProxy() ->itemAt(selectedBar.x(), selectedBar.y()) .value(); float endAngleY = barValue >= 0.0f ? 30.0f : -30.0f; if (m_graph->valueAxis()->reversed()) endAngleY *= -1.0f; //! [center camera] m_animationCameraX.setEndValue(QVariant::fromValue(float(endAngleX))); m_animationCameraY.setEndValue(QVariant::fromValue(endAngleY)); m_animationCameraZoom.setEndValue(QVariant::fromValue(100)); //! [set target] m_animationCameraTarget.setEndValue(QVariant::fromValue(endTarget)); //! [set target] } else { // No selected bar, so return to the default view m_animationCameraX.setEndValue(QVariant::fromValue(m_defaultAngleX)); m_animationCameraY.setEndValue(QVariant::fromValue(m_defaultAngleY)); m_animationCameraZoom.setEndValue(QVariant::fromValue(m_defaultZoom)); m_animationCameraTarget.setEndValue(QVariant::fromValue(m_defaultTarget)); } m_animationCameraX.start(); m_animationCameraY.start(); m_animationCameraZoom.start(); m_animationCameraTarget.start(); } void GraphModifier::setDataModeToWeather(bool enabled) { if (enabled) changeDataMode(false); } void GraphModifier::setDataModeToCustom(bool enabled) { if (enabled) changeDataMode(true); } void GraphModifier::changeShadowQuality(int quality) { const auto sq = QtGraphs3D::ShadowQuality(quality); m_graph->setShadowQuality(sq); emit shadowQualityChanged(quality); } //! [rotate camera] void GraphModifier::rotateX(int angle) { m_xRotation = angle; m_graph->setCameraPosition(m_xRotation, m_yRotation); } //! [rotate camera] void GraphModifier::rotateY(int angle) { m_yRotation = angle; m_graph->setCameraPosition(m_xRotation, m_yRotation); } void GraphModifier::setBackgroundVisible(int visible) { m_graph->activeTheme()->setPlotAreaBackgroundVisible(bool(visible)); } void GraphModifier::setGridVisible(int visible) { m_graph->activeTheme()->setGridVisible(bool(visible)); } void GraphModifier::setSmoothBars(int smooth) { m_smooth = bool(smooth); m_primarySeries->setMeshSmooth(m_smooth); m_secondarySeries->setMeshSmooth(m_smooth); m_customData->customSeries()->setMeshSmooth(m_smooth); } void GraphModifier::setSeriesVisibility(int visible) { m_secondarySeries->setVisible(bool(visible)); } void GraphModifier::setReverseValueAxis(int enabled) { m_graph->valueAxis()->setReversed(enabled); } void GraphModifier::changeDataMode(bool customData) { int enabled = false; // Change between weather data and data from custom proxy if (customData) { m_graph->removeSeries(m_primarySeries); m_graph->removeSeries(m_secondarySeries); m_graph->addSeries(m_customData->customSeries()); if (m_graph->valueAxis()->reversed()) enabled = true; m_graph->setValueAxis(m_customData->valueAxis()); m_graph->setRowAxis(m_customData->rowAxis()); m_graph->setColumnAxis(m_customData->colAxis()); } else { m_graph->removeSeries(m_customData->customSeries()); m_graph->addSeries(m_primarySeries); m_graph->addSeries(m_secondarySeries); if (m_graph->valueAxis()->reversed()) enabled = true; m_graph->setValueAxis(m_temperatureAxis); m_graph->setRowAxis(m_yearAxis); m_graph->setColumnAxis(m_monthAxis); } setReverseValueAxis(enabled); }