mirror of
https://github.com/GoldenCheetah/GoldenCheetah.git
synced 2026-02-17 01:49:55 +00:00
If the model plot cannot be refreshed when data is invalid, the plot needs to be hidden since the qwtplot3d api does not redraw empty plots and does not have any methods for clearing the canvas. This workaround just hides the plot and shows a label when the plot is invalidated. Fixes #429.
344 lines
10 KiB
C++
344 lines
10 KiB
C++
/*
|
|
* Copyright (c) 2009 Mark Liversedge (liversedge@gmail.com)
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify it
|
|
* under the terms of the GNU General Public License as published by the Free
|
|
* Software Foundation; either version 2 of the License, or (at your option)
|
|
* any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful, but WITHOUT
|
|
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
|
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
|
|
* more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License along
|
|
* with this program; if not, write to the Free Software Foundation, Inc., 51
|
|
* Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
|
*/
|
|
|
|
#include "ModelWindow.h"
|
|
#include "ModelPlot.h"
|
|
#include "MainWindow.h"
|
|
#include "RideItem.h"
|
|
#include "IntervalItem.h"
|
|
#include "math.h"
|
|
#include "Units.h" // for MILES_PER_KM
|
|
|
|
#include <QtGui>
|
|
#include <QString>
|
|
|
|
void
|
|
ModelWindow::addStandardChannels(QComboBox *box)
|
|
{
|
|
box->addItem(tr("Power"), MODEL_POWER);
|
|
box->addItem(tr("Cadence"), MODEL_CADENCE);
|
|
box->addItem(tr("Heartrate"), MODEL_HEARTRATE);
|
|
box->addItem(tr("Speed"), MODEL_SPEED);
|
|
box->addItem(tr("Altitude"), MODEL_ALT);
|
|
box->addItem(tr("Torque"), MODEL_TORQUE);
|
|
box->addItem(tr("AEPF"), MODEL_AEPF);
|
|
box->addItem(tr("CPV"), MODEL_CPV);
|
|
box->addItem(tr("Time"), MODEL_TIME);
|
|
box->addItem(tr("Distance"), MODEL_DISTANCE);
|
|
box->addItem(tr("Latitude"), MODEL_LAT);
|
|
box->addItem(tr("Longitude"), MODEL_LONG);
|
|
}
|
|
|
|
ModelWindow::ModelWindow(MainWindow *parent, const QDir &home) :
|
|
GcWindow(parent), home(home), main(parent), ride(NULL), current(NULL)
|
|
{
|
|
setInstanceName("3D Window");
|
|
|
|
QWidget *c = new QWidget;
|
|
QFormLayout *cl = new QFormLayout(c);
|
|
setControls(c);
|
|
|
|
// hidden text when plot invalid
|
|
nodata = new QLabel(tr("No data or bin size too large."), this);
|
|
nodata->hide();
|
|
|
|
// the plot widget
|
|
QHBoxLayout *mainLayout = new QHBoxLayout;
|
|
modelPlot= new ModelPlot(main, NULL);
|
|
zpane = new QSlider(Qt::Vertical);
|
|
zpane->setTickInterval(1);
|
|
zpane->setMinimum(0);
|
|
zpane->setMaximum(100);
|
|
zpane->setValue(0);
|
|
mainLayout->addWidget(zpane);
|
|
mainLayout->addWidget(modelPlot);
|
|
mainLayout->addWidget(nodata);
|
|
setLayout(mainLayout);
|
|
|
|
// preset Values
|
|
presetLabel = new QLabel(tr("Analyse"), this);
|
|
presetValues = new QComboBox;
|
|
fillPresets(presetValues);
|
|
presetValues->setCurrentIndex(1);
|
|
cl->addRow(presetLabel, presetValues);
|
|
|
|
// labels
|
|
xLabel = new QLabel(tr("X-Axis:"), this);
|
|
xSelector = new QComboBox;
|
|
addStandardChannels(xSelector);
|
|
xSelector->setCurrentIndex(0); // power
|
|
cl->addRow(xLabel, xSelector);
|
|
|
|
yLabel = new QLabel(tr("Y-Axis:"), this);
|
|
ySelector = new QComboBox;
|
|
addStandardChannels(ySelector);
|
|
ySelector->setCurrentIndex(1); // cadence
|
|
cl->addRow(yLabel, ySelector);
|
|
|
|
zLabel = new QLabel(tr("Z-Axis:"), this);
|
|
zSelector = new QComboBox;
|
|
addStandardChannels(zSelector);
|
|
zSelector->addItem(tr("Time at X&Y"), MODEL_XYTIME);
|
|
zSelector->setCurrentIndex(12); // time at xy
|
|
cl->addRow(zLabel, zSelector);
|
|
|
|
colorLabel = new QLabel(tr("Color:"), this);
|
|
colorSelector = new QComboBox;
|
|
addStandardChannels(colorSelector);
|
|
colorSelector->addItem(tr("Power Zone"), MODEL_POWERZONE);
|
|
colorSelector->addItem(tr("Time at X&Y"), MODEL_XYTIME);
|
|
colorSelector->setCurrentIndex(12); // power zone
|
|
cl->addRow(colorLabel, colorSelector);
|
|
|
|
binLabel = new QLabel(tr("Bin Width:"), this);
|
|
binWidthLineEdit = new QLineEdit(this);
|
|
binWidthLineEdit->setFixedWidth(30);
|
|
binWidthLineEdit->setText("5");
|
|
cl->addRow(binLabel, binWidthLineEdit);
|
|
|
|
binWidthSlider = new QSlider(Qt::Horizontal);
|
|
binWidthSlider->setTickPosition(QSlider::TicksBelow);
|
|
binWidthSlider->setTickInterval(1);
|
|
binWidthSlider->setMinimum(3);
|
|
binWidthSlider->setMaximum(100);
|
|
binWidthSlider->setValue(5);
|
|
cl->addRow(binWidthSlider);
|
|
|
|
// selectors
|
|
styleSelector = new QComboBox;
|
|
styleSelector->addItem(tr("Bar"));
|
|
styleSelector->addItem(tr("Grid"));
|
|
styleSelector->addItem(tr("Surface"));
|
|
styleSelector->addItem(tr("Dots"));
|
|
styleSelector->setCurrentIndex(0);
|
|
cl->addRow(styleSelector);
|
|
|
|
ignore = new QCheckBox(tr("Ignore Zero"));
|
|
ignore->setChecked(true);
|
|
cl->addRow(ignore);
|
|
|
|
grid = new QCheckBox(tr("Show Grid"));
|
|
grid->setChecked(true);
|
|
cl->addRow(grid);
|
|
|
|
frame = new QCheckBox(tr("Frame Intervals"));
|
|
frame->setChecked(true);
|
|
cl->addRow(frame);
|
|
|
|
legend = new QCheckBox(tr("Legend"));
|
|
legend->setChecked(true);
|
|
cl->addRow(legend);
|
|
|
|
resetView = new QPushButton(tr("Reset View"));
|
|
cl->addRow(resetView);
|
|
|
|
// now connect up the widgets
|
|
//connect(main, SIGNAL(rideSelected()), this, SLOT(rideSelected()));
|
|
connect(this, SIGNAL(rideItemChanged(RideItem*)), this, SLOT(rideSelected()));
|
|
connect(main, SIGNAL(intervalSelected()), this, SLOT(intervalSelected()));
|
|
connect(presetValues, SIGNAL(currentIndexChanged(int)), this, SLOT(applyPreset(int)));
|
|
connect(xSelector, SIGNAL(currentIndexChanged(int)), this, SLOT(setDirty()));
|
|
connect(ySelector, SIGNAL(currentIndexChanged(int)), this, SLOT(setDirty()));
|
|
connect(zSelector, SIGNAL(currentIndexChanged(int)), this, SLOT(setDirty()));
|
|
connect(colorSelector, SIGNAL(currentIndexChanged(int)), this, SLOT(setDirty()));
|
|
connect(grid, SIGNAL(stateChanged(int)), this, SLOT(setGrid()));
|
|
connect(legend, SIGNAL(stateChanged(int)), this, SLOT(setLegend()));
|
|
connect(frame, SIGNAL(stateChanged(int)), this, SLOT(setFrame()));
|
|
connect(styleSelector, SIGNAL(currentIndexChanged(int)), this, SLOT(styleSelected(int)));
|
|
connect(ignore, SIGNAL(stateChanged(int)), this, SLOT(setDirty()));
|
|
connect(binWidthSlider, SIGNAL(valueChanged(int)), this, SLOT(setBinWidthFromSlider()));
|
|
connect(binWidthLineEdit, SIGNAL(editingFinished()), this, SLOT(setBinWidthFromLineEdit()));
|
|
connect(resetView, SIGNAL(clicked()), this, SLOT(resetViewPoint()));
|
|
connect(zpane, SIGNAL(valueChanged(int)), this, SLOT(setZPane(int)));
|
|
}
|
|
|
|
void
|
|
ModelWindow::rideSelected()
|
|
{
|
|
if (!amVisible())
|
|
return;
|
|
ride = myRideItem;
|
|
|
|
if (!ride || !ride->ride() || ride == current)
|
|
|
|
current = ride;
|
|
setData(true);
|
|
}
|
|
|
|
void
|
|
ModelWindow::styleSelected(int index)
|
|
{
|
|
modelPlot->setStyle(index); // 0 = bar, 1 = surface
|
|
}
|
|
|
|
void
|
|
ModelWindow::setGrid()
|
|
{
|
|
modelPlot->setGrid(grid->isChecked());
|
|
}
|
|
|
|
void
|
|
ModelWindow::setLegend()
|
|
{
|
|
modelPlot->setLegend(legend->isChecked(), settings.color);
|
|
}
|
|
|
|
void
|
|
ModelWindow::setFrame()
|
|
{
|
|
modelPlot->setFrame(frame->isChecked());
|
|
}
|
|
void
|
|
ModelWindow::setZPane(int z)
|
|
{
|
|
modelPlot->setZPane(z);
|
|
}
|
|
|
|
void
|
|
ModelWindow::intervalSelected()
|
|
{
|
|
if (!amVisible())
|
|
return;
|
|
setData(false);
|
|
}
|
|
|
|
void
|
|
ModelWindow::setData(bool adjustPlot)
|
|
{
|
|
settings.ride = ride;
|
|
settings.x = xSelector->itemData(xSelector->currentIndex()).toInt();
|
|
settings.y = ySelector->itemData(ySelector->currentIndex()).toInt();
|
|
settings.z = zSelector->itemData(zSelector->currentIndex()).toInt();
|
|
settings.color = colorSelector->itemData(colorSelector->currentIndex()).toInt();
|
|
settings.xbin = binWidthSlider->value(); // XXX fixed to single bin width
|
|
settings.ybin = binWidthSlider->value(); // XXX due to issues with bar geometry
|
|
settings.crop = false; // XXX not implemented
|
|
settings.zpane = 0;
|
|
settings.ignore = ignore->isChecked();
|
|
settings.gridlines = grid->isChecked();
|
|
settings.frame = frame->isChecked();
|
|
settings.legend = legend->isChecked();
|
|
settings.adjustPlot = adjustPlot;
|
|
zpane->setValue(0); // reset it!
|
|
|
|
// any intervals to plot?
|
|
settings.intervals.clear();
|
|
for (int i=0; i<main->allIntervalItems()->childCount(); i++) {
|
|
IntervalItem *current = dynamic_cast<IntervalItem *>(main->allIntervalItems()->child(i));
|
|
if (current != NULL && current->isSelected() == true)
|
|
settings.intervals.append(current);
|
|
}
|
|
|
|
setUpdatesEnabled(false);
|
|
|
|
// reset the model parameters
|
|
modelPlot->setData(&settings);
|
|
|
|
// if setdata resulted in the plot being hidden
|
|
// then the settings were not valid.
|
|
if (modelPlot->basicModelPlot->isHidden()) {
|
|
zpane->hide();
|
|
nodata->show();
|
|
} else {
|
|
zpane->show();
|
|
nodata->hide();
|
|
}
|
|
setClean();
|
|
|
|
setUpdatesEnabled(true);
|
|
}
|
|
|
|
void
|
|
ModelWindow::setBinWidthFromSlider()
|
|
{
|
|
binWidthLineEdit->setText(QString("%1").arg(binWidthSlider->value()));
|
|
setDirty();
|
|
}
|
|
|
|
void
|
|
ModelWindow::setBinWidthFromLineEdit()
|
|
{
|
|
binWidthSlider->setValue(binWidthLineEdit->text().toInt());
|
|
setDirty();
|
|
}
|
|
|
|
void
|
|
ModelWindow::resetViewPoint()
|
|
{
|
|
// either replot or center after fiddling
|
|
if (dirty) setData(true);
|
|
else modelPlot->resetViewPoint();
|
|
}
|
|
|
|
void
|
|
ModelWindow::setDirty()
|
|
{
|
|
dirty = true;
|
|
resetView->setText(tr("Plot"));
|
|
}
|
|
|
|
void
|
|
ModelWindow::setClean()
|
|
{
|
|
dirty = false;
|
|
resetView->setText(tr("Reset View"));
|
|
}
|
|
|
|
//
|
|
// Prepare some preset analysis
|
|
//
|
|
static struct preset {
|
|
QString name; // QComboBox value
|
|
int x, y, z, color; // values for xselector, yselector and zselector and color
|
|
bool ignore;
|
|
int bin; // value for binwidth
|
|
} presets[] = {
|
|
|
|
{ "User Defined", 0, 0, 0, 0, true, 20 },
|
|
{ "Natural Cadence Selection", 0, 1, 12, 12, false, 5 }, // don't ignore zero for cadences!
|
|
{ "Route Visualisation", 11, 10, 4, 4, false, 5 }, // don't ignore zero for cadences!
|
|
{ "Power Fatigue", 9, 0, 12, 12, true, 5 },
|
|
{ "Impact of Altitude", 4, 2, 0, 12, true, 10 },
|
|
{ "", 0, 0, 0, 0, false, 0 }
|
|
};
|
|
|
|
void
|
|
ModelWindow::applyPreset(int index)
|
|
{
|
|
if (index >=0) {
|
|
xSelector->setCurrentIndex(presets[index].x);
|
|
ySelector->setCurrentIndex(presets[index].y);
|
|
zSelector->setCurrentIndex(presets[index].z);
|
|
colorSelector->setCurrentIndex(presets[index].color);
|
|
ignore->setChecked(presets[index].ignore);
|
|
binWidthSlider->setValue(presets[index].bin);
|
|
binWidthLineEdit->setText(QString("%1").arg(presets[index].bin));
|
|
|
|
setDirty();
|
|
if (index) setData(true);
|
|
}
|
|
}
|
|
|
|
void
|
|
ModelWindow::fillPresets(QComboBox *p)
|
|
{
|
|
for (int i=0; presets[i].name != ""; i++) {
|
|
p->addItem(presets[i].name);
|
|
}
|
|
}
|