1
0
mirror of https://github.com/OpenRCT2/OpenRCT2 synced 2026-01-22 14:24:33 +01:00
Files
OpenRCT2/src/openrct2/actions/RideSetVehicleAction.cpp
2025-09-08 16:21:29 +02:00

280 lines
10 KiB
C++

/*****************************************************************************
* Copyright (c) 2014-2025 OpenRCT2 developers
*
* For a complete list of all authors, please refer to contributors.md
* Interested in contributing? Visit https://github.com/OpenRCT2/OpenRCT2
*
* OpenRCT2 is licensed under the GNU General Public License version 3.
*****************************************************************************/
#include "RideSetVehicleAction.h"
#include "../Cheats.h"
#include "../Context.h"
#include "../Diagnostic.h"
#include "../GameState.h"
#include "../core/EnumUtils.hpp"
#include "../core/MemoryStream.h"
#include "../drawing/Drawing.h"
#include "../localisation/StringIds.h"
#include "../management/Research.h"
#include "../object/ObjectManager.h"
#include "../ride/Ride.h"
#include "../ride/RideData.h"
#include "../ui/WindowManager.h"
#include "../world/Map.h"
#include "../world/Park.h"
namespace OpenRCT2::GameActions
{
constexpr static StringId kSetVehicleTypeErrorTitle[] = {
STR_RIDE_SET_VEHICLE_SET_NUM_TRAINS_FAIL,
STR_RIDE_SET_VEHICLE_SET_NUM_CARS_PER_TRAIN_FAIL,
STR_RIDE_SET_VEHICLE_TYPE_FAIL,
STR_RIDE_SET_VEHICLE_REVERSED_FAIL,
};
RideSetVehicleAction::RideSetVehicleAction(RideId rideIndex, RideSetVehicleType type, uint16_t value, uint8_t colour)
: _rideIndex(rideIndex)
, _type(type)
, _value(value)
, _colour(colour)
{
}
void RideSetVehicleAction::AcceptParameters(GameActionParameterVisitor& visitor)
{
visitor.Visit("ride", _rideIndex);
visitor.Visit("type", _type);
visitor.Visit("value", _value);
visitor.Visit("colour", _colour);
}
uint16_t RideSetVehicleAction::GetActionFlags() const
{
return GameAction::GetActionFlags() | Flags::AllowWhilePaused;
}
void RideSetVehicleAction::Serialise(DataSerialiser& stream)
{
GameAction::Serialise(stream);
stream << DS_TAG(_rideIndex) << DS_TAG(_type) << DS_TAG(_value) << DS_TAG(_colour);
}
Result RideSetVehicleAction::Query(GameState_t& gameState) const
{
if (_type >= RideSetVehicleType::Count)
{
LOG_ERROR("Invalid ride vehicle type %d", _type);
}
auto errTitle = kSetVehicleTypeErrorTitle[EnumValue(_type)];
auto ride = GetRide(_rideIndex);
if (ride == nullptr)
{
LOG_ERROR("Ride not found for rideIndex %u", _rideIndex.ToUnderlying());
return Result(Status::InvalidParameters, errTitle, STR_ERR_RIDE_NOT_FOUND);
}
if (ride->lifecycleFlags & RIDE_LIFECYCLE_BROKEN_DOWN)
{
return Result(Status::Broken, errTitle, STR_HAS_BROKEN_DOWN_AND_REQUIRES_FIXING);
}
if (ride->status != RideStatus::closed && ride->status != RideStatus::simulating)
{
return Result(Status::NotClosed, errTitle, STR_MUST_BE_CLOSED_FIRST);
}
switch (_type)
{
case RideSetVehicleType::NumTrains:
case RideSetVehicleType::NumCarsPerTrain:
case RideSetVehicleType::TrainsReversed:
break;
case RideSetVehicleType::RideEntry:
{
if (!RideIsVehicleTypeValid(*ride))
{
LOG_ERROR("Invalid vehicle type %d", _value);
return Result(Status::InvalidParameters, errTitle, STR_ERR_VALUE_OUT_OF_RANGE);
}
auto rideEntry = GetRideEntryByIndex(_value);
if (rideEntry == nullptr)
{
LOG_ERROR("Ride entry not found for _value %d", _value);
return Result(Status::InvalidParameters, errTitle, kStringIdNone);
}
// Validate preset
VehicleColourPresetList* presetList = rideEntry->vehicle_preset_list;
if (_colour >= presetList->count && _colour != 255 && _colour != 0)
{
LOG_ERROR("Unknown vehicle colour preset. colour = %d", _colour);
return Result(Status::InvalidParameters, errTitle, STR_ERR_INVALID_COLOUR);
}
break;
}
default:
LOG_ERROR("Invalid ride vehicle setting %d", _type);
return Result(Status::InvalidParameters, errTitle, STR_ERR_VALUE_OUT_OF_RANGE);
}
return Result();
}
Result RideSetVehicleAction::Execute(GameState_t& gameState) const
{
auto errTitle = kSetVehicleTypeErrorTitle[EnumValue(_type)];
auto ride = GetRide(_rideIndex);
if (ride == nullptr)
{
LOG_ERROR("Ride not found for rideIndex %u", _rideIndex.ToUnderlying());
return Result(Status::InvalidParameters, errTitle, STR_ERR_RIDE_NOT_FOUND);
}
switch (_type)
{
case RideSetVehicleType::NumTrains:
RideClearForConstruction(*ride);
ride->removePeeps();
ride->vehicleChangeTimeout = 100;
ride->proposedNumTrains = _value;
break;
case RideSetVehicleType::NumCarsPerTrain:
{
RideClearForConstruction(*ride);
ride->removePeeps();
ride->vehicleChangeTimeout = 100;
InvalidateTestResults(*ride);
auto rideEntry = GetRideEntryByIndex(ride->subtype);
if (rideEntry == nullptr)
{
LOG_ERROR("Ride entry not found for index %d", ride->subtype);
return Result(Status::InvalidParameters, errTitle, kStringIdNone);
}
uint8_t clampValue = _value;
static_assert(sizeof(clampValue) == sizeof(ride->proposedNumCarsPerTrain));
if (!getGameState().cheats.disableTrainLengthLimit)
{
clampValue = std::clamp(clampValue, rideEntry->min_cars_in_train, rideEntry->max_cars_in_train);
}
ride->proposedNumCarsPerTrain = clampValue;
break;
}
case RideSetVehicleType::RideEntry:
{
RideClearForConstruction(*ride);
ride->removePeeps();
ride->vehicleChangeTimeout = 100;
InvalidateTestResults(*ride);
ride->subtype = _value;
auto rideEntry = GetRideEntryByIndex(ride->subtype);
if (rideEntry == nullptr)
{
LOG_ERROR("Ride entry not found for index %d", ride->subtype);
return Result(Status::InvalidParameters, errTitle, kStringIdNone);
}
RideSetVehicleColoursToRandomPreset(*ride, _colour);
if (!getGameState().cheats.disableTrainLengthLimit)
{
ride->proposedNumCarsPerTrain = std::clamp(
ride->proposedNumCarsPerTrain, rideEntry->min_cars_in_train, rideEntry->max_cars_in_train);
}
break;
}
case RideSetVehicleType::TrainsReversed:
{
RideClearForConstruction(*ride);
ride->removePeeps();
ride->vehicleChangeTimeout = 100;
ride->setLifecycleFlag(RIDE_LIFECYCLE_REVERSED_TRAINS, _value);
break;
}
default:
LOG_ERROR("Invalid ride vehicle setting %d", _type);
return Result(Status::InvalidParameters, errTitle, kStringIdNone);
}
ride->numCircuits = 1;
ride->updateMaxVehicles();
auto res = Result();
if (!ride->overallView.IsNull())
{
auto location = ride->overallView.ToTileCentre();
res.Position = { location, TileElementHeight(res.Position) };
}
auto intent = Intent(INTENT_ACTION_RIDE_PAINT_RESET_VEHICLE);
intent.PutExtra(INTENT_EXTRA_RIDE_ID, _rideIndex.ToUnderlying());
ContextBroadcastIntent(&intent);
GfxInvalidateScreen();
return res;
}
bool RideSetVehicleAction::RideIsVehicleTypeValid(const Ride& ride) const
{
bool selectionShouldBeExpanded;
ride_type_t rideTypeIterator, rideTypeIteratorMax;
auto& gameState = getGameState();
{
const auto& rtd = ride.getRideTypeDescriptor();
if (gameState.cheats.showVehiclesFromOtherTrackTypes
&& !(
ride.getRideTypeDescriptor().HasFlag(RtdFlag::isFlatRide) || rtd.specialType == RtdSpecialType::maze
|| rtd.specialType == RtdSpecialType::miniGolf))
{
selectionShouldBeExpanded = true;
rideTypeIterator = 0;
rideTypeIteratorMax = RIDE_TYPE_COUNT - 1;
}
else
{
selectionShouldBeExpanded = false;
rideTypeIterator = ride.type;
rideTypeIteratorMax = ride.type;
}
}
for (; rideTypeIterator <= rideTypeIteratorMax; rideTypeIterator++)
{
if (selectionShouldBeExpanded)
{
if (GetRideTypeDescriptor(rideTypeIterator).HasFlag(RtdFlag::isFlatRide))
continue;
const auto& rtd = GetRideTypeDescriptor(rideTypeIterator);
if (rtd.specialType == RtdSpecialType::maze || rtd.specialType == RtdSpecialType::miniGolf)
continue;
}
auto& objManager = GetContext()->GetObjectManager();
auto& rideEntries = objManager.GetAllRideEntries(rideTypeIterator);
for (auto rideEntryIndex : rideEntries)
{
if (rideEntryIndex == _value)
{
if (!RideEntryIsInvented(rideEntryIndex) && !gameState.cheats.ignoreResearchStatus)
{
return false;
}
return true;
}
}
}
return false;
}
} // namespace OpenRCT2::GameActions