2020-06-07 14:43:47 +02:00
|
|
|
/****************************************************************************
|
|
|
|
** ScStw Libraries
|
|
|
|
** Copyright (C) 2020 Itsblue development
|
|
|
|
**
|
|
|
|
** 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 3 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, see <http://www.gnu.org/licenses/>.
|
|
|
|
****************************************************************************/
|
|
|
|
|
2020-04-11 23:41:34 +02:00
|
|
|
#include "../headers/scstwtimer.h"
|
|
|
|
|
2020-06-13 12:38:24 +02:00
|
|
|
ScStwTimer::ScStwTimer(QObject *parent, bool directControlEnabled, QString letter) : QObject(parent)
|
2020-04-11 23:41:34 +02:00
|
|
|
{
|
2020-04-18 14:25:48 +02:00
|
|
|
this->directControlEnabled = directControlEnabled;
|
2020-06-13 15:57:27 +02:00
|
|
|
|
|
|
|
if(letter.length() > 1)
|
|
|
|
this->letter = letter[0];
|
|
|
|
else
|
|
|
|
this->letter = letter;
|
|
|
|
|
2020-04-11 23:41:34 +02:00
|
|
|
this->startTime = 0;
|
|
|
|
this->stopTime = 0;
|
|
|
|
this->reactionTime = 0;
|
2020-04-14 00:08:58 +02:00
|
|
|
|
|
|
|
this->state = IDLE;
|
2020-04-11 23:41:34 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
bool ScStwTimer::start() {
|
2020-04-14 00:08:58 +02:00
|
|
|
return this->start(QDateTime::currentMSecsSinceEpoch());
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ScStwTimer::start(double timeOfStart) {
|
2020-04-11 23:41:34 +02:00
|
|
|
switch (this->state) {
|
2020-04-14 00:08:58 +02:00
|
|
|
case IDLE: {
|
|
|
|
// in case of IDLE, start the race!
|
2020-04-11 23:41:34 +02:00
|
|
|
|
2020-04-14 00:08:58 +02:00
|
|
|
this->startTime = timeOfStart;
|
2020-04-11 23:41:34 +02:00
|
|
|
this->stopTime = 0;
|
|
|
|
|
2020-04-14 00:08:58 +02:00
|
|
|
if(timeOfStart - QDateTime::currentMSecsSinceEpoch() > 0) {
|
|
|
|
this->setState(STARTING);
|
|
|
|
QTimer::singleShot(timeOfStart - QDateTime::currentMSecsSinceEpoch(), [=](){
|
|
|
|
if(this->state == STARTING)
|
|
|
|
this->setState(RUNNING);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
else
|
|
|
|
this->setState(RUNNING);
|
2020-04-11 23:41:34 +02:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
default: {
|
|
|
|
// otherwise the timer is not supposed to be started!
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-04-14 00:08:58 +02:00
|
|
|
void ScStwTimer::handleClimberStart(double timeOfStart) {
|
|
|
|
this->reactionTime = timeOfStart - this->startTime;
|
|
|
|
qDebug() << "+ [INFO][TIMER] reaction time: " << this->reactionTime;
|
|
|
|
|
|
|
|
if(this->reactionTime <= 0){
|
|
|
|
this->stop(FailStop);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-05-18 10:49:11 +02:00
|
|
|
emit this->reactionTimeChanged();
|
2020-04-14 00:08:58 +02:00
|
|
|
}
|
|
|
|
|
2020-04-11 23:41:34 +02:00
|
|
|
bool ScStwTimer::cancel() {
|
2020-04-14 00:08:58 +02:00
|
|
|
if(!(this->state == IDLE || this->state == STARTING || this->state == RUNNING))
|
2020-04-11 23:41:34 +02:00
|
|
|
return false;
|
|
|
|
|
|
|
|
this->setState(CANCELLED);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ScStwTimer::stop() {
|
2020-04-15 19:21:32 +02:00
|
|
|
return this->stop(QDateTime::currentMSecsSinceEpoch());
|
2020-04-11 23:41:34 +02:00
|
|
|
}
|
|
|
|
|
2020-04-15 19:21:32 +02:00
|
|
|
bool ScStwTimer::stop(double timeOfStop) {
|
|
|
|
return this->stop(ManualStop, timeOfStop);
|
2020-04-11 23:41:34 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
bool ScStwTimer::stop(StopReason reason) {
|
2020-04-15 19:21:32 +02:00
|
|
|
return this->stop(reason, QDateTime::currentMSecsSinceEpoch());
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ScStwTimer::stop(StopReason reason, double timeOfStop) {
|
2020-04-11 23:41:34 +02:00
|
|
|
if(this->state != STARTING && this->state != RUNNING && this->state != WAITING){
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (reason) {
|
|
|
|
case ManualStop: {
|
|
|
|
if(this->state == STARTING){
|
|
|
|
this->setState(CANCELLED);
|
|
|
|
}
|
|
|
|
else {
|
2020-04-15 19:21:32 +02:00
|
|
|
this->stopTime = timeOfStop;
|
2020-04-11 23:41:34 +02:00
|
|
|
|
2020-04-15 19:21:32 +02:00
|
|
|
// trigger an external state refresh to set the state to either WON or LOST depending on the other timers values (see ScStwRace::refreshTimerStates())
|
|
|
|
this->setState(WAITING);
|
2020-04-11 23:41:34 +02:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case FailStop: {
|
2020-04-19 22:39:33 +02:00
|
|
|
qDebug() << "[INFO][TIMER] False Start detected: " << "start Time: " << startTime << " reactionTime: " << reactionTime;
|
2020-04-14 00:08:58 +02:00
|
|
|
this->setState(FAILED);
|
2020-04-19 22:39:33 +02:00
|
|
|
break;
|
2020-04-11 23:41:34 +02:00
|
|
|
}
|
|
|
|
default: {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-04-19 22:39:33 +02:00
|
|
|
qDebug() << "[INFO][TIMER] Stopped: " << "start Time: " << startTime << " stopTime: " << stopTime << " stoppedTime: " << this->getCurrentTime() << " reactionTime: " << reactionTime;
|
2020-04-11 23:41:34 +02:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2020-04-15 19:21:32 +02:00
|
|
|
bool ScStwTimer::setResult(TimerState result) {
|
|
|
|
if(this->state != WAITING)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
switch (result) {
|
|
|
|
case WON:
|
|
|
|
this->setState(WON);
|
|
|
|
return true;
|
|
|
|
case LOST:
|
|
|
|
this->setState(LOST);
|
|
|
|
return true;
|
|
|
|
default:
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-04-11 23:41:34 +02:00
|
|
|
bool ScStwTimer::reset(){
|
|
|
|
if( this->state < WON || this->state == DISABLED ){
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
this->startTime = 0;
|
|
|
|
this->stopTime = 0;
|
|
|
|
this->reactionTime = 0;
|
|
|
|
|
|
|
|
this->setState(IDLE);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// ------------------------
|
|
|
|
// --- helper functions ---
|
|
|
|
// ------------------------
|
|
|
|
|
2020-04-18 14:25:48 +02:00
|
|
|
bool ScStwTimer::setStartTime(double startTime) {
|
|
|
|
if(!this->directControlEnabled)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
this->startTime = startTime;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ScStwTimer::setStopTime(double stopTime) {
|
|
|
|
if(!this->directControlEnabled)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
this->stopTime = stopTime;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ScStwTimer::setReactionTime(double reactionTime) {
|
|
|
|
if(!this->directControlEnabled)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
this->reactionTime = reactionTime;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ScStwTimer::setState(TimerState newState, bool force) {
|
|
|
|
if(!this->directControlEnabled || !force)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
this->setState(newState);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2020-04-11 23:41:34 +02:00
|
|
|
void ScStwTimer::setState(TimerState newState){
|
|
|
|
if(this->state == DISABLED && newState != IDLE)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if(this->state != newState) {
|
|
|
|
this->state = newState;
|
|
|
|
qDebug() << "+ [INFO][TIMER] timer state changed: " << newState;
|
|
|
|
emit this->stateChanged();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-06-13 15:57:27 +02:00
|
|
|
bool ScStwTimer::setLetter(QString newLetter) {
|
|
|
|
if(!this->directControlEnabled)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
this->letter = newLetter;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2020-04-11 23:41:34 +02:00
|
|
|
ScStwTimer::TimerState ScStwTimer::getState() {
|
|
|
|
return this->state;
|
|
|
|
}
|
|
|
|
|
|
|
|
double ScStwTimer::getCurrentTime() {
|
2020-04-15 19:21:32 +02:00
|
|
|
switch (this->state) {
|
|
|
|
case RUNNING:
|
|
|
|
return QDateTime::currentMSecsSinceEpoch() - this->startTime;
|
|
|
|
default: {
|
|
|
|
if(this->state == WAITING || this->state == WON || this->state == LOST)
|
2020-06-13 18:24:27 +02:00
|
|
|
return abs(this->stopTime - this->startTime);
|
2020-04-15 19:21:32 +02:00
|
|
|
else
|
|
|
|
return -1;
|
2020-04-11 23:41:34 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
double ScStwTimer::getReactionTime() {
|
|
|
|
return this->reactionTime;
|
|
|
|
}
|
|
|
|
|
2020-06-13 12:42:55 +02:00
|
|
|
QString ScStwTimer::getLetter() {
|
|
|
|
return this->letter;
|
|
|
|
}
|
|
|
|
|
2020-04-17 19:57:00 +02:00
|
|
|
QString ScStwTimer::getText() {
|
2020-06-13 15:57:27 +02:00
|
|
|
|
|
|
|
QString newText = "";
|
|
|
|
int newTime = 0;
|
2020-04-17 19:57:00 +02:00
|
|
|
switch (this->state) {
|
|
|
|
case ScStwTimer::IDLE:
|
2020-06-13 15:57:27 +02:00
|
|
|
newTime = 0;
|
2020-04-17 19:57:00 +02:00
|
|
|
break;
|
|
|
|
case ScStwTimer::STARTING:
|
2020-06-13 15:57:27 +02:00
|
|
|
newTime = 0;
|
2020-04-17 19:57:00 +02:00
|
|
|
break;
|
|
|
|
case ScStwTimer::WAITING:
|
|
|
|
newText = "please wait...";
|
|
|
|
break;
|
|
|
|
case ScStwTimer::RUNNING:
|
2020-06-13 15:57:27 +02:00
|
|
|
newTime = this->getCurrentTime();
|
2020-04-17 19:57:00 +02:00
|
|
|
break;
|
|
|
|
case ScStwTimer::WON:
|
2020-06-13 15:57:27 +02:00
|
|
|
newTime = this->getCurrentTime();
|
2020-04-17 19:57:00 +02:00
|
|
|
break;
|
|
|
|
case ScStwTimer::LOST:
|
2020-06-13 15:57:27 +02:00
|
|
|
newTime = this->getCurrentTime();
|
2020-04-17 19:57:00 +02:00
|
|
|
break;
|
|
|
|
case ScStwTimer::FAILED:
|
|
|
|
newText = "false start";
|
|
|
|
break;
|
|
|
|
case ScStwTimer::CANCELLED:
|
|
|
|
newText = "cancelled";
|
|
|
|
break;
|
|
|
|
case ScStwTimer::DISABLED:
|
|
|
|
newText = "---";
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2020-06-13 15:57:27 +02:00
|
|
|
if(newText == "")
|
|
|
|
newText = QString::number( newTime / 1000.0, 'f', 3 ).rightJustified(6, '0');
|
|
|
|
|
2020-04-17 19:57:00 +02:00
|
|
|
return newText;
|
|
|
|
}
|
|
|
|
|
2020-04-11 23:41:34 +02:00
|
|
|
void ScStwTimer::setDisabled(bool disabled) {
|
|
|
|
if(disabled)
|
|
|
|
this->setState(DISABLED);
|
|
|
|
else
|
|
|
|
this->setState(IDLE);
|
|
|
|
}
|