mirror of
https://github.com/OpenRCT2/OpenRCT2
synced 2026-01-26 00:04:43 +01:00
654 lines
19 KiB
C++
654 lines
19 KiB
C++
/*****************************************************************************
|
|
* Copyright (c) 2014-2020 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 "Award.h"
|
|
|
|
#include "../config/Config.h"
|
|
#include "../interface/Window.h"
|
|
#include "../localisation/StringIds.h"
|
|
#include "../peep/Peep.h"
|
|
#include "../ride/Ride.h"
|
|
#include "../ride/RideData.h"
|
|
#include "../scenario/Scenario.h"
|
|
#include "../world/Park.h"
|
|
#include "NewsItem.h"
|
|
|
|
#include <algorithm>
|
|
|
|
constexpr uint8_t NEGATIVE = 0;
|
|
constexpr uint8_t POSITIVE = 1;
|
|
|
|
static constexpr const uint8_t AwardPositiveMap[] = {
|
|
NEGATIVE, // PARK_AWARD_MOST_UNTIDY
|
|
POSITIVE, // PARK_AWARD_MOST_TIDY
|
|
POSITIVE, // PARK_AWARD_BEST_ROLLERCOASTERS
|
|
POSITIVE, // PARK_AWARD_BEST_VALUE
|
|
POSITIVE, // PARK_AWARD_MOST_BEAUTIFUL
|
|
NEGATIVE, // PARK_AWARD_WORST_VALUE
|
|
POSITIVE, // PARK_AWARD_SAFEST
|
|
POSITIVE, // PARK_AWARD_BEST_STAFF
|
|
POSITIVE, // PARK_AWARD_BEST_FOOD
|
|
NEGATIVE, // PARK_AWARD_WORST_FOOD
|
|
POSITIVE, // PARK_AWARD_BEST_RESTROOMS
|
|
NEGATIVE, // PARK_AWARD_MOST_DISAPPOINTING
|
|
POSITIVE, // PARK_AWARD_BEST_WATER_RIDES
|
|
POSITIVE, // PARK_AWARD_BEST_CUSTOM_DESIGNED_RIDES
|
|
POSITIVE, // PARK_AWARD_MOST_DAZZLING_RIDE_COLOURS
|
|
NEGATIVE, // PARK_AWARD_MOST_CONFUSING_LAYOUT
|
|
POSITIVE, // PARK_AWARD_BEST_GENTLE_RIDES
|
|
};
|
|
|
|
static constexpr const rct_string_id AwardNewsStrings[] = {
|
|
STR_NEWS_ITEM_AWARD_MOST_UNTIDY,
|
|
STR_NEWS_ITEM_MOST_TIDY,
|
|
STR_NEWS_ITEM_BEST_ROLLERCOASTERS,
|
|
STR_NEWS_ITEM_BEST_VALUE,
|
|
STR_NEWS_ITEM_MOST_BEAUTIFUL,
|
|
STR_NEWS_ITEM_WORST_VALUE,
|
|
STR_NEWS_ITEM_SAFEST,
|
|
STR_NEWS_ITEM_BEST_STAFF,
|
|
STR_NEWS_ITEM_BEST_FOOD,
|
|
STR_NEWS_ITEM_WORST_FOOD,
|
|
STR_NEWS_ITEM_BEST_RESTROOMS,
|
|
STR_NEWS_ITEM_MOST_DISAPPOINTING,
|
|
STR_NEWS_ITEM_BEST_WATER_RIDES,
|
|
STR_NEWS_ITEM_BEST_CUSTOM_DESIGNED_RIDES,
|
|
STR_NEWS_ITEM_MOST_DAZZLING_RIDE_COLOURS,
|
|
STR_NEWS_ITEM_MOST_CONFUSING_LAYOUT,
|
|
STR_NEWS_ITEM_BEST_GENTLE_RIDES,
|
|
};
|
|
|
|
Award gCurrentAwards[MAX_AWARDS];
|
|
|
|
bool award_is_positive(int32_t type)
|
|
{
|
|
return AwardPositiveMap[type];
|
|
}
|
|
|
|
#pragma region Award checks
|
|
|
|
/** More than 1/16 of the total guests must be thinking untidy thoughts. */
|
|
static bool award_is_deserved_most_untidy(int32_t activeAwardTypes)
|
|
{
|
|
if (activeAwardTypes & (1 << PARK_AWARD_MOST_BEAUTIFUL))
|
|
return false;
|
|
if (activeAwardTypes & (1 << PARK_AWARD_BEST_STAFF))
|
|
return false;
|
|
if (activeAwardTypes & (1 << PARK_AWARD_MOST_TIDY))
|
|
return false;
|
|
|
|
uint32_t negativeCount = 0;
|
|
for (auto peep : EntityList<Guest>(EntityListId::Peep))
|
|
{
|
|
if (peep->OutsideOfPark)
|
|
continue;
|
|
|
|
if (peep->Thoughts[0].freshness > 5)
|
|
continue;
|
|
|
|
if (peep->Thoughts[0].type == PEEP_THOUGHT_TYPE_BAD_LITTER
|
|
|| peep->Thoughts[0].type == PEEP_THOUGHT_TYPE_PATH_DISGUSTING
|
|
|| peep->Thoughts[0].type == PEEP_THOUGHT_TYPE_VANDALISM)
|
|
{
|
|
negativeCount++;
|
|
}
|
|
}
|
|
|
|
return (negativeCount > gNumGuestsInPark / 16);
|
|
}
|
|
|
|
/** More than 1/64 of the total guests must be thinking tidy thoughts and less than 6 guests thinking untidy thoughts. */
|
|
static bool award_is_deserved_most_tidy(int32_t activeAwardTypes)
|
|
{
|
|
if (activeAwardTypes & (1 << PARK_AWARD_MOST_UNTIDY))
|
|
return false;
|
|
if (activeAwardTypes & (1 << PARK_AWARD_MOST_DISAPPOINTING))
|
|
return false;
|
|
|
|
uint32_t positiveCount = 0;
|
|
uint32_t negativeCount = 0;
|
|
for (auto peep : EntityList<Guest>(EntityListId::Peep))
|
|
{
|
|
if (peep->OutsideOfPark)
|
|
continue;
|
|
|
|
if (peep->Thoughts[0].freshness > 5)
|
|
continue;
|
|
|
|
if (peep->Thoughts[0].type == PEEP_THOUGHT_TYPE_VERY_CLEAN)
|
|
positiveCount++;
|
|
|
|
if (peep->Thoughts[0].type == PEEP_THOUGHT_TYPE_BAD_LITTER
|
|
|| peep->Thoughts[0].type == PEEP_THOUGHT_TYPE_PATH_DISGUSTING
|
|
|| peep->Thoughts[0].type == PEEP_THOUGHT_TYPE_VANDALISM)
|
|
{
|
|
negativeCount++;
|
|
}
|
|
}
|
|
|
|
return (negativeCount <= 5 && positiveCount > gNumGuestsInPark / 64);
|
|
}
|
|
|
|
/** At least 6 open roller coasters. */
|
|
static bool award_is_deserved_best_rollercoasters([[maybe_unused]] int32_t activeAwardTypes)
|
|
{
|
|
auto rollerCoasters = 0;
|
|
for (const auto& ride : GetRideManager())
|
|
{
|
|
auto rideEntry = ride.GetRideEntry();
|
|
if (rideEntry == nullptr)
|
|
{
|
|
continue;
|
|
}
|
|
|
|
if (ride.status != RIDE_STATUS_OPEN || (ride.lifecycle_flags & RIDE_LIFECYCLE_CRASHED))
|
|
{
|
|
continue;
|
|
}
|
|
|
|
if (!ride_entry_has_category(rideEntry, RIDE_CATEGORY_ROLLERCOASTER))
|
|
{
|
|
continue;
|
|
}
|
|
|
|
rollerCoasters++;
|
|
}
|
|
return (rollerCoasters >= 6);
|
|
}
|
|
|
|
/** Entrance fee is 0.10 less than half of the total ride value. */
|
|
static bool award_is_deserved_best_value(int32_t activeAwardTypes)
|
|
{
|
|
if (activeAwardTypes & (1 << PARK_AWARD_WORST_VALUE))
|
|
return false;
|
|
|
|
if (activeAwardTypes & (1 << PARK_AWARD_MOST_DISAPPOINTING))
|
|
return false;
|
|
|
|
if (gParkFlags & (PARK_FLAGS_NO_MONEY | PARK_FLAGS_PARK_FREE_ENTRY))
|
|
return false;
|
|
|
|
if (gTotalRideValueForMoney < MONEY(10, 00))
|
|
return false;
|
|
|
|
if (park_get_entrance_fee() + MONEY(0, 10) >= gTotalRideValueForMoney / 2)
|
|
return false;
|
|
|
|
return true;
|
|
}
|
|
|
|
/** More than 1/128 of the total guests must be thinking scenic thoughts and fewer than 16 untidy thoughts. */
|
|
static bool award_is_deserved_most_beautiful(int32_t activeAwardTypes)
|
|
{
|
|
if (activeAwardTypes & (1 << PARK_AWARD_MOST_UNTIDY))
|
|
return false;
|
|
if (activeAwardTypes & (1 << PARK_AWARD_MOST_DISAPPOINTING))
|
|
return false;
|
|
|
|
uint32_t positiveCount = 0;
|
|
uint32_t negativeCount = 0;
|
|
auto list = EntityList<Guest>(EntityListId::Peep);
|
|
for (auto peep : list)
|
|
{
|
|
if (peep->OutsideOfPark)
|
|
continue;
|
|
|
|
if (peep->Thoughts[0].freshness > 5)
|
|
continue;
|
|
|
|
if (peep->Thoughts[0].type == PEEP_THOUGHT_TYPE_SCENERY)
|
|
positiveCount++;
|
|
|
|
if (peep->Thoughts[0].type == PEEP_THOUGHT_TYPE_BAD_LITTER
|
|
|| peep->Thoughts[0].type == PEEP_THOUGHT_TYPE_PATH_DISGUSTING
|
|
|| peep->Thoughts[0].type == PEEP_THOUGHT_TYPE_VANDALISM)
|
|
{
|
|
negativeCount++;
|
|
}
|
|
}
|
|
|
|
return (negativeCount <= 15 && positiveCount > gNumGuestsInPark / 128);
|
|
}
|
|
|
|
/** Entrance fee is more than total ride value. */
|
|
static bool award_is_deserved_worst_value(int32_t activeAwardTypes)
|
|
{
|
|
if (activeAwardTypes & (1 << PARK_AWARD_BEST_VALUE))
|
|
return false;
|
|
if (gParkFlags & PARK_FLAGS_NO_MONEY)
|
|
return false;
|
|
|
|
money32 parkEntranceFee = park_get_entrance_fee();
|
|
if (parkEntranceFee == MONEY(0, 00))
|
|
return false;
|
|
if (parkEntranceFee <= gTotalRideValueForMoney)
|
|
return false;
|
|
return true;
|
|
}
|
|
|
|
/** No more than 2 people who think the vandalism is bad and no crashes. */
|
|
static bool award_is_deserved_safest([[maybe_unused]] int32_t activeAwardTypes)
|
|
{
|
|
auto peepsWhoDislikeVandalism = 0;
|
|
for (auto peep : EntityList<Guest>(EntityListId::Peep))
|
|
{
|
|
if (peep->OutsideOfPark)
|
|
continue;
|
|
if (peep->Thoughts[0].freshness <= 5 && peep->Thoughts[0].type == PEEP_THOUGHT_TYPE_VANDALISM)
|
|
peepsWhoDislikeVandalism++;
|
|
}
|
|
|
|
if (peepsWhoDislikeVandalism > 2)
|
|
return false;
|
|
|
|
// Check for rides that have crashed maybe?
|
|
const auto& rideManager = GetRideManager();
|
|
if (std::any_of(rideManager.begin(), rideManager.end(), [](const Ride& ride) {
|
|
return ride.last_crash_type != RIDE_CRASH_TYPE_NONE;
|
|
}))
|
|
{
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
/** All staff types, at least 20 staff, one staff per 32 peeps. */
|
|
static bool award_is_deserved_best_staff(int32_t activeAwardTypes)
|
|
{
|
|
if (activeAwardTypes & (1 << PARK_AWARD_MOST_UNTIDY))
|
|
return false;
|
|
|
|
auto peepCount = 0;
|
|
auto staffCount = 0;
|
|
auto staffTypeFlags = 0;
|
|
for (auto peep : EntityList<Peep>(EntityListId::Peep))
|
|
{
|
|
if (peep->AssignedPeepType == PeepType::Staff)
|
|
{
|
|
staffCount++;
|
|
staffTypeFlags |= (1 << peep->StaffType);
|
|
}
|
|
else
|
|
{
|
|
peepCount++;
|
|
}
|
|
}
|
|
|
|
return ((staffTypeFlags & 0xF) && staffCount >= 20 && staffCount >= peepCount / 32);
|
|
}
|
|
|
|
/** At least 7 shops, 4 unique, one shop per 128 guests and no more than 12 hungry guests. */
|
|
static bool award_is_deserved_best_food(int32_t activeAwardTypes)
|
|
{
|
|
if (activeAwardTypes & (1 << PARK_AWARD_WORST_FOOD))
|
|
return false;
|
|
|
|
uint32_t shops = 0;
|
|
uint32_t uniqueShops = 0;
|
|
uint32_t shopTypes = 0;
|
|
for (const auto& ride : GetRideManager())
|
|
{
|
|
if (ride.status != RIDE_STATUS_OPEN)
|
|
continue;
|
|
if (!ride_type_has_flag(ride.type, RIDE_TYPE_FLAG_SELLS_FOOD))
|
|
continue;
|
|
|
|
shops++;
|
|
auto rideEntry = get_ride_entry(ride.subtype);
|
|
if (rideEntry != nullptr)
|
|
{
|
|
if (!(shopTypes & (1ULL << rideEntry->shop_item[0])))
|
|
{
|
|
shopTypes |= (1ULL << rideEntry->shop_item[0]);
|
|
uniqueShops++;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (shops < 7 || uniqueShops < 4 || shops < gNumGuestsInPark / 128)
|
|
return false;
|
|
|
|
// Count hungry peeps
|
|
auto hungryPeeps = 0;
|
|
for (auto peep : EntityList<Guest>(EntityListId::Peep))
|
|
{
|
|
if (peep->OutsideOfPark)
|
|
continue;
|
|
|
|
if (peep->Thoughts[0].freshness <= 5 && peep->Thoughts[0].type == PEEP_THOUGHT_TYPE_HUNGRY)
|
|
hungryPeeps++;
|
|
}
|
|
return (hungryPeeps <= 12);
|
|
}
|
|
|
|
/** No more than 2 unique shops, less than one shop per 256 guests and more than 15 hungry guests. */
|
|
static bool award_is_deserved_worst_food(int32_t activeAwardTypes)
|
|
{
|
|
if (activeAwardTypes & (1 << PARK_AWARD_BEST_FOOD))
|
|
return false;
|
|
|
|
uint32_t shops = 0;
|
|
uint32_t uniqueShops = 0;
|
|
uint32_t shopTypes = 0;
|
|
for (const auto& ride : GetRideManager())
|
|
{
|
|
if (ride.status != RIDE_STATUS_OPEN)
|
|
continue;
|
|
if (!ride_type_has_flag(ride.type, RIDE_TYPE_FLAG_SELLS_FOOD))
|
|
continue;
|
|
|
|
shops++;
|
|
auto rideEntry = ride.GetRideEntry();
|
|
if (rideEntry != nullptr)
|
|
{
|
|
if (!(shopTypes & (1ULL << rideEntry->shop_item[0])))
|
|
{
|
|
shopTypes |= (1ULL << rideEntry->shop_item[0]);
|
|
uniqueShops++;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (uniqueShops > 2 || shops > gNumGuestsInPark / 256)
|
|
return false;
|
|
|
|
// Count hungry peeps
|
|
auto hungryPeeps = 0;
|
|
for (auto peep : EntityList<Guest>(EntityListId::Peep))
|
|
{
|
|
if (peep->OutsideOfPark)
|
|
continue;
|
|
|
|
if (peep->Thoughts[0].freshness <= 5 && peep->Thoughts[0].type == PEEP_THOUGHT_TYPE_HUNGRY)
|
|
hungryPeeps++;
|
|
}
|
|
return (hungryPeeps > 15);
|
|
}
|
|
|
|
/** At least 4 restrooms, 1 restroom per 128 guests and no more than 16 guests who think they need the restroom. */
|
|
static bool award_is_deserved_best_restrooms([[maybe_unused]] int32_t activeAwardTypes)
|
|
{
|
|
// Count open restrooms
|
|
const auto& rideManager = GetRideManager();
|
|
auto numRestrooms = static_cast<size_t>(std::count_if(rideManager.begin(), rideManager.end(), [](const Ride& ride) {
|
|
return ride.type == RIDE_TYPE_TOILETS && ride.status == RIDE_STATUS_OPEN;
|
|
}));
|
|
|
|
// At least 4 open restrooms
|
|
if (numRestrooms < 4)
|
|
return false;
|
|
|
|
// At least one open restroom for every 128 guests
|
|
if (numRestrooms < gNumGuestsInPark / 128U)
|
|
return false;
|
|
|
|
// Count number of guests who are thinking they need the restroom
|
|
auto guestsWhoNeedRestroom = 0;
|
|
for (auto peep : EntityList<Guest>(EntityListId::Peep))
|
|
{
|
|
if (peep->OutsideOfPark)
|
|
continue;
|
|
|
|
if (peep->Thoughts[0].freshness <= 5 && peep->Thoughts[0].type == PEEP_THOUGHT_TYPE_TOILET)
|
|
guestsWhoNeedRestroom++;
|
|
}
|
|
return (guestsWhoNeedRestroom <= 16);
|
|
}
|
|
|
|
/** More than half of the rides have satisfaction <= 6 and park rating <= 650. */
|
|
static bool award_is_deserved_most_disappointing(int32_t activeAwardTypes)
|
|
{
|
|
if (activeAwardTypes & (1 << PARK_AWARD_BEST_VALUE))
|
|
return false;
|
|
if (gParkRating > 650)
|
|
return false;
|
|
|
|
// Count the number of disappointing rides
|
|
auto countedRides = 0;
|
|
auto disappointingRides = 0;
|
|
for (const auto& ride : GetRideManager())
|
|
{
|
|
if (ride_has_ratings(&ride) && ride.popularity != 0xFF)
|
|
{
|
|
countedRides++;
|
|
if (ride.popularity <= 6)
|
|
{
|
|
disappointingRides++;
|
|
}
|
|
}
|
|
}
|
|
|
|
// Half of the rides are disappointing
|
|
return (disappointingRides >= countedRides / 2);
|
|
}
|
|
|
|
/** At least 6 open water rides. */
|
|
static bool award_is_deserved_best_water_rides([[maybe_unused]] int32_t activeAwardTypes)
|
|
{
|
|
auto waterRides = 0;
|
|
for (const auto& ride : GetRideManager())
|
|
{
|
|
auto rideEntry = ride.GetRideEntry();
|
|
if (rideEntry == nullptr)
|
|
{
|
|
continue;
|
|
}
|
|
|
|
if (ride.status != RIDE_STATUS_OPEN || (ride.lifecycle_flags & RIDE_LIFECYCLE_CRASHED))
|
|
{
|
|
continue;
|
|
}
|
|
|
|
if (!ride_entry_has_category(rideEntry, RIDE_CATEGORY_WATER))
|
|
{
|
|
continue;
|
|
}
|
|
|
|
waterRides++;
|
|
}
|
|
|
|
return (waterRides >= 6);
|
|
}
|
|
|
|
/** At least 6 custom designed rides. */
|
|
static bool award_is_deserved_best_custom_designed_rides(int32_t activeAwardTypes)
|
|
{
|
|
if (activeAwardTypes & (1 << PARK_AWARD_MOST_DISAPPOINTING))
|
|
return false;
|
|
|
|
auto customDesignedRides = 0;
|
|
for (const auto& ride : GetRideManager())
|
|
{
|
|
if (!ride_type_has_flag(ride.type, RIDE_TYPE_FLAG_HAS_TRACK))
|
|
continue;
|
|
if (ride.lifecycle_flags & RIDE_LIFECYCLE_NOT_CUSTOM_DESIGN)
|
|
continue;
|
|
if (ride.excitement < RIDE_RATING(5, 50))
|
|
continue;
|
|
if (ride.status != RIDE_STATUS_OPEN || (ride.lifecycle_flags & RIDE_LIFECYCLE_CRASHED))
|
|
continue;
|
|
|
|
customDesignedRides++;
|
|
}
|
|
|
|
return (customDesignedRides >= 6);
|
|
}
|
|
|
|
static bool award_is_deserved_most_dazzling_ride_colours(int32_t activeAwardTypes)
|
|
{
|
|
/** At least 5 colourful rides and more than half of the rides are colourful. */
|
|
static constexpr const colour_t dazzling_ride_colours[] = { COLOUR_BRIGHT_PURPLE, COLOUR_BRIGHT_GREEN, COLOUR_LIGHT_ORANGE,
|
|
COLOUR_BRIGHT_PINK };
|
|
|
|
if (activeAwardTypes & (1 << PARK_AWARD_MOST_DISAPPOINTING))
|
|
return false;
|
|
|
|
auto countedRides = 0;
|
|
auto colourfulRides = 0;
|
|
for (const auto& ride : GetRideManager())
|
|
{
|
|
if (!ride_type_has_flag(ride.type, RIDE_TYPE_FLAG_HAS_TRACK))
|
|
continue;
|
|
|
|
countedRides++;
|
|
|
|
auto mainTrackColour = ride.track_colour[0].main;
|
|
for (auto dazzling_ride_colour : dazzling_ride_colours)
|
|
{
|
|
if (mainTrackColour == dazzling_ride_colour)
|
|
{
|
|
colourfulRides++;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
return (colourfulRides >= 5 && colourfulRides >= countedRides - colourfulRides);
|
|
}
|
|
|
|
/** At least 10 peeps and more than 1/64 of total guests are lost or can't find something. */
|
|
static bool award_is_deserved_most_confusing_layout([[maybe_unused]] int32_t activeAwardTypes)
|
|
{
|
|
uint32_t peepsCounted = 0;
|
|
uint32_t peepsLost = 0;
|
|
for (auto peep : EntityList<Guest>(EntityListId::Peep))
|
|
{
|
|
if (peep->OutsideOfPark)
|
|
continue;
|
|
|
|
peepsCounted++;
|
|
if (peep->Thoughts[0].freshness <= 5
|
|
&& (peep->Thoughts[0].type == PEEP_THOUGHT_TYPE_LOST || peep->Thoughts[0].type == PEEP_THOUGHT_TYPE_CANT_FIND))
|
|
peepsLost++;
|
|
}
|
|
|
|
return (peepsLost >= 10 && peepsLost >= peepsCounted / 64);
|
|
}
|
|
|
|
/** At least 10 open gentle rides. */
|
|
static bool award_is_deserved_best_gentle_rides([[maybe_unused]] int32_t activeAwardTypes)
|
|
{
|
|
auto gentleRides = 0;
|
|
for (const auto& ride : GetRideManager())
|
|
{
|
|
auto rideEntry = ride.GetRideEntry();
|
|
if (rideEntry == nullptr)
|
|
{
|
|
continue;
|
|
}
|
|
|
|
if (ride.status != RIDE_STATUS_OPEN || (ride.lifecycle_flags & RIDE_LIFECYCLE_CRASHED))
|
|
{
|
|
continue;
|
|
}
|
|
|
|
if (!ride_entry_has_category(rideEntry, RIDE_CATEGORY_GENTLE))
|
|
{
|
|
continue;
|
|
}
|
|
|
|
gentleRides++;
|
|
}
|
|
|
|
return (gentleRides >= 10);
|
|
}
|
|
|
|
using award_deserved_check = bool (*)(int32_t);
|
|
|
|
static constexpr const award_deserved_check _awardChecks[] = {
|
|
award_is_deserved_most_untidy,
|
|
award_is_deserved_most_tidy,
|
|
award_is_deserved_best_rollercoasters,
|
|
award_is_deserved_best_value,
|
|
award_is_deserved_most_beautiful,
|
|
award_is_deserved_worst_value,
|
|
award_is_deserved_safest,
|
|
award_is_deserved_best_staff,
|
|
award_is_deserved_best_food,
|
|
award_is_deserved_worst_food,
|
|
award_is_deserved_best_restrooms,
|
|
award_is_deserved_most_disappointing,
|
|
award_is_deserved_best_water_rides,
|
|
award_is_deserved_best_custom_designed_rides,
|
|
award_is_deserved_most_dazzling_ride_colours,
|
|
award_is_deserved_most_confusing_layout,
|
|
award_is_deserved_best_gentle_rides,
|
|
};
|
|
|
|
static bool award_is_deserved(int32_t awardType, int32_t activeAwardTypes)
|
|
{
|
|
return _awardChecks[awardType](activeAwardTypes);
|
|
}
|
|
|
|
#pragma endregion
|
|
|
|
void award_reset()
|
|
{
|
|
for (auto& award : gCurrentAwards)
|
|
{
|
|
award.Time = 0;
|
|
award.Type = 0;
|
|
}
|
|
}
|
|
|
|
/**
|
|
*
|
|
* rct2: 0x0066A86C
|
|
*/
|
|
void award_update_all()
|
|
{
|
|
// Only add new awards if park is open
|
|
if (gParkFlags & PARK_FLAGS_PARK_OPEN)
|
|
{
|
|
// Set active award types as flags
|
|
int32_t activeAwardTypes = 0;
|
|
int32_t freeAwardEntryIndex = -1;
|
|
for (int32_t i = 0; i < MAX_AWARDS; i++)
|
|
{
|
|
if (gCurrentAwards[i].Time != 0)
|
|
activeAwardTypes |= (1 << gCurrentAwards[i].Type);
|
|
else if (freeAwardEntryIndex == -1)
|
|
freeAwardEntryIndex = i;
|
|
}
|
|
|
|
// Check if there was a free award entry
|
|
if (freeAwardEntryIndex != -1)
|
|
{
|
|
// Get a random award type not already active
|
|
int32_t awardType;
|
|
do
|
|
{
|
|
awardType = (((scenario_rand() & 0xFF) * 17) >> 8) & 0xFF;
|
|
} while (activeAwardTypes & (1 << awardType));
|
|
|
|
// Check if award is deserved
|
|
if (award_is_deserved(awardType, activeAwardTypes))
|
|
{
|
|
// Add award
|
|
gCurrentAwards[freeAwardEntryIndex].Type = awardType;
|
|
gCurrentAwards[freeAwardEntryIndex].Time = 5;
|
|
if (gConfigNotifications.park_award)
|
|
{
|
|
news_item_add_to_queue(NEWS_ITEM_AWARD, AwardNewsStrings[awardType], 0);
|
|
}
|
|
window_invalidate_by_class(WC_PARK_INFORMATION);
|
|
}
|
|
}
|
|
}
|
|
|
|
// Decrease award times
|
|
for (auto& award : gCurrentAwards)
|
|
{
|
|
if (award.Time != 0)
|
|
if (--award.Time == 0)
|
|
window_invalidate_by_class(WC_PARK_INFORMATION);
|
|
}
|
|
}
|