498 lines
15 KiB
C++
498 lines
15 KiB
C++
/*
|
|
Copyright (c) 2013 yvt
|
|
based on code of pysnip (c) Mathias Kaerlev 2011-2012.
|
|
|
|
This file is part of OpenSpades.
|
|
|
|
OpenSpades is free software: you can redistribute it and/or modify
|
|
it under the terms of the GNU General Public License as published by
|
|
the Free Software Foundation, either version 3 of the License, or
|
|
(at your option) any later version.
|
|
|
|
OpenSpades is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with OpenSpades. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
#pragma once
|
|
|
|
#include <list>
|
|
#include <memory>
|
|
#include <string>
|
|
#include <tuple>
|
|
|
|
#include "ClientCameraMode.h"
|
|
#include "ILocalEntity.h"
|
|
#include "IRenderer.h"
|
|
#include "IWorldListener.h"
|
|
#include "MumbleLink.h"
|
|
#include "NoiseSampler.h"
|
|
#include "Player.h"
|
|
#include <Core/Math.h>
|
|
#include <Core/ServerAddress.h>
|
|
#include <Core/Stopwatch.h>
|
|
#include <Gui/View.h>
|
|
|
|
namespace spades {
|
|
class IStream;
|
|
class Stopwatch;
|
|
namespace client {
|
|
class IRenderer;
|
|
struct SceneDefinition;
|
|
class GameMap;
|
|
class GameMapWrapper;
|
|
class World;
|
|
struct PlayerInput;
|
|
struct WeaponInput;
|
|
class IAudioDevice;
|
|
class IAudioChunk;
|
|
class NetClient;
|
|
class IFont;
|
|
class FontManager;
|
|
class ChatWindow;
|
|
class CenterMessageView;
|
|
class Corpse;
|
|
class HurtRingView;
|
|
class MapView;
|
|
class ScoreboardView;
|
|
class LimboView;
|
|
class Player;
|
|
class PaletteView;
|
|
class TCProgressView;
|
|
class ClientPlayer;
|
|
|
|
class ClientUI;
|
|
|
|
class Client : public IWorldListener, public gui::View {
|
|
friend class ScoreboardView;
|
|
friend class LimboView;
|
|
friend class MapView;
|
|
friend class FallingBlock;
|
|
friend class PaletteView;
|
|
friend class TCProgressView;
|
|
friend class ClientPlayer;
|
|
friend class ClientUI;
|
|
|
|
/** used to keep the input state of keypad so that
|
|
* after user pressed left and right, and then
|
|
* released right, left is internally pressed. */
|
|
struct KeypadInput {
|
|
bool left, right, forward, backward;
|
|
KeypadInput() : left(false), right(false), forward(false), backward(false) {}
|
|
};
|
|
|
|
class FPSCounter {
|
|
Stopwatch sw;
|
|
int numFrames;
|
|
double lastFps;
|
|
|
|
public:
|
|
FPSCounter();
|
|
void MarkFrame();
|
|
double GetFps() { return lastFps; }
|
|
};
|
|
|
|
FPSCounter fpsCounter;
|
|
FPSCounter upsCounter;
|
|
|
|
std::unique_ptr<NetClient> net;
|
|
std::string playerName;
|
|
std::unique_ptr<IStream> logStream;
|
|
|
|
Handle<ClientUI> scriptedUI;
|
|
|
|
ServerAddress hostname;
|
|
|
|
std::unique_ptr<World> world;
|
|
Handle<GameMap> map;
|
|
std::unique_ptr<GameMapWrapper> mapWrapper;
|
|
Handle<IRenderer> renderer;
|
|
Handle<IAudioDevice> audioDevice;
|
|
float time;
|
|
bool readyToClose;
|
|
float worldSubFrame;
|
|
|
|
int frameToRendererInit;
|
|
float timeSinceInit;
|
|
|
|
MumbleLink mumbleLink;
|
|
|
|
// view/drawing state for some world objects
|
|
std::vector<Handle<ClientPlayer>> clientPlayers;
|
|
|
|
// other windows
|
|
std::unique_ptr<CenterMessageView> centerMessageView;
|
|
std::unique_ptr<HurtRingView> hurtRingView;
|
|
std::unique_ptr<MapView> mapView;
|
|
std::unique_ptr<MapView> largeMapView;
|
|
std::unique_ptr<ScoreboardView> scoreboard;
|
|
std::unique_ptr<LimboView> limbo;
|
|
std::unique_ptr<PaletteView> paletteView;
|
|
std::unique_ptr<TCProgressView> tcView;
|
|
|
|
// chat
|
|
std::unique_ptr<ChatWindow> chatWindow;
|
|
std::unique_ptr<ChatWindow> killfeedWindow;
|
|
|
|
// player state
|
|
PlayerInput playerInput;
|
|
WeaponInput weapInput;
|
|
KeypadInput keypadInput;
|
|
Player::ToolType lastTool;
|
|
bool hasLastTool;
|
|
Vector3 lastFront;
|
|
float lastPosSentTime;
|
|
int lastHealth;
|
|
float lastHurtTime;
|
|
float lastAliveTime;
|
|
int lastKills;
|
|
float worldSetTime;
|
|
bool hasDelayedReload;
|
|
struct HurtSprite {
|
|
float angle;
|
|
float horzShift;
|
|
float scale;
|
|
float strength;
|
|
};
|
|
std::vector<HurtSprite> hurtSprites;
|
|
|
|
float GetAimDownState();
|
|
float GetSprintState();
|
|
|
|
/**
|
|
* Queries whether the local player is allowed to use a tool in this state.
|
|
*
|
|
* The following factors are considered by this function:
|
|
*
|
|
* - The player cannot use a tool while / soon after sprinting.
|
|
* - The player cannot use a tool while switching tools.
|
|
* - The player must exist and be alive to use a tool.
|
|
*
|
|
* The following factors also affect whether a tool can actually be used, but they
|
|
* do not affect the result of this function:
|
|
*
|
|
* - Tool-specific status — e.g., out of ammo, out of block, "cannot build there"
|
|
* - Firing rate limit imposed by the tool
|
|
*/
|
|
bool CanLocalPlayerUseToolNow();
|
|
|
|
/** Retrieves `ClientPlayer` for the local player, or `{}` if it does not exist. */
|
|
stmp::optional<ClientPlayer &> GetLocalClientPlayer();
|
|
|
|
float toolRaiseState;
|
|
void SetSelectedTool(Player::ToolType, bool quiet = false);
|
|
|
|
// view
|
|
SceneDefinition lastSceneDef;
|
|
/** Derived from `lastSceneDef`. */
|
|
Matrix4 lastViewProjectionScreenMatrix;
|
|
float localFireVibrationTime;
|
|
float grenadeVibration;
|
|
float grenadeVibrationSlow;
|
|
bool scoreboardVisible;
|
|
bool flashlightOn;
|
|
float flashlightOnTime;
|
|
CoherentNoiseSampler1D coherentNoiseSamplers[3];
|
|
void KickCamera(float strength);
|
|
|
|
float hitFeedbackIconState;
|
|
bool hitFeedbackFriendly;
|
|
|
|
// manual focus
|
|
float focalLength;
|
|
float targetFocalLength;
|
|
bool autoFocusEnabled;
|
|
|
|
// Spectator camera control
|
|
/** The state of the following camera used for spectating. */
|
|
struct {
|
|
/**
|
|
* Indicates whether the current camera mode is first-person or not.
|
|
* Ignored and locked to third-person when the target player
|
|
* (`followedPlayerId`) is dead.
|
|
*/
|
|
bool firstPerson = true;
|
|
|
|
/** Controls whether the follow camera is enabled. */
|
|
bool enabled = false;
|
|
} followCameraState;
|
|
|
|
/** The state of the free floating camera used for spectating. */
|
|
struct {
|
|
/** The temporally smoothed position (I guess). */
|
|
Vector3 position{0.0f, 0.0f, 0.0f};
|
|
|
|
/** The temporally smoothed velocity (I guess). */
|
|
Vector3 velocity{0.0f, 0.0f, 0.0f};
|
|
} freeCameraState;
|
|
|
|
/**
|
|
* The state shared by both of the third-person and free-floating cameras.
|
|
*
|
|
* Note: These values are not used in the `cg_thirdperson` mode.
|
|
*/
|
|
struct {
|
|
/** The yaw angle. */
|
|
float yaw = 0.0f;
|
|
|
|
/** The pitch angle. */
|
|
float pitch = 0.0f;
|
|
} followAndFreeCameraState;
|
|
|
|
/**
|
|
* The ID of the player being followed (in a spectator mode, or when the local player is
|
|
* dead). Must be valid as long as the follow cam is enabled.
|
|
*
|
|
* Must *not* specify a local player.
|
|
*/
|
|
int followedPlayerId;
|
|
|
|
/**
|
|
* Chooses the next player to follow and assigns it to `this->followingPlayerId`.
|
|
* Enables the follow cam by assigning `true` to `followCameraState.enabled`.
|
|
* If the next player is the local player, disables the follow cam.
|
|
*/
|
|
void FollowNextPlayer(bool reverse);
|
|
|
|
/**
|
|
* Retrieves the current camera mode.
|
|
*/
|
|
ClientCameraMode GetCameraMode();
|
|
|
|
/**
|
|
* Retrieves the target player ID of the current camera mode (as returned by
|
|
* `GetCameraMode`).
|
|
*
|
|
* Throws an exception if the current camera mode does not have a player in concern.
|
|
*/
|
|
int GetCameraTargetPlayerId();
|
|
|
|
/**
|
|
* Retrieves the target player of the current camera mode (as returned by
|
|
* `GetCameraMode`).
|
|
*
|
|
* Throws an exception if the current camera mode does not have a player in concern.
|
|
*/
|
|
Player &GetCameraTargetPlayer();
|
|
|
|
/**
|
|
* Calculate the zoom value incorporating the effect of ADS for a first-person view.
|
|
*
|
|
* The camera mode must be first-person.
|
|
*/
|
|
float GetAimDownZoomScale();
|
|
|
|
bool inGameLimbo;
|
|
|
|
float GetLocalFireVibration();
|
|
void CaptureColor();
|
|
bool IsLimboViewActive();
|
|
void SpawnPressed();
|
|
|
|
stmp::optional<std::tuple<Player &, hitTag_t>> HotTrackedPlayer();
|
|
|
|
// effects (local entity, etc)
|
|
std::vector<DynamicLightParam> flashDlights;
|
|
std::vector<DynamicLightParam> flashDlightsOld;
|
|
void Bleed(Vector3);
|
|
void EmitBlockFragments(Vector3, IntVector3 color);
|
|
void EmitBlockDestroyFragments(IntVector3, IntVector3 color);
|
|
void GrenadeExplosion(Vector3);
|
|
void GrenadeExplosionUnderwater(Vector3);
|
|
void MuzzleFire(Vector3, Vector3 dir, bool local);
|
|
void BulletHitWaterSurface(Vector3);
|
|
|
|
// drawings
|
|
Handle<FontManager> fontManager;
|
|
|
|
enum class AlertType { Notice, Warning, Error };
|
|
AlertType alertType;
|
|
std::string alertContents;
|
|
float alertDisappearTime;
|
|
float alertAppearTime;
|
|
|
|
// Loading screen
|
|
float mapReceivingProgressSmoothed = 0.0;
|
|
|
|
std::list<std::unique_ptr<ILocalEntity>> localEntities;
|
|
std::list<std::unique_ptr<Corpse>> corpses;
|
|
Corpse *lastMyCorpse;
|
|
float corpseSoftTimeLimit;
|
|
unsigned int corpseSoftLimit;
|
|
unsigned int corpseHardLimit;
|
|
void RemoveAllCorpses();
|
|
void RemoveInvisibleCorpses();
|
|
void RemoveAllLocalEntities();
|
|
void RemoveCorpseForPlayer(int playerId);
|
|
|
|
int nextScreenShotIndex;
|
|
int nextMapShotIndex;
|
|
|
|
/** Project the specified world-space position to a screen space. */
|
|
Vector3 Project(Vector3);
|
|
|
|
/** Recalculate `lastViewProjectionScreenMatrix` based on the current value of
|
|
* `lastSceneDef`. */
|
|
void UpdateMatrices();
|
|
|
|
void DrawSplash();
|
|
void DrawStartupScreen();
|
|
void DrawDisconnectScreen();
|
|
void DoInit();
|
|
|
|
void ShowAlert(const std::string &contents, AlertType type);
|
|
void ShowAlert(const std::string &contents, AlertType type, float timeout,
|
|
bool quiet = false);
|
|
void PlayAlertSound();
|
|
|
|
void UpdateWorld(float dt);
|
|
void UpdateLocalSpectator(float dt);
|
|
void UpdateLocalPlayer(float dt);
|
|
void UpdateAutoFocus(float dt);
|
|
float RayCastForAutoFocus(const Vector3 &origin, const Vector3 &direction);
|
|
|
|
void Draw2D();
|
|
|
|
void Draw2DWithoutWorld();
|
|
void Draw2DWithWorld();
|
|
|
|
/** Called when the local plyaer is alive. */
|
|
void DrawJoinedAlivePlayerHUD();
|
|
/** Called when the local plyaer is dead. */
|
|
void DrawDeadPlayerHUD();
|
|
|
|
/**
|
|
* Called when `IsFirstPerson(GetCameraMode()).` Renders the follwing element:
|
|
* - The center reticule
|
|
*/
|
|
void DrawFirstPersonHUD();
|
|
|
|
/**
|
|
* Called when the local player is dead or a spectator.
|
|
*/
|
|
void DrawSpectateHUD();
|
|
|
|
void DrawHottrackedPlayerName();
|
|
void DrawHurtScreenEffect();
|
|
void DrawHurtSprites();
|
|
void DrawHealth();
|
|
void DrawAlert();
|
|
void DrawDebugAim();
|
|
void DrawStats();
|
|
|
|
void DrawScene();
|
|
void AddGrenadeToScene(Grenade &);
|
|
void AddDebugObjectToScene(const OBB3 &, const Vector4 &col = MakeVector4(1, 1, 1, 1));
|
|
void DrawCTFObjects();
|
|
void DrawTCObjects();
|
|
|
|
SceneDefinition CreateSceneDefinition();
|
|
|
|
std::string ScreenShotPath();
|
|
void TakeScreenShot(bool sceneOnly);
|
|
|
|
std::string MapShotPath();
|
|
void TakeMapShot();
|
|
|
|
void NetLog(const char *format, ...);
|
|
|
|
protected:
|
|
~Client();
|
|
|
|
public:
|
|
Client(Handle<IRenderer>, Handle<IAudioDevice>, const ServerAddress &host,
|
|
Handle<FontManager>);
|
|
|
|
void RunFrame(float dt) override;
|
|
void RunFrameLate(float dt) override;
|
|
|
|
void Closing() override;
|
|
void MouseEvent(float x, float y) override;
|
|
void WheelEvent(float x, float y) override;
|
|
void KeyEvent(const std::string &, bool down) override;
|
|
void TextInputEvent(const std::string &) override;
|
|
void TextEditingEvent(const std::string &, int start, int len) override;
|
|
bool AcceptsTextInput() override;
|
|
AABB2 GetTextInputRect() override;
|
|
bool NeedsAbsoluteMouseCoordinate() override;
|
|
bool ExecCommand(const Handle<gui::ConsoleCommand> &) override;
|
|
Handle<gui::ConsoleCommandCandidateIterator>
|
|
AutocompleteCommandName(const std::string &name) override;
|
|
|
|
void SetWorld(World *);
|
|
World *GetWorld() const { return world.get(); }
|
|
void AddLocalEntity(std::unique_ptr<ILocalEntity> &&ent) {
|
|
localEntities.emplace_back(std::move(ent));
|
|
}
|
|
|
|
void MarkWorldUpdate();
|
|
|
|
IRenderer &GetRenderer() { return *renderer; }
|
|
SceneDefinition GetLastSceneDef() { return lastSceneDef; }
|
|
IAudioDevice &GetAudioDevice() { return *audioDevice; }
|
|
|
|
bool WantsToBeClosed() override;
|
|
bool IsMuted();
|
|
|
|
void PlayerSentChatMessage(Player &, bool global, const std::string &);
|
|
void ServerSentMessage(const std::string &);
|
|
|
|
void PlayerCapturedIntel(Player &);
|
|
void PlayerCreatedBlock(Player &);
|
|
void PlayerPickedIntel(Player &);
|
|
void PlayerDropIntel(Player &);
|
|
void TeamCapturedTerritory(int teamId, int territoryId);
|
|
void TeamWon(int);
|
|
void JoinedGame();
|
|
void LocalPlayerCreated();
|
|
void PlayerDestroyedBlockWithWeaponOrTool(IntVector3);
|
|
void PlayerDiggedBlock(IntVector3);
|
|
void GrenadeDestroyedBlock(IntVector3);
|
|
void PlayerLeaving(Player &);
|
|
void PlayerJoinedTeam(Player &);
|
|
void PlayerSpawned(Player &);
|
|
|
|
// IWorldListener begin
|
|
void PlayerObjectSet(int) override;
|
|
void PlayerMadeFootstep(Player &) override;
|
|
void PlayerJumped(Player &) override;
|
|
void PlayerLanded(Player &, bool hurt) override;
|
|
void PlayerFiredWeapon(Player &) override;
|
|
void PlayerDryFiredWeapon(Player &) override;
|
|
void PlayerReloadingWeapon(Player &) override;
|
|
void PlayerReloadedWeapon(Player &) override;
|
|
void PlayerChangedTool(Player &) override;
|
|
void PlayerThrewGrenade(Player &, stmp::optional<const Grenade &>) override;
|
|
void PlayerMissedSpade(Player &) override;
|
|
void PlayerRestocked(Player &) override;
|
|
|
|
/** @deprecated use BulletHitPlayer */
|
|
void PlayerHitBlockWithSpade(Player &, Vector3 hitPos, IntVector3 blockPos,
|
|
IntVector3 normal) override;
|
|
void PlayerKilledPlayer(Player &killer, Player &victim, KillType) override;
|
|
|
|
void BulletHitPlayer(Player &hurtPlayer, HitType, Vector3 hitPos, Player &by,
|
|
std::unique_ptr<IBulletHitScanState> &stateCell) override;
|
|
void BulletHitBlock(Vector3, IntVector3 blockPos, IntVector3 normal) override;
|
|
void AddBulletTracer(Player &player, Vector3 muzzlePos, Vector3 hitPos) override;
|
|
void GrenadeExploded(const Grenade &) override;
|
|
void GrenadeBounced(const Grenade &) override;
|
|
void GrenadeDroppedIntoWater(const Grenade &) override;
|
|
|
|
void BlocksFell(std::vector<IntVector3>) override;
|
|
|
|
void LocalPlayerPulledGrenadePin() override;
|
|
void LocalPlayerBlockAction(IntVector3, BlockActionType type) override;
|
|
void LocalPlayerCreatedLineBlock(IntVector3, IntVector3) override;
|
|
void LocalPlayerHurt(HurtType type, bool sourceGiven, Vector3 source) override;
|
|
void LocalPlayerBuildError(BuildFailureReason reason) override;
|
|
// IWorldListener end
|
|
};
|
|
} // namespace client
|
|
} // namespace spades
|