/** * Thanks to Simon for his early research on the racing system. */ #pragma once #include "BitStream.h" #include "Entity.h" #include "Component.h" #include "eReplicaComponentType.h" #include /** * Information for each player in the race */ struct RacingPlayerInfo { /** * The ID of the player */ LWOOBJID playerID; /** * The ID of the car the player is driving */ LWOOBJID vehicleID; /** * The index of this player in the list of players */ uint32_t playerIndex; /** * Whether the player has finished loading or not */ bool playerLoaded; /** * Scripted activity component score */ float data[10]{}; /** * Point that the player will respawn at if they smash their car */ NiPoint3 respawnPosition; /** * Rotation that the player will respawn at if they smash their car */ NiQuaternion respawnRotation; /** * The index in the respawn point the player is now at */ uint32_t respawnIndex; /** * The number of laps the player has completed */ uint32_t lap; /** * Whether or not the player has finished the race */ uint32_t finished; /** * Unused */ uint16_t reachedPoints; /** * The fastest lap time of the player */ std::chrono::milliseconds bestLapTime; /** * The current lap time of the player */ std::chrono::high_resolution_clock::time_point lapTime; /** * The number of times this player smashed their car */ uint32_t smashedTimes = 0; /** * Whether or not the player should be smashed if the game is reloaded */ bool noSmashOnReload = false; /** * Whether or not this player has collected their rewards from completing the race */ bool collectedRewards = false; /** * Unused */ std::chrono::milliseconds raceTime; }; /** * Component that's attached to a manager entity in each race zone that loads player vehicles, keep scores, etc. */ class RacingControlComponent final : public Component { public: static constexpr eReplicaComponentType ComponentType = eReplicaComponentType::RACING_CONTROL; RacingControlComponent(Entity* parentEntity); ~RacingControlComponent(); void Serialize(RakNet::BitStream& outBitStream, bool bIsInitialUpdate) override; void Update(float deltaTime) override; /** * Invoked when a player loads into the zone. */ void OnPlayerLoaded(Entity* player); /** * Initalize the player's vehicle. * * @param player The player who's vehicle to initialize. * @param initialLoad Is this the first time the player is loading in this race? */ void LoadPlayerVehicle(Entity* player, uint32_t positionNumber, bool initialLoad = false); /** * Invoked when the client says it has loaded in. */ void OnRacingClientReady(Entity* player); /** * Invoked when the client says it should be smashed. */ void OnRequestDie(Entity* player); /** * Invoked when the player has finished respawning. */ void OnRacingPlayerInfoResetFinished(Entity* player); /** * Invoked when the player responds to the GUI. */ void HandleMessageBoxResponse(Entity* player, int32_t button, const std::string& id); /** * Get the racing data from a player's LWOOBJID. */ RacingPlayerInfo* GetPlayerData(LWOOBJID playerID); void MsgConfigureRacingControl(const GameMessages::ConfigureRacingControl& msg); private: /** * The players that are currently racing */ std::vector m_RacingPlayers; /** * The paths that are followed for the camera scenes * Configurable in the ConfigureRacingControl msg with the key `Race_PathName`. */ std::u16string m_PathName; /** * The ID of the activity for participating in this race * Configurable in the ConfigureRacingControl msg with the key `activityID`. */ uint32_t m_ActivityID; /** * The world the players return to when they finish the race */ uint32_t m_MainWorld; /** * The number of laps that are remaining for the winning player */ uint16_t m_RemainingLaps; /** * The ID of the player that's currently winning the race */ LWOOBJID m_LeadingPlayer; /** * The overall best lap from all the players */ float m_RaceBestLap; /** * The overall best time from all the players */ float m_RaceBestTime; /** * Whether or not the race has started */ bool m_Started; /** * The time left until the race will start */ float m_StartTimer; /** * The time left for loading the players */ float m_LoadTimer; /** * Whether or not all players have loaded */ bool m_Loaded; /** * The number of loaded players */ uint32_t m_LoadedPlayers; /** * All the players that are in the lobby, loaded or not */ std::vector m_LobbyPlayers; /** * The number of players that have finished the race */ uint32_t m_Finished; /** * The time the race was started */ std::chrono::high_resolution_clock::time_point m_StartTime; /** * Timer for tracking how long a player was alone in this race */ float m_EmptyTimer; bool m_SoloRacing; /** * Value for message box response to know if we are exiting the race via the activity dialogue */ const int32_t m_ActivityExitConfirm = 1; bool m_AllPlayersReady = false; /** * @brief The number of laps in this race. Configurable in the ConfigureRacingControl msg * with the key `Number_of_Laps`. * */ int32_t m_NumberOfLaps{ 3 }; /** * @brief The minimum number of players required to progress group achievements. * Configurable with the ConfigureRacingControl msg with the key `Minimum_Players_for_Group_Achievements`. * */ int32_t m_MinimumPlayersForGroupAchievements{ 2 }; };