OpenRCT2/src/openrct2/actions/GameAction.cpp

288 lines
8.8 KiB
C++
Raw Normal View History

/*****************************************************************************
* Copyright (c) 2014-2018 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.
*****************************************************************************/
2018-06-22 22:56:21 +02:00
#include "GameAction.h"
2017-09-27 23:07:26 +02:00
#include "../Context.h"
#include "../core/Guard.hpp"
#include "../core/Memory.hpp"
2017-03-23 20:49:19 +01:00
#include "../core/MemoryStream.h"
#include "../core/Util.hpp"
#include "../localisation/Localisation.h"
#include "../network/network.h"
2017-09-27 23:07:26 +02:00
#include "../platform/platform.h"
#include "../scenario/Scenario.h"
2017-12-31 21:40:00 +01:00
#include "../world/Park.h"
2018-06-22 22:56:21 +02:00
#include <algorithm>
GameActionResult::GameActionResult(GA_ERROR error, rct_string_id message)
{
Error = error;
ErrorMessage = message;
}
GameActionResult::GameActionResult(GA_ERROR error, rct_string_id title, rct_string_id message)
{
Error = error;
ErrorTitle = title;
ErrorMessage = message;
}
2018-06-22 22:56:21 +02:00
GameActionResult::GameActionResult(GA_ERROR error, rct_string_id title, rct_string_id message, uint8_t* args)
{
Error = error;
ErrorTitle = title;
ErrorMessage = message;
2018-02-01 13:52:39 +01:00
std::copy_n(args, ErrorMessageArgs.size(), ErrorMessageArgs.begin());
}
namespace GameActions
{
static GameActionFactory _actions[GAME_COMMAND_COUNT];
GameActionFactory Register(uint32_t id, GameActionFactory factory)
{
Guard::Assert(id < Util::CountOf(_actions));
Guard::ArgumentNotNull(factory);
_actions[id] = factory;
return factory;
}
void Initialize()
{
static bool initialized = false;
if (initialized)
return;
Register();
initialized = true;
}
std::unique_ptr<GameAction> Create(uint32_t id)
{
Initialize();
2018-06-22 22:56:21 +02:00
GameAction* result = nullptr;
if (id < Util::CountOf(_actions))
{
GameActionFactory factory = _actions[id];
if (factory != nullptr)
{
result = factory();
}
}
2018-05-14 12:36:45 +02:00
Guard::ArgumentNotNull(result, "Attempting to create unregistered gameaction: %u", id);
2017-07-21 19:54:05 +02:00
return std::unique_ptr<GameAction>(result);
}
static bool CheckActionInPausedMode(uint32_t actionFlags)
{
2018-06-22 22:56:21 +02:00
if (gGamePaused == 0)
return true;
if (gCheatsBuildInPauseMode)
return true;
if (actionFlags & GA_FLAGS::ALLOW_WHILE_PAUSED)
return true;
return false;
}
2018-06-22 22:56:21 +02:00
static bool CheckActionAffordability(const GameActionResult* result)
{
2018-06-22 22:56:21 +02:00
if (gParkFlags & PARK_FLAGS_NO_MONEY)
return true;
if (result->Cost <= 0)
return true;
if (result->Cost <= gCash)
return true;
return false;
}
2018-06-22 22:56:21 +02:00
GameActionResult::Ptr Query(const GameAction* action)
{
Guard::ArgumentNotNull(action);
uint16_t actionFlags = action->GetActionFlags();
if (!CheckActionInPausedMode(actionFlags))
{
GameActionResult::Ptr result = std::make_unique<GameActionResult>();
result->Error = GA_ERROR::GAME_PAUSED;
result->ErrorTitle = STR_RIDE_CONSTRUCTION_CANT_CONSTRUCT_THIS_HERE;
result->ErrorMessage = STR_CONSTRUCTION_NOT_POSSIBLE_WHILE_GAME_IS_PAUSED;
return result;
}
auto result = action->Query();
gCommandPosition.x = result->Position.x;
gCommandPosition.y = result->Position.y;
gCommandPosition.z = result->Position.z;
if (result->Error == GA_ERROR::OK)
{
if (!CheckActionAffordability(result.get()))
{
result->Error = GA_ERROR::INSUFFICIENT_FUNDS;
result->ErrorMessage = STR_NOT_ENOUGH_CASH_REQUIRES;
set_format_arg_on(result->ErrorMessageArgs.data(), 0, uint32_t, result->Cost);
}
}
return result;
}
2018-11-20 06:04:42 +01:00
static const char* GetRealm()
{
if (network_get_mode() == NETWORK_MODE_CLIENT)
return "cl";
else if (network_get_mode() == NETWORK_MODE_SERVER)
return "sv";
return "";
}
struct ActionLogContext_t
{
MemoryStream output;
};
static void LogActionBegin(ActionLogContext_t& ctx, const GameAction* action)
{
MemoryStream& output = ctx.output;
char temp[128] = {};
snprintf(temp, sizeof(temp), "[%s] Game Action %08X (", GetRealm(), action->GetType());
output.Write(temp, strlen(temp));
DataSerialiser ds(true, ctx.output, true); // Logging mode.
// Write all parameters into output as text.
action->Serialise(ds);
}
static void LogActionFinish(ActionLogContext_t& ctx, const GameAction* action, const GameActionResult::Ptr& result)
{
MemoryStream& output = ctx.output;
char temp[128] = {};
if (result->Error != GA_ERROR::OK)
{
snprintf(temp, sizeof(temp), ") Failed, %u", (uint32_t)result->Error);
}
else
{
snprintf(temp, sizeof(temp), ") OK");
}
output.Write(temp, strlen(temp) + 1);
const char* text = (const char*)output.GetData();
log_info(text);
network_append_server_log(text);
}
2018-06-22 22:56:21 +02:00
GameActionResult::Ptr Execute(const GameAction* action)
{
Guard::ArgumentNotNull(action);
uint16_t actionFlags = action->GetActionFlags();
uint32_t flags = action->GetFlags();
GameActionResult::Ptr result = Query(action);
if (result->Error == GA_ERROR::OK)
{
2017-03-21 20:05:53 +01:00
// Networked games send actions to the server to be run
if (network_get_mode() == NETWORK_MODE_CLIENT)
2017-03-21 20:05:53 +01:00
{
// As a client we have to wait or send it first.
if (!(actionFlags & GA_FLAGS::CLIENT_ONLY) && !(flags & GAME_COMMAND_FLAG_NETWORKED))
2017-03-21 20:05:53 +01:00
{
2017-10-27 11:14:37 +02:00
log_verbose("[%s] GameAction::Execute\n", "cl");
network_send_game_action(action);
return result;
}
}
else if (network_get_mode() == NETWORK_MODE_SERVER)
{
// If player is the server it would execute right away as where clients execute the commands
// at the beginning of the frame, so we have to put them into the queue.
if (!(actionFlags & GA_FLAGS::CLIENT_ONLY) && !(flags & GAME_COMMAND_FLAG_NETWORKED))
{
2017-10-27 11:14:37 +02:00
log_verbose("[%s] GameAction::Execute\n", "sv-cl");
network_enqueue_game_action(action);
return result;
2017-03-21 20:05:53 +01:00
}
}
2018-11-20 06:04:42 +01:00
ActionLogContext_t logContext;
LogActionBegin(logContext, action);
// Execute the action, changing the game state
result = action->Execute();
2018-11-20 06:04:42 +01:00
LogActionFinish(logContext, action, result);
gCommandPosition.x = result->Position.x;
gCommandPosition.y = result->Position.y;
gCommandPosition.z = result->Position.z;
// Update money balance
2018-06-22 22:56:21 +02:00
if (!(gParkFlags & PARK_FLAGS_NO_MONEY) && !(flags & GAME_COMMAND_FLAG_GHOST) && !(flags & GAME_COMMAND_FLAG_5)
&& result->Cost != 0)
{
finance_payment(result->Cost, result->ExpenditureType);
money_effect_create(result->Cost);
}
if (!(actionFlags & GA_FLAGS::CLIENT_ONLY))
{
if (network_get_mode() == NETWORK_MODE_SERVER && result->Error == GA_ERROR::OK)
{
2018-11-20 06:04:42 +01:00
NetworkPlayerId_t playerId = action->GetPlayer();
2018-11-20 06:04:42 +01:00
network_set_player_last_action(network_get_player_index(playerId.id), action->GetType());
2017-09-30 23:57:38 +02:00
if (result->Cost != 0)
{
2018-11-20 06:04:42 +01:00
network_add_player_money_spent(playerId.id, result->Cost);
2017-09-30 23:57:38 +02:00
}
}
}
// Allow autosave to commence
if (gLastAutoSaveUpdate == AUTOSAVE_PAUSE)
{
2017-07-14 08:02:56 +02:00
gLastAutoSaveUpdate = platform_get_ticks();
}
}
// Call callback for asynchronous events
auto cb = action->GetCallback();
if (cb != nullptr)
{
cb(action, result.get());
}
2018-06-22 22:56:21 +02:00
if (result->Error != GA_ERROR::OK && !(flags & GAME_COMMAND_FLAG_GHOST) && !(flags & GAME_COMMAND_FLAG_5))
{
// Show the error box
2018-02-01 13:52:39 +01:00
std::copy(result->ErrorMessageArgs.begin(), result->ErrorMessageArgs.end(), gCommonFormatArgs);
2017-09-27 23:07:26 +02:00
context_show_error(result->ErrorTitle, result->ErrorMessage);
}
2017-03-25 18:05:18 +01:00
return result;
}
2018-11-20 06:04:42 +01:00
2018-05-04 22:40:09 +02:00
} // namespace GameActions