240 Zeilen
6.3 KiB
C++

/**
* server/src/throttle/throttle.cpp
*
* This file is part of the traintastic source code.
*
* Copyright (C) 2022,2025 Reinder Feenstra
*
* 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 "throttle.hpp"
#include "list/throttlelist.hpp"
#include "list/throttlelisttablemodel.hpp"
#include "../core/attributes.hpp"
#include "../core/method.tpp"
#include "../core/objectproperty.tpp"
#include "../core/objectvectorproperty.tpp"
#include "../hardware/decoder/decoder.hpp"
#include "../log/log.hpp"
#include "../train/train.hpp"
#include "../utils/displayname.hpp"
#include "../utils/valuestep.hpp"
#include "../world/world.hpp"
namespace {
constexpr double getValueStep(double value, SpeedUnit unit)
{
switch(unit)
{
case SpeedUnit::KiloMeterPerHour:
return (value >= 40.0) ? 5.0 : 2.0;
case SpeedUnit::MilePerHour:
return (value >= 30.0) ? 5.0 : 2.0;
case SpeedUnit::MeterPerSecond:
return (value >= 10.0) ? 1.0 : 0.5;
}
return 1;
}
}
std::unordered_set<std::string, StringHash, StringEqual> Throttle::s_logIds;
std::string_view Throttle::getUniqueLogId(std::string_view prefix)
{
std::string uniqueLogId{prefix};
uniqueLogId.append("_");
uint32_t number = 0;
do
{
uniqueLogId.resize(prefix.size() + 1);
uniqueLogId.append(std::to_string(++number));
}
while(s_logIds.contains(uniqueLogId));
return *s_logIds.emplace(std::move(uniqueLogId)).first;
}
Throttle::Throttle(World& world, std::string_view logId)
: m_world{world}
, m_logId{logId}
, name{this, "name", std::string(logId), PropertyFlags::ReadWrite | PropertyFlags::NoStore | PropertyFlags::ScriptReadWrite}
, train{this, "train", nullptr, PropertyFlags::ReadOnly | PropertyFlags::NoStore | PropertyFlags::ScriptReadOnly}
, onAcquire{*this, "on_acquire", EventFlags::Scriptable | EventFlags::Public}
, onRelease{*this, "on_release", EventFlags::Scriptable | EventFlags::Public}
{
Attributes::addDisplayName(name, DisplayName::Object::name);
m_interfaceItems.add(name);
Attributes::addObjectEditor(train, false);
m_interfaceItems.add(train);
}
Throttle::~Throttle()
{
assert(!acquired());
if(auto it = s_logIds.find(m_logId); it != s_logIds.end()) [[likely]]
{
s_logIds.erase(it);
}
}
bool Throttle::acquired() const
{
return train.operator bool();
}
std::error_code Throttle::acquire(const std::shared_ptr<Train>& acquireTrain, bool steal)
{
assert(acquireTrain);
const auto stole = steal && acquireTrain->hasThrottle.value();
const std::string stoleFrom = stole ? acquireTrain->throttleName.value() : std::string{};
const auto ec = acquireTrain->acquire(*this, steal);
if(ec)
{
Log::log(m_logId, LogMessage::D3001_ACQUIRING_TRAIN_X_FAILED_X, acquireTrain->name.value(), ec.message());
return ec;
}
if(acquired())
{
release();
}
assert(!train);
train.setValueInternal(acquireTrain);
if(stole)
{
Log::log(m_logId, LogMessage::N3005_THROTTLE_X_STOLE_TRAIN_X_FROM_THROTTLE_X, name.value(), train->name.value(), stoleFrom);
}
else
{
Log::log(m_logId, LogMessage::I3001_THROTTLE_X_ACQUIRED_TRAIN_X, name.value(), train->name.value());
}
fireEvent(onAcquire, shared_ptr<Throttle>(), acquireTrain);
return {};
}
void Throttle::release(bool stopIt)
{
if(!acquired())
return;
if(stopIt)
{
emergencyStop();
}
const auto logMessage = !stopIt && !train->isStopped.value() ? LogMessage::N3006_THROTTLE_X_RELEASED_TRAIN_X_WITHOUT_STOPPING_IT : LogMessage::I3002_THROTTLE_X_RELEASED_TRAIN_X;
Log::log(m_logId, logMessage, name.value(), train->name.value());
train->release(*this);
train.setValueInternal(nullptr);
fireEvent(onRelease, shared_ptr<Throttle>());
}
bool Throttle::emergencyStop()
{
if(acquired())
{
train->emergencyStop = true;
return true;
}
return false;
}
bool Throttle::setDirection(Direction value)
{
if(acquired() && (value == Direction::Forward || value == Direction::Reverse))
{
return !train->setDirection(*this, value);
}
return false;
}
bool Throttle::setSpeed(double value, SpeedUnit unit)
{
if(acquired())
{
train->emergencyStop = false;
return !train->setSpeed(*this, convertUnit(value, unit, train->speed.unit()));
}
return false;
}
bool Throttle::setTargetSpeed(double value, SpeedUnit unit)
{
if(acquired())
{
train->emergencyStop = false;
return !train->setTargetSpeed(*this, convertUnit(value, unit, train->throttleSpeed.unit()));
}
return false;
}
bool Throttle::slower(bool immediate)
{
if(acquired())
{
train->emergencyStop = false;
if(immediate)
{
const double value = train->speed.value();
const double step = getValueStep(value, train->speed.unit());
return !train->setSpeed(*this, valueStepDown(value, step));
}
const double value = train->throttleSpeed.value();
const double step = getValueStep(value, train->throttleSpeed.unit());
return !train->setTargetSpeed(*this, valueStepDown(value, step));
}
return false;
}
bool Throttle::faster(bool immediate)
{
if(acquired())
{
train->emergencyStop = false;
if(immediate)
{
const double value = train->speed.value();
const double step = getValueStep(value, train->speed.unit());
return !train->setSpeed(*this, valueStepUp(value, step));
}
const double value = train->throttleSpeed.value();
const double step = getValueStep(value, train->throttleSpeed.unit());
return !train->setTargetSpeed(*this, valueStepUp(value, step));
}
return false;
}
void Throttle::destroying()
{
m_world.throttles->removeObject(shared_ptr<Throttle>());
release();
NonPersistentObject::destroying();
}
void Throttle::addToList()
{
m_world.throttles->addObject(shared_ptr<Throttle>());
}