Files
GoldenCheetah/src/ComputrainerController.cpp
Mark Liversedge 43b1c61d6b Add Device Wizard
Introduce a wizard for adding realtime devices since it
is prone to user error, is relatively complicated and is
something most users will only ever do once or twice.

There are several logical updates within this patch:

First; Fix intermittent ANT+ not working
* LibUsb     - check bufRemaining is > 0 not non-zero
* LibUsb     - Always reset the USB stick on open
* ANT.cpp    - Support > 4 channels on USB2
* ANTChannel -  Do not use pairing, but always
*               unassign, assign and set channel id.

Second; Fix device discovery
* Find and discover support in realtime controllers
* Extend Serial.cpp to cover more Serial devices
* Support for 4 or 8 ANT channels for USB1/USB2

Third; Introduce Add Device Wizard with
* General and Specific wizard pages for each device type
* Device pairing with telemetry display
* fixed compile time warnings

Fourth; Update Device Config Page
* use wizard to add new device
* remove edit fields, replaced by wizard
* remove pair, firmware buttons replaced by wizard

Fifth; Deprecate/Remove Device Types
* Null Device - used by developers only
* Quarqd client - replaced by Native ANT+
* GC Server - not implemented yet

I have also introduced a device specific wheel size
and flags for controlling the default selection and
sources for sensor data when in multi-device mode. These
are not yet supported in the code.

Fixes #611.
Fixes #497.
2012-01-21 20:28:33 +00:00

185 lines
4.4 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 "ComputrainerController.h"
#include "Computrainer.h"
#include "RealtimeData.h"
ComputrainerController::ComputrainerController(TrainTool *parent, DeviceConfiguration *dc) : RealtimeController(parent, dc)
{
myComputrainer = new Computrainer (parent, dc ? dc->portSpec : ""); // we may get NULL passed when configuring
}
int
ComputrainerController::start()
{
return myComputrainer->start();
}
int
ComputrainerController::restart()
{
return myComputrainer->restart();
}
int
ComputrainerController::pause()
{
return myComputrainer->pause();
}
int
ComputrainerController::stop()
{
return myComputrainer->stop();
}
bool
ComputrainerController::discover(QString name)
{
return myComputrainer->discover(name); // go probe it...
}
bool ComputrainerController::doesPush() { return false; }
bool ComputrainerController::doesPull() { return true; }
bool ComputrainerController::doesLoad() { return true; }
/*
* gets called from the GUI to get updated telemetry.
* so whilst we are at it we check button status too and
* act accordingly.
*
*/
void
ComputrainerController::getRealtimeData(RealtimeData &rtData)
{
int Buttons, Status;
bool calibration;
double Power, HeartRate, Cadence, Speed, RRC, Load;
uint8_t ss[24];
if(!myComputrainer->isRunning())
{
QMessageBox msgBox;
msgBox.setText("Cannot Connect to Computrainer");
msgBox.setIcon(QMessageBox::Critical);
msgBox.exec();
parent->Stop(1);
return;
}
// get latest telemetry
myComputrainer->getTelemetry(Power, HeartRate, Cadence, Speed,
RRC, calibration, Buttons, ss, Status);
//
// PASS BACK TELEMETRY
//
rtData.setWatts(Power);
rtData.setHr(HeartRate);
rtData.setCadence(Cadence);
rtData.setSpeed(Speed);
memcpy(rtData.spinScan, ss, 24);
// post processing, probably not used
// since its used to compute power for
// non-power devices, but we may add other
// calculations later that might apply
// means we could calculate power based
// upon speed even for CT!
processRealtimeData(rtData);
//
// BUTTONS
//
// toggle calibration
if (Buttons&CT_F3) {
parent->Calibrate();
}
// ignore other buttons if calibrating
if (parent->calibrating) return;
// ADJUST LOAD
Load = myComputrainer->getLoad();
if ((Buttons&CT_PLUS) && !(Buttons&CT_F3)) {
parent->Higher();
}
if ((Buttons&CT_MINUS) && !(Buttons&CT_F3)) {
parent->Lower();
}
rtData.setLoad(Load);
#if 0 // F3 now toggles calibration
// FFWD/REWIND
if ((Buttons&CT_PLUS) && (Buttons&CT_F3)) {
parent->FFwd();
}
if ((Buttons&CT_MINUS) && (Buttons&CT_F3)) {
parent->Rewind();
}
#endif
// LAP/INTERVAL
if (Buttons&CT_F1 && !(Buttons&CT_F3)) {
parent->newLap();
}
if ((Buttons&CT_F1) && (Buttons&CT_F3)) {
parent->FFwdLap();
}
// if Buttons == 0 we just pressed stop!
if (Buttons&CT_RESET) {
parent->Stop(0);
}
// displaymode
if (Buttons&CT_F2) {
parent->nextDisplayMode();
}
}
void ComputrainerController::pushRealtimeData(RealtimeData &) { } // update realtime data with current values
void
ComputrainerController::setLoad(double load)
{
myComputrainer->setLoad(load);
}
void
ComputrainerController::setGradient(double grade)
{
myComputrainer->setGradient(grade);
}
void
ComputrainerController::setMode(int mode)
{
if (mode == RT_MODE_ERGO) mode = CT_ERGOMODE;
if (mode == RT_MODE_SPIN) mode = CT_SSMODE;
if (mode == RT_MODE_CALIBRATE) mode = CT_CALIBRATE;
myComputrainer->setMode(mode);
}