mirror of
https://github.com/GoldenCheetah/GoldenCheetah.git
synced 2026-02-14 08:38:45 +00:00
759 lines
26 KiB
C++
759 lines
26 KiB
C++
/*
|
|
* Copyright (c) 2013 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 "ComparePane.h"
|
|
#include "Settings.h"
|
|
#include "RideFile.h"
|
|
#include "RideFileCache.h"
|
|
#include "RideMetric.h"
|
|
#include "SummaryMetrics.h"
|
|
#include "MetricAggregator.h"
|
|
#include "ColorButton.h"
|
|
#include "TimeUtils.h"
|
|
#include "Units.h"
|
|
#include "Zones.h"
|
|
|
|
#include <QCheckBox>
|
|
|
|
//
|
|
// A selection of distinct colours, user can adjust also
|
|
//
|
|
static QList<QColor> standardColors;
|
|
static bool initStandardColors()
|
|
{
|
|
standardColors << QColor(Qt::magenta);
|
|
standardColors << QColor(Qt::cyan);
|
|
standardColors << QColor(Qt::yellow);
|
|
standardColors << QColor(Qt::red);
|
|
standardColors << QColor(Qt::blue);
|
|
standardColors << QColor(Qt::gray);
|
|
standardColors << QColor(Qt::darkCyan);
|
|
standardColors << QColor(Qt::green);
|
|
standardColors << QColor(Qt::darkRed);
|
|
standardColors << QColor(Qt::darkGreen);
|
|
standardColors << QColor(Qt::darkBlue);
|
|
standardColors << QColor(Qt::darkMagenta);
|
|
|
|
return true;
|
|
}
|
|
static bool init = initStandardColors();
|
|
|
|
// we need to fix the sort order!
|
|
class CTableWidgetItem : public QTableWidgetItem
|
|
{
|
|
public:
|
|
CTableWidgetItem(int type = Type) : QTableWidgetItem(type) {}
|
|
~CTableWidgetItem() {}
|
|
|
|
bool operator<(const QTableWidgetItem&other) const // for sorting our way
|
|
{
|
|
switch(column()) {
|
|
|
|
case 2 : return text() < other.text(); // athlete
|
|
case 3 : return QDate::fromString(text(), "dd, MMM yyyy") <
|
|
QDate::fromString(other.text(), "dd, MMM yyyy"); // date
|
|
case 4 : // date or time depending on which view
|
|
if (text().contains(":")) {
|
|
|
|
return QTime::fromString(text(), "hh:mm:ss") <
|
|
QTime::fromString(other.text(), "hh:mm:ss");
|
|
|
|
} else {
|
|
|
|
return QDate::fromString(text(), "dd, MMM yyyy") <
|
|
QDate::fromString(other.text(), "dd, MMM yyyy"); // date
|
|
|
|
}
|
|
case 5 : return QTime::fromString(text(), "hh:mm") < // Duration
|
|
QTime::fromString(other.text(), "hh:mm");
|
|
|
|
default: // work it out ..
|
|
if (text().contains(":")) { // time
|
|
|
|
return QTime::fromString(text()) <
|
|
QTime::fromString(other.text());
|
|
|
|
} else if (text().contains(QRegExp("[^0-9.,]")) ||
|
|
other.text().contains(QRegExp("[^0-9.,]"))) { // alpha
|
|
|
|
return text() < other.text();
|
|
|
|
} else { // assume numeric
|
|
|
|
return text().toDouble() < other.text().toDouble();
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
};
|
|
|
|
ComparePane::ComparePane(Context *context, QWidget *parent, CompareMode mode) : QWidget(parent), context(context), mode_(mode)
|
|
{
|
|
QVBoxLayout *layout = new QVBoxLayout(this);
|
|
layout->setContentsMargins(0,0,0,0);
|
|
layout->setSpacing(0);
|
|
|
|
setAcceptDrops(true);
|
|
setAutoFillBackground(true);
|
|
QPalette pal;
|
|
pal.setBrush(QPalette::Active, QPalette::Window, Qt::white);
|
|
pal.setBrush(QPalette::Inactive, QPalette::Window, Qt::white);
|
|
setPalette(pal);
|
|
|
|
scrollArea = new QScrollArea(this);
|
|
scrollArea->setAutoFillBackground(false);
|
|
scrollArea->setWidgetResizable(true);
|
|
scrollArea->setFrameStyle(QFrame::NoFrame);
|
|
scrollArea->setContentsMargins(0,0,0,0);
|
|
scrollArea->setHorizontalScrollBarPolicy(Qt::ScrollBarAlwaysOff);
|
|
scrollArea->setVerticalScrollBarPolicy(Qt::ScrollBarAlwaysOff);
|
|
layout->addWidget(scrollArea);
|
|
|
|
table = new QTableWidget(this);
|
|
#ifdef Q_OS_MAC
|
|
table->setAttribute(Qt::WA_MacShowFocusRect, 0);
|
|
table->horizontalHeader()->setSortIndicatorShown(false); // blue looks nasty
|
|
#endif
|
|
table->setVerticalScrollBarPolicy(Qt::ScrollBarAlwaysOff);
|
|
table->setHorizontalScrollBarPolicy(Qt::ScrollBarAlwaysOff);
|
|
table->setAcceptDrops(false);
|
|
scrollArea->setWidget(table);
|
|
|
|
configChanged(); // set up ready to go...
|
|
|
|
connect(context, SIGNAL(configChanged()), this, SLOT(configChanged()));
|
|
connect(table->horizontalHeader(), SIGNAL(sectionClicked(int)), this, SLOT(itemsWereSorted()));
|
|
}
|
|
|
|
void
|
|
ComparePane::configChanged()
|
|
{
|
|
// refresh table...
|
|
refreshTable();
|
|
}
|
|
|
|
void
|
|
ComparePane::refreshTable()
|
|
{
|
|
if (mode_ == interval) { // INTERVALS
|
|
|
|
// STEP ONE : SET THE TABLE HEADINGS
|
|
|
|
// clear current contents
|
|
table->clear();
|
|
table->setRowCount(0);
|
|
|
|
// metric summary
|
|
QStringList always;
|
|
always << "workout_time" << "total_distance";
|
|
QString s = appsettings->value(this, GC_SETTINGS_INTERVAL_METRICS, GC_SETTINGS_INTERVAL_METRICS_DEFAULT).toString();
|
|
if (s == "") s = GC_SETTINGS_INTERVAL_METRICS_DEFAULT;
|
|
QStringList metricColumns = always + s.split(","); // always showm metrics plus user defined summary metrics
|
|
metricColumns.removeDuplicates(); // where user has already added workout_time, total_distance
|
|
|
|
// called after config is updated typically
|
|
QStringList list;
|
|
list << "" // checkbox
|
|
<< "" // color
|
|
<< "Athlete"
|
|
<< "Date"
|
|
<< "Time";
|
|
|
|
QStringList worklist; // metrics to compute
|
|
RideMetricFactory &factory = RideMetricFactory::instance();
|
|
|
|
foreach(QString metric, metricColumns) {
|
|
|
|
// get the metric name
|
|
const RideMetric *m = factory.rideMetric(metric);
|
|
if (m) {
|
|
worklist << metric;
|
|
QString units;
|
|
if (m->units(context->athlete->useMetricUnits) != "seconds") units = m->units(context->athlete->useMetricUnits);
|
|
if (units != "") list << QString("%1 (%2)").arg(m->name()).arg(units);
|
|
else list << QString("%1").arg(m->name());
|
|
}
|
|
}
|
|
|
|
list << "Interval";
|
|
|
|
table->setColumnCount(list.count()+1);
|
|
table->horizontalHeader()->setSectionHidden(list.count(), true);
|
|
table->setHorizontalHeaderLabels(list);
|
|
table->setSortingEnabled(true);
|
|
table->verticalHeader()->hide();
|
|
table->setShowGrid(false);
|
|
table->setSelectionMode(QAbstractItemView::MultiSelection);
|
|
table->setSelectionBehavior(QAbstractItemView::SelectRows);
|
|
|
|
// STEP TWO : CLEAR AND RE-ADD TO REFLECT CHANGES
|
|
|
|
table->setRowCount(context->compareIntervals.count());
|
|
int counter = 0;
|
|
foreach(CompareInterval x, context->compareIntervals) {
|
|
|
|
// compute the metrics for this ride
|
|
SummaryMetrics metrics;
|
|
QHash<QString, RideMetricPtr> computed = RideMetric::computeMetrics(context, x.data,
|
|
context->athlete->zones(), context->athlete->hrZones(), worklist);
|
|
|
|
for(int i = 0; i < worklist.count(); i++) {
|
|
if (worklist[i] != "") {
|
|
RideMetricPtr m = computed.value(worklist[i]);
|
|
if (m) metrics.setForSymbol(worklist[i], m->value(true));
|
|
else metrics.setForSymbol(worklist[i], 0.00);
|
|
}
|
|
}
|
|
|
|
// First few cols always the same
|
|
// check - color - athlete - date - time
|
|
// now create a row on the compare pane
|
|
|
|
// Checkbox
|
|
QCheckBox *check = new QCheckBox(this);
|
|
check->setChecked(x.checked);
|
|
if (!counter) check->setEnabled(false);
|
|
table->setCellWidget(counter, 0, check);
|
|
connect(check, SIGNAL(stateChanged(int)), this, SLOT(intervalButtonsChanged()));
|
|
|
|
// Color Button
|
|
ColorButton *colorButton = new ColorButton(this, "Color", x.color);
|
|
table->setCellWidget(counter, 1, colorButton);
|
|
connect(colorButton, SIGNAL(colorChosen(QColor)), this, SLOT(intervalButtonsChanged()));
|
|
|
|
// athlete
|
|
CTableWidgetItem *t = new CTableWidgetItem;
|
|
t->setText(x.sourceContext->athlete->cyclist);
|
|
t->setFlags(t->flags() & (~Qt::ItemIsEditable));
|
|
table->setItem(counter, 2, t);
|
|
|
|
// date
|
|
t = new CTableWidgetItem;
|
|
t->setText(x.data->startTime().date().toString("dd MMM, yyyy"));
|
|
t->setFlags(t->flags() & (~Qt::ItemIsEditable));
|
|
table->setItem(counter, 3, t);
|
|
|
|
// time
|
|
t = new CTableWidgetItem;
|
|
t->setText(x.data->startTime().time().toString("hh:mm:ss"));
|
|
t->setFlags(t->flags() & (~Qt::ItemIsEditable));
|
|
table->setItem(counter, 4, t);
|
|
|
|
// metrics
|
|
for(int i = 0; i < worklist.count(); i++) {
|
|
|
|
RideMetricPtr m = computed.value(worklist[i]);
|
|
|
|
QString strValue;
|
|
|
|
if (m) {
|
|
// get value and convert if needed
|
|
double value = metrics.getForSymbol(worklist[i])
|
|
* (context->athlete->useMetricUnits ? 1 : m->conversion())
|
|
+ (context->athlete->useMetricUnits ? 0 : m->conversionSum());
|
|
|
|
// use right precision
|
|
strValue = QString("%1").arg(value, 0, 'f', m->precision());
|
|
|
|
// or maybe its a duration (worry about local lang or translated)
|
|
if (m->units(true) == "seconds" || m->units(true) == tr("seconds"))
|
|
strValue = time_to_string(value);
|
|
|
|
}
|
|
|
|
// add to the table
|
|
t = new CTableWidgetItem;
|
|
t->setText(strValue);
|
|
t->setFlags(t->flags() & (~Qt::ItemIsEditable));
|
|
table->setItem(counter, i + 5, t);
|
|
}
|
|
|
|
// Interval name
|
|
t = new CTableWidgetItem;
|
|
t->setText(x.name);
|
|
t->setFlags(t->flags() & (~Qt::ItemIsEditable));
|
|
table->setItem(counter, worklist.count() + 5, t);
|
|
|
|
// INDEX
|
|
t = new CTableWidgetItem;
|
|
t->setText(QString("%1").arg(counter));
|
|
t->setFlags(t->flags() & (~Qt::ItemIsEditable));
|
|
table->setItem(counter, worklist.count() + 6, t);
|
|
|
|
// align center
|
|
for (int i=3; i<(worklist.count()+5); i++)
|
|
table->item(counter,i)->setTextAlignment(Qt::AlignVCenter | Qt::AlignHCenter);
|
|
|
|
table->setRowHeight(counter, 23);
|
|
counter++;
|
|
}
|
|
|
|
table->resizeColumnsToContents(); // set columns to fit
|
|
#if QT_VERSION > 0x050000 // fix the first two if we can
|
|
for (int i=0; i<list.count(); i++) {
|
|
if (i < 2) {
|
|
table->horizontalHeader()->setSectionResizeMode(i, QHeaderView::Fixed);
|
|
} else {
|
|
table->horizontalHeader()->setSectionResizeMode(i, QHeaderView::Interactive);
|
|
}
|
|
}
|
|
#else
|
|
table->horizontalHeader()->setResizeMode(QHeaderView::Interactive);
|
|
#endif
|
|
table->horizontalHeader()->setStretchLastSection(true);
|
|
|
|
} else { //SEASONS
|
|
|
|
// STEP ONE : SET THE TABLE HEADINGS
|
|
|
|
// clear current contents
|
|
table->clear();
|
|
table->setRowCount(0);
|
|
|
|
// metric summary
|
|
QStringList always;
|
|
always << "workout_time" << "total_distance";
|
|
QString s = appsettings->value(this, GC_SETTINGS_SUMMARY_METRICS, GC_SETTINGS_SUMMARY_METRICS_DEFAULT).toString();
|
|
if (s == "") s = GC_SETTINGS_SUMMARY_METRICS_DEFAULT;
|
|
QStringList metricColumns = always + s.split(","); // always showm metrics plus user defined summary metrics
|
|
metricColumns.removeDuplicates(); // where user has already added workout_time, total_distance
|
|
|
|
// called after config is updated typically
|
|
QStringList list;
|
|
list << "" // checkbox
|
|
<< "" // color
|
|
<< "Athlete"
|
|
<< "From"
|
|
<< "To";
|
|
|
|
QStringList worklist; // metrics to compute
|
|
RideMetricFactory &factory = RideMetricFactory::instance();
|
|
|
|
foreach(QString metric, metricColumns) {
|
|
|
|
// get the metric name
|
|
const RideMetric *m = factory.rideMetric(metric);
|
|
if (m) {
|
|
worklist << metric;
|
|
QString units;
|
|
if (m->units(context->athlete->useMetricUnits) != "seconds") units = m->units(context->athlete->useMetricUnits);
|
|
if (units != "") list << QString("%1 (%2)").arg(m->name()).arg(units);
|
|
else list << QString("%1").arg(m->name());
|
|
}
|
|
}
|
|
|
|
list << "Date Range";
|
|
|
|
table->setColumnCount(list.count()+1);
|
|
table->horizontalHeader()->setSectionHidden(list.count(), true);
|
|
table->setHorizontalHeaderLabels(list);
|
|
table->setSortingEnabled(true);
|
|
table->verticalHeader()->hide();
|
|
table->setShowGrid(false);
|
|
table->setSelectionMode(QAbstractItemView::MultiSelection);
|
|
table->setSelectionBehavior(QAbstractItemView::SelectRows);
|
|
|
|
// STEP TWO : CLEAR AND RE-ADD TO REFLECT CHANGES
|
|
|
|
table->setRowCount(context->compareDateRanges.count());
|
|
int counter = 0;
|
|
foreach(CompareDateRange x, context->compareDateRanges) {
|
|
|
|
// First few cols always the same
|
|
// check - color - athlete - date - time
|
|
// now create a row on the compare pane
|
|
|
|
// Checkbox
|
|
QCheckBox *check = new QCheckBox(this);
|
|
check->setChecked(x.checked);
|
|
if (!counter) check->setEnabled(false);
|
|
table->setCellWidget(counter, 0, check);
|
|
connect(check, SIGNAL(stateChanged(int)), this, SLOT(daterangeButtonsChanged()));
|
|
|
|
// Color Button
|
|
ColorButton *colorButton = new ColorButton(this, "Color", x.color);
|
|
table->setCellWidget(counter, 1, colorButton);
|
|
connect(colorButton, SIGNAL(colorChosen(QColor)), this, SLOT(daterangeButtonsChanged()));
|
|
|
|
// athlete
|
|
CTableWidgetItem *t = new CTableWidgetItem;
|
|
t->setText(x.sourceContext->athlete->cyclist);
|
|
t->setFlags(t->flags() & (~Qt::ItemIsEditable));
|
|
table->setItem(counter, 2, t);
|
|
|
|
// date from
|
|
t = new CTableWidgetItem;
|
|
t->setText(x.start.toString("dd MMM, yyyy"));
|
|
t->setFlags(t->flags() & (~Qt::ItemIsEditable));
|
|
table->setItem(counter, 3, t);
|
|
|
|
// date to
|
|
t = new CTableWidgetItem;
|
|
t->setText(x.end.toString("dd MMM, yyyy"));
|
|
t->setFlags(t->flags() & (~Qt::ItemIsEditable));
|
|
table->setItem(counter, 4, t);
|
|
|
|
// metrics
|
|
for(int i = 0; i < worklist.count(); i++) {
|
|
|
|
QString value = SummaryMetrics::getAggregated(x.sourceContext, worklist[i],
|
|
x.metrics, QStringList(), false, context->athlete->useMetricUnits);
|
|
|
|
// add to the table
|
|
t = new CTableWidgetItem;
|
|
t->setText(value);
|
|
t->setFlags(t->flags() & (~Qt::ItemIsEditable));
|
|
table->setItem(counter, i + 5, t);
|
|
}
|
|
|
|
// Date Range name
|
|
t = new CTableWidgetItem;
|
|
t->setText(x.name);
|
|
t->setFlags(t->flags() & (~Qt::ItemIsEditable));
|
|
table->setItem(counter, worklist.count() + 5, t);
|
|
|
|
// INDEX
|
|
t = new CTableWidgetItem;
|
|
t->setText(QString("%1").arg(counter));
|
|
t->setFlags(t->flags() & (~Qt::ItemIsEditable));
|
|
table->setItem(counter, worklist.count() + 6, t);
|
|
|
|
|
|
// align center
|
|
for (int i=3; i<(worklist.count()+5); i++)
|
|
table->item(counter,i)->setTextAlignment(Qt::AlignVCenter | Qt::AlignHCenter);
|
|
|
|
table->setRowHeight(counter, 23);
|
|
counter++;
|
|
}
|
|
|
|
table->resizeColumnsToContents(); // set columns to fit
|
|
#if QT_VERSION > 0x050000 // fix the first two if we can
|
|
for (int i=0; i<list.count(); i++) {
|
|
if (i < 2) {
|
|
table->horizontalHeader()->setSectionResizeMode(i, QHeaderView::Fixed);
|
|
} else {
|
|
table->horizontalHeader()->setSectionResizeMode(i, QHeaderView::Interactive);
|
|
}
|
|
}
|
|
#else
|
|
table->horizontalHeader()->setResizeMode(QHeaderView::Interactive);
|
|
#endif
|
|
table->horizontalHeader()->setStretchLastSection(true);
|
|
}
|
|
}
|
|
|
|
void
|
|
ComparePane::itemsWereSorted()
|
|
{
|
|
if (mode_ == interval) {
|
|
|
|
QList<CompareInterval> newOrder;
|
|
|
|
for(int i=0;i<table->rowCount(); i++) {
|
|
QCheckBox *check = static_cast<QCheckBox*>(table->cellWidget(i,0));
|
|
if (i) check->setEnabled(true);
|
|
else {
|
|
check->setChecked(true);
|
|
check->setEnabled(false);
|
|
}
|
|
int oldIndex = table->item(i,table->columnCount()-1)->text().toInt();
|
|
table->item(i,table->columnCount()-1)->setText(QString("%1").arg(i));
|
|
newOrder << context->compareIntervals.at(oldIndex);
|
|
}
|
|
|
|
context->compareIntervals = newOrder;
|
|
context->notifyCompareIntervalsChanged();
|
|
|
|
}
|
|
else {
|
|
|
|
QList<CompareDateRange> newOrder;
|
|
|
|
for(int i=0;i<table->rowCount(); i++) {
|
|
QCheckBox *check = static_cast<QCheckBox*>(table->cellWidget(i,0));
|
|
if (i) check->setEnabled(true);
|
|
else {
|
|
check->setChecked(true);
|
|
check->setEnabled(false);
|
|
}
|
|
int oldIndex = table->item(i,table->columnCount()-1)->text().toInt();
|
|
table->item(i,table->columnCount()-1)->setText(QString("%1").arg(i));
|
|
newOrder << context->compareDateRanges.at(oldIndex);
|
|
}
|
|
|
|
context->compareDateRanges = newOrder;
|
|
context->notifyCompareDateRangesChanged();
|
|
|
|
}
|
|
}
|
|
|
|
void
|
|
ComparePane::clear()
|
|
{
|
|
if (mode_ == interval) { // INTERVALS
|
|
|
|
// wipe all away
|
|
foreach(CompareInterval ci, context->compareIntervals) {
|
|
delete ci.data;
|
|
}
|
|
context->compareIntervals.clear();
|
|
|
|
// refresh table
|
|
refreshTable();
|
|
|
|
// tell the charts
|
|
context->notifyCompareIntervalsChanged();
|
|
|
|
} else {
|
|
|
|
// wipe em
|
|
context->compareDateRanges.clear();
|
|
|
|
// refresh table
|
|
refreshTable();
|
|
|
|
// tell the charts
|
|
context->notifyCompareDateRangesChanged();
|
|
|
|
}
|
|
}
|
|
|
|
void
|
|
ComparePane::intervalButtonsChanged()
|
|
{
|
|
// run through the table and see if anything changed
|
|
bool changed = false;
|
|
for (int i=0; i<table->rowCount(); i++) {
|
|
|
|
bool isChecked = static_cast<QCheckBox*>(table->cellWidget(i,0))->isChecked();
|
|
QColor color = static_cast<ColorButton*>(table->cellWidget(i,1))->getColor();
|
|
|
|
if (context->compareIntervals[i].checked != isChecked ||
|
|
context->compareIntervals[i].color != color) {
|
|
|
|
context->compareIntervals[i].checked = isChecked;
|
|
context->compareIntervals[i].color = color;
|
|
changed = true;
|
|
}
|
|
}
|
|
if (changed) context->notifyCompareIntervalsChanged();
|
|
}
|
|
|
|
void
|
|
ComparePane::daterangeButtonsChanged()
|
|
{
|
|
// run through the table and see if anything changed
|
|
bool changed = false;
|
|
for (int i=0; i<table->rowCount(); i++) {
|
|
|
|
bool isChecked = static_cast<QCheckBox*>(table->cellWidget(i,0))->isChecked();
|
|
QColor color = static_cast<ColorButton*>(table->cellWidget(i,1))->getColor();
|
|
|
|
if (context->compareDateRanges[i].checked != isChecked ||
|
|
context->compareDateRanges[i].color != color) {
|
|
|
|
context->compareDateRanges[i].checked = isChecked;
|
|
context->compareDateRanges[i].color = color;
|
|
changed = true;
|
|
}
|
|
}
|
|
if (changed) context->notifyCompareDateRangesChanged();
|
|
}
|
|
|
|
void
|
|
ComparePane::dragEnterEvent(QDragEnterEvent *event)
|
|
{
|
|
if ((mode_ == interval && event->mimeData()->formats().contains("application/x-gc-intervals")) ||
|
|
(mode_ == season && event->mimeData()->formats().contains("application/x-gc-seasons"))) {
|
|
event->acceptProposedAction();
|
|
}
|
|
}
|
|
|
|
void
|
|
ComparePane::dragLeaveEvent(QDragLeaveEvent *)
|
|
{
|
|
// we might consider hiding on this?
|
|
}
|
|
|
|
void
|
|
ComparePane::dropEvent(QDropEvent *event)
|
|
{
|
|
// set action to copy and accept that so the source data
|
|
// is left intact and not wiped or removed
|
|
event->setDropAction(Qt::CopyAction);
|
|
event->accept();
|
|
|
|
// here we can unpack and add etc...
|
|
// lets get the context!
|
|
QString fmt = (mode_ == interval) ? "application/x-gc-intervals" : "application/x-gc-seasons";
|
|
|
|
// get the context out
|
|
QByteArray rawData = event->mimeData()->data(fmt);
|
|
QDataStream stream(&rawData, QIODevice::ReadOnly);
|
|
stream.setVersion(QDataStream::Qt_4_6);
|
|
|
|
// pack data
|
|
quint64 from;
|
|
stream >> from; // where did this come from?
|
|
|
|
// lets look at the context..
|
|
Context *sourceContext = (Context*)(from);
|
|
|
|
// NOW LETS UNPACK
|
|
if (mode_ == interval) { // INTERVALS
|
|
|
|
int count;
|
|
|
|
QList<CompareInterval> newOnes;
|
|
|
|
// lets get the basic data
|
|
stream >> count;
|
|
for (int i=0; i<count; i++) {
|
|
|
|
CompareInterval add;
|
|
|
|
add.checked = true; // UPDATE COMPARE INTERVAL
|
|
add.context = context; // UPDATE COMPARE INTERVAL
|
|
add.sourceContext = sourceContext; // UPDATE COMPARE INTERVAL
|
|
|
|
quint64 ridep;
|
|
quint64 start, stop, startKM, stopKM;
|
|
quint64 seq; // not relevant here
|
|
|
|
// serialize
|
|
stream >> add.name; // UPDATE COMPARE INTERVAL
|
|
|
|
stream >> ridep;
|
|
RideFile *ride = (RideFile*)ridep;
|
|
|
|
// index into ridefile
|
|
stream >> start;
|
|
stream >> stop;
|
|
stream >> startKM;
|
|
stream >> stopKM;
|
|
stream >> seq;
|
|
|
|
// construct a ridefile for the interval
|
|
|
|
// RideFile *data;
|
|
add.data = new RideFile(ride->startTime(), ride->recIntSecs());
|
|
add.data->context = context;
|
|
|
|
// manage offsets
|
|
bool first = true;
|
|
double offset = 0.0f, offsetKM = 0.0f;
|
|
|
|
foreach(RideFilePoint *p, ride->dataPoints()) {
|
|
|
|
if (p->secs > stop) break;
|
|
|
|
if (p->secs >= start) {
|
|
|
|
// intervals always start from zero when comparing
|
|
if (first) {
|
|
first = false;
|
|
offset = p->secs;
|
|
offsetKM = p->km;
|
|
}
|
|
|
|
add.data->appendPoint(p->secs - offset, p->cad, p->hr, p->km - offsetKM, p->kph, p->nm,
|
|
p->watts, p->alt, p->lon, p->lat, p->headwind,
|
|
p->slope, p->temp, p->lrbalance, 0);
|
|
|
|
// get derived data calculated
|
|
RideFilePoint *l = add.data->dataPoints().last();
|
|
l->np = p->np;
|
|
l->xp = p->xp;
|
|
l->apower = p->apower;
|
|
}
|
|
}
|
|
add.data->recalculateDerivedSeries();
|
|
|
|
// just use standard colors and cycle round
|
|
// we will of course repeat, but the user can
|
|
// just edit them using the button
|
|
add.color = standardColors.at((i + context->compareIntervals.count()) % standardColors.count());
|
|
|
|
// now add but only if not empty
|
|
if (!add.data->dataPoints().empty()) newOnes << add;
|
|
|
|
}
|
|
// how many we get ?
|
|
if (newOnes.count()) {
|
|
|
|
context->compareIntervals.append(newOnes);
|
|
|
|
// refresh the table to reflect the new list
|
|
refreshTable();
|
|
|
|
// let all the charts know
|
|
context->notifyCompareIntervalsChanged();
|
|
}
|
|
|
|
} else { // SEASONS
|
|
|
|
int count;
|
|
|
|
QList<CompareDateRange> newOnes;
|
|
|
|
// lets get the basic data
|
|
stream >> count;
|
|
for (int i=0; i<count; i++) {
|
|
|
|
CompareDateRange add;
|
|
|
|
add.checked = true; // UPDATE COMPARE INTERVAL
|
|
add.context = context; // UPDATE COMPARE INTERVAL
|
|
add.sourceContext = sourceContext; // UPDATE COMPARE INTERVAL
|
|
|
|
stream >> add.name;
|
|
stream >> add.start;
|
|
stream >> add.end;
|
|
stream >> add.days;
|
|
|
|
// get summary metrics for the season
|
|
// FROM THE SOURCE CONTEXT
|
|
// WE DON'T FETCH BESTS -- THEY NEED TO BE DONE AS NEEDED
|
|
add.metrics = sourceContext->athlete->metricDB->getAllMetricsFor(QDateTime(add.start, QTime()),QDateTime(add.end, QTime()));
|
|
add.measures = sourceContext->athlete->metricDB->getAllMeasuresFor(QDateTime(add.start, QTime()),QDateTime(add.end, QTime()));
|
|
|
|
// just use standard colors and cycle round
|
|
// we will of course repeat, but the user can
|
|
// just edit them using the button
|
|
add.color = standardColors.at((i + context->compareDateRanges.count()) % standardColors.count());
|
|
|
|
// even empty date ranges are valid
|
|
newOnes << add;
|
|
|
|
}
|
|
// how many we get ?
|
|
if (newOnes.count()) {
|
|
|
|
context->compareDateRanges.append(newOnes);
|
|
|
|
// refresh the table to reflect the new list
|
|
refreshTable();
|
|
|
|
// let all the charts know
|
|
context->notifyCompareDateRangesChanged();
|
|
}
|
|
|
|
}
|
|
}
|