2020-04-12 10:00:02 -04:00
|
|
|
/*****************************************************************************
|
2023-01-01 05:58:01 -05:00
|
|
|
* Copyright (c) 2014-2023 OpenRCT2 developers
|
2020-04-12 10:00:02 -04:00
|
|
|
*
|
|
|
|
* 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 "TestData.h"
|
|
|
|
|
|
|
|
#include <gtest/gtest.h>
|
2022-07-29 12:45:10 -04:00
|
|
|
#include <memory>
|
2020-04-12 10:00:02 -04:00
|
|
|
#include <openrct2/Context.h>
|
|
|
|
#include <openrct2/Game.h>
|
|
|
|
#include <openrct2/GameState.h>
|
|
|
|
#include <openrct2/OpenRCT2.h>
|
|
|
|
#include <openrct2/ParkImporter.h>
|
2023-01-02 08:42:28 -05:00
|
|
|
#include <openrct2/actions/ParkSetEntranceFeeAction.h>
|
Split actions hpp files into separate h and cpp files (#13548)
* Split up SmallSceneryPlace/Remove
Added undo function for Remove Scenery
* Refactor: Balloon and Banner actions hpp=>h/cpp
* Refactor: rename all action *.hpp files to *.cpp
This is preparation for separation in later commits. Note that without
the complete set of commits in this branch, the code will not build.
* Refactor Clear, Climate, Custom, and Footpath actions hpp=>h/cpp
* VSCode: add src subdirectories to includePath
* Refactor Guest actions hpp=>h/cpp
* Refactor Land actions hpp=>h/cpp
* Refactor LargeScenery actions hpp=>h/cpp
* Refactor Load, Maze, Network actions hpp=>h/cpp
* Refactor Park actions hpp=>h/cpp
* Refactor/style: move private function declarations in actions *.h
Previous action .h files included private function declarations with
private member variables, before public function declarations. This
commit re-orders the header files to the following order:
- public member variables
- private member variables
- public functions
- private functions
* Refactor Pause action hpp=>h/cpp
* Refactor Peep, Place, Player actions hpp=>h/cpp
* Refactor Ride actions hpp=>h/cpp
* Refactor Scenario, Set*, Sign* actions hpp=>h/cpp
* Refactor SmallScenerySetColourAction hpp=>h/cpp
* Refactor Staff actions hpp=>h/cpp
* Refactor Surface, Tile, Track* actions hpp=>h/cpp
* Refactor Wall and Water actions hpp=>h/cpp
* Fix various includes and other compile errors
Update includes for tests.
Move static function declarations to .h files
Add explicit includes to various files that were previously implicit
(the required header was a nested include in an action hpp file, and the
action .h file does not include that header)
Move RideSetStatus string enum to the cpp file to avoid unused imports
* Xcode: modify project file for actions refactor
* Cleanup whitespace and end-of-file newlines
Co-authored-by: duncanspumpkin <duncans_pumpkin@hotmail.co.uk>
2020-12-10 01:39:10 -05:00
|
|
|
#include <openrct2/actions/ParkSetParameterAction.h>
|
|
|
|
#include <openrct2/actions/RideSetPriceAction.h>
|
2022-08-10 18:00:58 -04:00
|
|
|
#include <openrct2/actions/RideSetStatusAction.h>
|
2021-11-24 09:58:01 -05:00
|
|
|
#include <openrct2/entity/EntityRegistry.h>
|
2021-11-24 09:48:33 -05:00
|
|
|
#include <openrct2/entity/EntityTweener.h>
|
2021-11-25 16:47:24 -05:00
|
|
|
#include <openrct2/entity/Peep.h>
|
2020-04-12 10:00:02 -04:00
|
|
|
#include <openrct2/object/ObjectManager.h>
|
2022-02-18 15:57:00 -05:00
|
|
|
#include <openrct2/platform/Platform.h>
|
2020-04-12 10:00:02 -04:00
|
|
|
#include <openrct2/ride/Ride.h>
|
|
|
|
#include <openrct2/world/MapAnimation.h>
|
|
|
|
#include <openrct2/world/Park.h>
|
|
|
|
#include <openrct2/world/Scenery.h>
|
|
|
|
#include <string>
|
|
|
|
|
|
|
|
using namespace OpenRCT2;
|
|
|
|
|
|
|
|
class PlayTests : public testing::Test
|
|
|
|
{
|
|
|
|
};
|
|
|
|
|
|
|
|
static std::unique_ptr<IContext> localStartGame(const std::string& parkPath)
|
|
|
|
{
|
|
|
|
gOpenRCT2Headless = true;
|
|
|
|
gOpenRCT2NoGraphics = true;
|
2022-02-18 15:57:00 -05:00
|
|
|
Platform::CoreInit();
|
2020-04-12 10:00:02 -04:00
|
|
|
|
|
|
|
auto context = CreateContext();
|
|
|
|
if (!context->Initialise())
|
|
|
|
return {};
|
|
|
|
|
|
|
|
auto importer = ParkImporter::CreateS6(context->GetObjectRepository());
|
|
|
|
auto loadResult = importer->LoadSavedGame(parkPath.c_str(), false);
|
2021-09-13 14:24:42 -04:00
|
|
|
context->GetObjectManager().LoadObjects(loadResult.RequiredObjects);
|
2020-04-12 10:00:02 -04:00
|
|
|
importer->Import();
|
|
|
|
|
2021-11-24 08:37:47 -05:00
|
|
|
ResetEntitySpatialIndices();
|
2020-04-12 10:00:02 -04:00
|
|
|
|
|
|
|
reset_all_sprite_quadrant_placements();
|
2022-10-08 05:56:17 -04:00
|
|
|
ScenerySetDefaultPlacementConfiguration();
|
2020-04-12 10:00:02 -04:00
|
|
|
load_palette();
|
2021-01-04 16:31:41 -05:00
|
|
|
EntityTweener::Get().Reset();
|
2022-10-04 15:24:38 -04:00
|
|
|
MapAnimationAutoCreate();
|
2020-04-12 10:00:02 -04:00
|
|
|
fix_invalid_vehicle_sprite_sizes();
|
|
|
|
|
|
|
|
gGameSpeed = 1;
|
|
|
|
|
|
|
|
return context;
|
|
|
|
}
|
|
|
|
|
|
|
|
template<class Fn> static bool updateUntil(GameState& gs, int maxSteps, Fn&& fn)
|
|
|
|
{
|
|
|
|
while (maxSteps-- && !fn())
|
|
|
|
{
|
|
|
|
gs.UpdateLogic();
|
|
|
|
}
|
|
|
|
return maxSteps > 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
template<class GA, class... Args> static void execute(Args&&... args)
|
|
|
|
{
|
|
|
|
GA ga(std::forward<Args>(args)...);
|
|
|
|
GameActions::Execute(&ga);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(PlayTests, SecondGuestInQueueShouldNotRideIfNoFunds)
|
|
|
|
{
|
|
|
|
/* This test verifies that a guest, when second in queue, won't be forced to enter
|
|
|
|
* the ride if it has not enough money to pay for it.
|
|
|
|
* To simulate this scenario, two guests (a rich and a poor) are encouraged to enter
|
|
|
|
* the ride queue, and then the price is raised such that the second guest in line
|
|
|
|
* (the poor one) cannot pay. The poor guest should not enter the ride.
|
|
|
|
*/
|
|
|
|
std::string initStateFile = TestData::GetParkPath("small_park_with_ferris_wheel.sv6");
|
|
|
|
|
|
|
|
auto context = localStartGame(initStateFile);
|
|
|
|
ASSERT_NE(context.get(), nullptr);
|
|
|
|
|
|
|
|
auto gs = context->GetGameState();
|
|
|
|
ASSERT_NE(gs, nullptr);
|
|
|
|
|
|
|
|
// Open park for free but charging for rides
|
|
|
|
execute<ParkSetParameterAction>(ParkParameter::Open);
|
2023-01-02 08:42:28 -05:00
|
|
|
execute<ParkSetEntranceFeeAction>(0);
|
2020-04-12 10:00:02 -04:00
|
|
|
gParkFlags |= PARK_FLAGS_UNLOCK_ALL_PRICES;
|
|
|
|
|
|
|
|
// Find ferris wheel
|
|
|
|
auto rideManager = GetRideManager();
|
|
|
|
auto it = std::find_if(
|
|
|
|
rideManager.begin(), rideManager.end(), [](auto& ride) { return ride.type == RIDE_TYPE_FERRIS_WHEEL; });
|
|
|
|
ASSERT_NE(it, rideManager.end());
|
|
|
|
Ride& ferrisWheel = *it;
|
|
|
|
|
|
|
|
// Open it for free
|
2022-08-10 18:00:58 -04:00
|
|
|
execute<RideSetStatusAction>(ferrisWheel.id, RideStatus::Open);
|
2020-04-12 10:00:02 -04:00
|
|
|
execute<RideSetPriceAction>(ferrisWheel.id, 0, true);
|
|
|
|
|
2021-01-10 13:23:35 -05:00
|
|
|
// Ignore intensity to stimulate peeps to queue into ferris wheel
|
2020-04-12 10:00:02 -04:00
|
|
|
gCheatsIgnoreRideIntensity = true;
|
|
|
|
|
|
|
|
// Insert a rich guest
|
|
|
|
auto richGuest = gs->GetPark().GenerateGuest();
|
2020-06-06 09:19:37 -04:00
|
|
|
richGuest->CashInPocket = 3000;
|
2020-04-12 10:00:02 -04:00
|
|
|
|
|
|
|
// Wait for rich guest to get in queue
|
2020-09-28 14:51:49 -04:00
|
|
|
bool matched = updateUntil(*gs, 1000, [&]() { return richGuest->State == PeepState::Queuing; });
|
2020-04-12 10:00:02 -04:00
|
|
|
ASSERT_TRUE(matched);
|
|
|
|
|
|
|
|
// Insert poor guest
|
|
|
|
auto poorGuest = gs->GetPark().GenerateGuest();
|
2020-06-06 09:19:37 -04:00
|
|
|
poorGuest->CashInPocket = 5;
|
2020-04-12 10:00:02 -04:00
|
|
|
|
|
|
|
// Wait for poor guest to get in queue
|
2020-09-28 14:51:49 -04:00
|
|
|
matched = updateUntil(*gs, 1000, [&]() { return poorGuest->State == PeepState::Queuing; });
|
2020-04-12 10:00:02 -04:00
|
|
|
ASSERT_TRUE(matched);
|
|
|
|
|
|
|
|
// Raise the price of the ride to a value poor guest can't pay
|
|
|
|
execute<RideSetPriceAction>(ferrisWheel.id, 10, true);
|
|
|
|
|
|
|
|
// Verify that the poor guest goes back to walking without riding
|
|
|
|
// since it doesn't have enough money to pay for it
|
|
|
|
bool enteredTheRide = false;
|
|
|
|
matched = updateUntil(*gs, 10000, [&]() {
|
2020-09-28 14:51:49 -04:00
|
|
|
enteredTheRide |= poorGuest->State == PeepState::OnRide;
|
|
|
|
return poorGuest->State == PeepState::Walking || enteredTheRide;
|
2020-04-12 10:00:02 -04:00
|
|
|
});
|
|
|
|
|
|
|
|
ASSERT_TRUE(matched);
|
|
|
|
ASSERT_FALSE(enteredTheRide);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(PlayTests, CarRideWithOneCarOnlyAcceptsTwoGuests)
|
|
|
|
{
|
|
|
|
// This test verifies that a car ride with one car will accept at most two guests
|
|
|
|
std::string initStateFile = TestData::GetParkPath("small_park_car_ride_one_car.sv6");
|
|
|
|
|
|
|
|
auto context = localStartGame(initStateFile);
|
|
|
|
ASSERT_NE(context.get(), nullptr);
|
|
|
|
|
|
|
|
auto gs = context->GetGameState();
|
|
|
|
ASSERT_NE(gs, nullptr);
|
|
|
|
|
|
|
|
// Open park for free but charging for rides
|
|
|
|
execute<ParkSetParameterAction>(ParkParameter::Open);
|
2023-01-02 08:42:28 -05:00
|
|
|
execute<ParkSetEntranceFeeAction>(0);
|
2020-04-12 10:00:02 -04:00
|
|
|
gParkFlags |= PARK_FLAGS_UNLOCK_ALL_PRICES;
|
|
|
|
|
|
|
|
// Find car ride
|
|
|
|
auto rideManager = GetRideManager();
|
|
|
|
auto it = std::find_if(rideManager.begin(), rideManager.end(), [](auto& ride) { return ride.type == RIDE_TYPE_CAR_RIDE; });
|
|
|
|
ASSERT_NE(it, rideManager.end());
|
|
|
|
Ride& carRide = *it;
|
|
|
|
|
|
|
|
// Open it for free
|
2022-08-10 18:00:58 -04:00
|
|
|
execute<RideSetStatusAction>(carRide.id, RideStatus::Open);
|
2020-04-12 10:00:02 -04:00
|
|
|
execute<RideSetPriceAction>(carRide.id, 0, true);
|
|
|
|
|
2021-01-10 13:23:35 -05:00
|
|
|
// Ignore intensity to stimulate peeps to queue into the ride
|
2020-04-12 10:00:02 -04:00
|
|
|
gCheatsIgnoreRideIntensity = true;
|
|
|
|
|
|
|
|
// Create some guests
|
|
|
|
std::vector<Peep*> guests;
|
|
|
|
for (int i = 0; i < 25; i++)
|
|
|
|
{
|
|
|
|
guests.push_back(gs->GetPark().GenerateGuest());
|
|
|
|
}
|
|
|
|
|
|
|
|
// Wait until one of them is riding
|
2020-09-28 14:51:49 -04:00
|
|
|
auto guestIsOnRide = [](auto* g) { return g->State == PeepState::OnRide; };
|
2020-04-12 10:00:02 -04:00
|
|
|
bool matched = updateUntil(*gs, 10000, [&]() { return std::any_of(guests.begin(), guests.end(), guestIsOnRide); });
|
|
|
|
ASSERT_TRUE(matched);
|
|
|
|
|
|
|
|
// For the next few ticks at most two guests can be on the ride
|
|
|
|
for (int i = 0; i < 100; i++)
|
|
|
|
{
|
|
|
|
int numRiding = std::count_if(guests.begin(), guests.end(), guestIsOnRide);
|
|
|
|
ASSERT_LE(numRiding, 2);
|
|
|
|
gs->UpdateLogic();
|
|
|
|
}
|
|
|
|
}
|