|
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538 |
- using System;
- using Gamecraft.Wires;
- using RobocraftX.Character;
- using RobocraftX.Character.Movement;
- using Unity.Mathematics;
- using RobocraftX.Common;
- using RobocraftX.Common.Players;
- using RobocraftX.GUI.Wires;
- using RobocraftX.Physics;
- using Svelto.ECS;
- using Techblox.BuildingDrone;
- using Techblox.Camera;
- using TechbloxModdingAPI.Blocks;
- using TechbloxModdingAPI.Players;
- using TechbloxModdingAPI.Utility;
- using UnityEngine;
-
- namespace TechbloxModdingAPI
- {
- /// <summary>
- /// An in-game player character. Any Leo you see is a player.
- /// </summary>
- public partial class Player : EcsObjectBase, IEquatable<Player>, IEquatable<EGID>
- {
- // static functionality
- private static PlayerEngine playerEngine = new PlayerEngine();
- private static PlayerEventsEngine playerEventsEngine = new PlayerEventsEngine();
- private static Player localPlayer;
-
- /// <summary>
- /// Checks if the specified player exists.
- /// </summary>
- /// <returns>Whether the player exists.</returns>
- /// <param name="player">Player type.</param>
- public static bool Exists(PlayerType player)
- {
- switch (player)
- {
- case PlayerType.Remote:
- return playerEngine.GetRemotePlayer() != uint.MaxValue;
- case PlayerType.Local:
- return playerEngine.GetLocalPlayer() != uint.MaxValue;
- }
- return false;
- }
-
- /// <summary>
- /// Checks if the specified player exists.
- /// </summary>
- /// <returns>Whether the player exists.</returns>
- /// <param name="player">The player's unique identifier.</param>
- public static bool Exists(uint player)
- {
- return playerEngine.ExistsById(player);
- }
-
- /// <summary>
- /// The amount of Players in the current game.
- /// </summary>
- /// <returns>The count.</returns>
- public static uint Count()
- {
- return (uint) playerEngine.GetAllPlayerCount();
- }
-
- /// <summary>
- /// Returns the current player belonging to this client.
- /// </summary>
- public static Player LocalPlayer
- {
- get
- {
- if (localPlayer == null || localPlayer.Id != playerEngine.GetLocalPlayer())
- localPlayer = new Player(PlayerType.Local);
- return localPlayer;
- }
- }
-
- /// <summary>
- /// Initializes a new instance of the <see cref="T:TechbloxModdingAPI.Player"/> class.
- /// </summary>
- /// <param name="id">The player's unique identifier.</param>
- public Player(uint id) : base(new EGID(id, CharacterExclusiveGroups.OnFootGroup))
- {
- this.Id = id;
- if (!Exists(id))
- {
- throw new PlayerNotFoundException($"No player with id {id} exists");
- }
- this.Type = playerEngine.GetLocalPlayer() == id ? PlayerType.Local : PlayerType.Remote;
- }
-
- /// <summary>
- /// Initializes a new instance of the <see cref="T:TechbloxModdingAPI.Player"/> class.
- /// </summary>
- /// <param name="player">The player type. Chooses the first available player matching the criteria.</param>
- public Player(PlayerType player) : base(ecs =>
- {
- uint id;
- switch (player)
- {
- case PlayerType.Local:
- id = playerEngine.GetLocalPlayer();
- break;
- case PlayerType.Remote:
- id = playerEngine.GetRemotePlayer();
- break;
- default:
- id = uint.MaxValue;
- break;
- }
-
- if (id == uint.MaxValue)
- {
- throw new PlayerNotFoundException($"No player of {player} type exists");
- }
-
- return new EGID(id, CharacterExclusiveGroups.OnFootGroup);
- })
- {
- this.Type = player;
- }
-
- // object fields & properties
-
- /// <summary>
- /// The player's type.
- /// The player type is always relative to the current client, not the game host.
- /// </summary>
- /// <value>The enumerated player type.</value>
- public PlayerType Type { get; }
-
- /// <summary>
- /// The player's unique identifier.
- /// </summary>
- /// <value>The identifier.</value>
- public new uint Id { get; }
-
- /// <summary>
- /// The player's current position.
- /// </summary>
- /// <value>The position.</value>
- public float3 Position
- {
- get => playerEngine.GetCharacterStruct<RigidBodyEntityStruct>(Id).Get().position;
- set => playerEngine.SetLocation(Id, value, false);
- }
-
- /// <summary>
- /// The player's current rotation.
- /// </summary>
- /// <value>The rotation.</value>
- public float3 Rotation
- {
- get => ((Quaternion) (GameState.IsBuildMode()
- ? playerEngine.GetCameraStruct<CameraEntityStruct>(Id).Get().rotation
- : playerEngine.GetCharacterStruct<RigidBodyEntityStruct>(Id).Get().rotation)).eulerAngles;
- set => _ = GameState.IsBuildMode()
- ? playerEngine.GetCameraStruct<CameraEntityStruct>(Id).Get().rotation = quaternion.Euler(value)
- : playerEngine.GetCharacterStruct<RigidBodyEntityStruct>(Id).Get().rotation = quaternion.Euler(value);
- }
-
- /// <summary>
- /// The player's current velocity.
- /// </summary>
- /// <value>The velocity.</value>
- public float3 Velocity
- {
- get => playerEngine.GetCharacterStruct<RigidBodyEntityStruct>(Id).Get().velocity;
- set => playerEngine.GetCharacterStruct<RigidBodyEntityStruct>(Id).Get().velocity = value;
- }
-
- /// <summary>
- /// The player's current angular velocity.
- /// </summary>
- /// <value>The angular velocity.</value>
- public float3 AngularVelocity
- {
- get => playerEngine.GetCharacterStruct<RigidBodyEntityStruct>(Id).Get().angularVelocity;
- set => playerEngine.GetCharacterStruct<RigidBodyEntityStruct>(Id).Get().angularVelocity = value;
- }
-
- /// <summary>
- /// The player's mass.
- /// </summary>
- /// <value>The mass.</value>
- public float Mass =>
- 1f / playerEngine.GetCharacterStruct<RigidBodyEntityStruct>(Id).Get().physicsMass.InverseMass;
-
- private float _ping = -1f;
-
- /// <summary>
- /// The player's latest network ping time.
- /// </summary>
- /// <value>The ping (s).</value>
- public float Ping
- {
- get
- {
- var opt = playerEngine.GetPlayerStruct<PlayerNetworkStatsEntityStruct>(Id, Type);
- if (opt)
- {
- _ping = opt.Get().lastPingTimeSinceLevelLoad ?? _ping;
- }
- return _ping;
- }
- }
-
- /// <summary>
- /// The player's initial health when entering Simulation (aka Time Running) mode.
- /// </summary>
- /// <value>The initial health.</value>
- public float InitialHealth
- {
- get
- {
- var opt = playerEngine.GetCharacterStruct<CharacterHealthEntityStruct>(Id);
- return opt ? opt.Get().initialHealth : -1f;
- }
-
- set => playerEngine.GetCharacterStruct<CharacterHealthEntityStruct>(Id).Get().initialHealth = value;
- }
-
- /// <summary>
- /// The player's current health in Simulation (aka Time Running) mode.
- /// </summary>
- /// <value>The current health.</value>
- public float CurrentHealth
- {
- get
- {
- var opt = playerEngine.GetCharacterStruct<CharacterHealthEntityStruct>(Id);
- return opt ? opt.Get().currentHealth : -1f;
- }
-
- set => playerEngine.GetCharacterStruct<CharacterHealthEntityStruct>(Id).Get().currentHealth = value;
- }
-
- /// <summary>
- /// Whether this <see cref="T:TechbloxModdingAPI.Player"/> is damageable.
- /// </summary>
- /// <value><c>true</c> if damageable; otherwise, <c>false</c>.</value>
- public bool Damageable
- {
- get
- {
- var opt = playerEngine.GetCharacterStruct<CharacterHealthEntityStruct>(Id);
- return opt.Get().canTakeDamageStat;
- }
-
- set
- {
- ref var healthStruct = ref playerEngine.GetCharacterStruct<CharacterHealthEntityStruct>(Id).Get();
- healthStruct.canTakeDamage = value;
- healthStruct.canTakeDamageStat = value;
- }
- }
-
- /// <summary>
- /// The player's lives when initially entering Simulation (aka Time Running) mode.
- /// </summary>
- /// <value>The initial lives.</value>
- public uint InitialLives
- {
- get
- {
- var opt = playerEngine.GetCharacterStruct<CharacterLivesEntityComponent>(Id);
- return opt ? opt.Get().initialLives : uint.MaxValue;
- }
-
- set => playerEngine.GetCharacterStruct<CharacterLivesEntityComponent>(Id).Get().initialLives = value;
- }
-
- /// <summary>
- /// The player's current lives in Simulation (aka Time Running) mode.
- /// </summary>
- /// <value>The current lives.</value>
- public uint CurrentLives
- {
- get
- {
- var opt = playerEngine.GetCharacterStruct<CharacterLivesEntityComponent>(Id);
- return opt ? opt.Get().currentLives : uint.MaxValue;
- }
-
- set => playerEngine.GetCharacterStruct<CharacterLivesEntityComponent>(Id).Get().currentLives = value;
- }
-
- /*/// <summary>
- /// Whether the Game Over screen is displayed for the player.
- /// </summary>
- /// <value><c>true</c> if game over; otherwise, <c>false</c>.</value>
- public bool GameOver
- {
- get => playerEngine.GetGameOverScreen(Id);
- }*/
-
- /// <summary>
- /// Whether the player is dead.
- /// If <c>true</c>, hopefully it was quick.
- /// </summary>
- /// <value><c>true</c> if dead; otherwise, <c>false</c>.</value>
- public bool Dead
- {
- get => playerEngine.IsDead(Id);
- }
-
- /// <summary>
- /// The player's selected block ID in their hand.
- /// </summary>
- /// <value>The selected block.</value>
- public BlockIDs SelectedBlock
- {
- get
- {
- var optstruct = playerEngine.GetCharacterStruct<EquippedPartStruct>(Id);
- return optstruct ? (BlockIDs) optstruct.Get().selectedDBPartID : BlockIDs.Invalid;
- }
- }
-
- /// <summary>
- /// The player's selected block color in their hand.
- /// </summary>
- /// <value>The selected block's color.</value>
- public BlockColor SelectedColor
- {
- get
- {
- var optstruct = playerEngine.GetCharacterStruct<EquippedColourStruct>(Id);
- return optstruct ? new BlockColor(optstruct.Get().indexInPalette) : BlockColors.Default;
- }
- }
-
- /// <summary>
- /// The player's selected block colour in their hand.
- /// </summary>
- /// <value>The selected block's colour.</value>
- public BlockColor SelectedColour
- {
- get
- {
- var optstruct = playerEngine.GetCharacterStruct<EquippedColourStruct>(Id);
- return optstruct ? new BlockColor(optstruct.Get().indexInPalette) : BlockColors.Default;
- }
- }
-
- /// <summary>
- /// The player's selected blueprint in their hand. Set to null to clear. Dispose after usage.
- /// </summary>
- public Blueprint SelectedBlueprint
- {
- get
- {
- var lbiso = playerEngine.GetPlayerStruct<LocalBlueprintInputStruct>(Id, Type);
- return lbiso ? new Blueprint(lbiso.Get().selectedBlueprintId) : null;
- }
- set => BlockGroup._engine.SelectBlueprint(value?.Id ?? uint.MaxValue);
- }
-
- /// <summary>
- /// The player's mode in time stopped mode, determining what they place.
- /// </summary>
- public PlayerBuildingMode BuildingMode => (PlayerBuildingMode)Math.Log((double)playerEngine
- .GetCharacterStruct<TimeStoppedModeComponent>(Id).Get().timeStoppedContext, 2); // It's a bit field in game now
-
- /// <summary>
- /// Whether the player is sprinting.
- /// </summary>
- public bool Sprinting
- {
- get => GameState.IsBuildMode()
- ? playerEngine.GetCharacterStruct<BuildingDroneMovementComponent>(Id).Get().sprinting
- : playerEngine.GetCharacterStruct<CharacterMovementEntityStruct>(Id).Get().isSprinting;
- set => _ = GameState.IsBuildMode()
- ? playerEngine.GetCharacterStruct<BuildingDroneMovementComponent>(Id).Get().sprinting = value
- : playerEngine.GetCharacterStruct<CharacterMovementEntityStruct>(Id).Get().isSprinting = value;
- }
-
- /// <summary>
- /// Movement speed setting. Build mode (camera) and simulation mode settings are separate.
- /// </summary>
- public float SpeedSetting
- {
- get => GameState.IsBuildMode()
- ? playerEngine.GetCharacterStruct<BuildingDroneMovementSettingsComponent>(Id).Get().speed
- : playerEngine.GetCharacterStruct<CharacterMovementSettingsEntityStruct>(Id).Get().moveSpeed;
- set => _ = GameState.IsBuildMode()
- ? playerEngine.GetCharacterStruct<BuildingDroneMovementSettingsComponent>(Id).Get().speed = value
- : playerEngine.GetCharacterStruct<CharacterMovementSettingsEntityStruct>(Id).Get().moveSpeed = value;
- }
-
- /// <summary>
- /// The multiplier setting to use when sprinting. Build mode (camera) and simulation mode settings are separate.
- /// </summary>
- public float SpeedSprintMultiplierSetting
- {
- get => GameState.IsBuildMode()
- ? playerEngine.GetCharacterStruct<BuildingDroneMovementSettingsComponent>(Id).Get().speedSprintMultiplier
- : playerEngine.GetCharacterStruct<CharacterMovementSettingsEntityStruct>(Id).Get().sprintSpeedMultiplier;
- set => _ = GameState.IsBuildMode()
- ? playerEngine.GetCharacterStruct<BuildingDroneMovementSettingsComponent>(Id).Get().speedSprintMultiplier = value
- : playerEngine.GetCharacterStruct<CharacterMovementSettingsEntityStruct>(Id).Get().sprintSpeedMultiplier = value;
- }
-
- /// <summary>
- /// The acceleration setting of the player. Build mode (camera) and simulation mode settings are separate.
- /// </summary>
- public float AccelerationSetting
- {
- get => GameState.IsBuildMode()
- ? playerEngine.GetCharacterStruct<BuildingDroneMovementSettingsComponent>(Id).Get().acceleration
- : playerEngine.GetCharacterStruct<CharacterMovementSettingsEntityStruct>(Id).Get().acceleration;
- set => _ = GameState.IsBuildMode()
- ? playerEngine.GetCharacterStruct<BuildingDroneMovementSettingsComponent>(Id).Get().acceleration = value
- : playerEngine.GetCharacterStruct<CharacterMovementSettingsEntityStruct>(Id).Get().acceleration = value;
- }
-
- // object methods
-
- /// <summary>
- /// Teleport the player to the specified coordinates.
- /// </summary>
- /// <param name="x">The x coordinate.</param>
- /// <param name="y">The y coordinate.</param>
- /// <param name="z">The z coordinate.</param>
- /// <param name="relative">If set to <c>true</c> teleport relative to the player's current position.</param>
- /// <param name="exitSeat">If set to <c>true</c> exit any seat the player is in.</param>
- public void Teleport(float x, float y, float z, bool relative = true, bool exitSeat = true)
- {
- float3 location = new float3(x, y, z);
- if (relative)
- {
- location += Position;
- }
- playerEngine.SetLocation(Id, location, exitSeat: exitSeat);
- }
-
- public void EnterSeat(Seat seat)
- {
- playerEngine.EnterSeat(Id, seat.Id);
- }
-
- public void ExitSeat()
- {
- playerEngine.ExitSeat(Id);
- }
-
- /// <summary>
- /// Returns the block the player is currently looking at in build mode.
- /// </summary>
- /// <param name="maxDistance">The maximum distance from the player (default is the player's building reach)</param>
- /// <returns>The block or null if not found</returns>
- public Block GetBlockLookedAt(float maxDistance = -1f)
- {
- var egid = playerEngine.GetThingLookedAt(Id, maxDistance);
- return egid != default && egid.groupID != CommonExclusiveGroups.SIMULATION_BODIES_GROUP
- && egid.groupID != WiresGUIExclusiveGroups.WireGroup
- ? Block.New(egid)
- : null;
- }
-
- /// <summary>
- /// Returns the rigid body the player is currently looking at during simulation.
- /// </summary>
- /// <param name="maxDistance">The maximum distance from the player (default is the player's building reach)</param>
- /// <returns>The body or null if not found</returns>
- public SimBody GetSimBodyLookedAt(float maxDistance = -1f)
- {
- var egid = playerEngine.GetThingLookedAt(Id, maxDistance);
- return egid != default && egid.groupID == CommonExclusiveGroups.SIMULATION_BODIES_GROUP
- ? EcsObjectBase.GetInstance(egid, e => new SimBody(e))
- : null;
- }
-
- /// <summary>
- /// Returns the wire the player is currently looking at in build mode.
- /// </summary>
- /// <param name="maxDistance">The maximum distance from the player (default is the player's building reach)</param>
- /// <returns>The wire or null if not found</returns>
- public Wire GetWireLookedAt(float maxDistance = -1f)
- {
- var egid = playerEngine.GetThingLookedAt(Id, maxDistance);
- return egid != default && egid.groupID == WiresGUIExclusiveGroups.WireGroup
- ? EcsObjectBase.GetInstance(new EGID(egid.entityID, NamedExclusiveGroup<WiresGroup>.Group),
- e => new Wire(e))
- : null;
- }
-
- /// <summary>
- /// Returns the blocks that are in the player's current selection.
- /// </summary>
- /// <returns>An array of blocks or an empty array</returns>
- public Block[] GetSelectedBlocks()
- {
- return playerEngine.GetSelectedBlocks(Id);
- }
-
- public bool Equals(Player other)
- {
- if (ReferenceEquals(null, other)) return false;
- if (ReferenceEquals(this, other)) return true;
- return Id == other.Id;
- }
-
- public bool Equals(EGID other)
- {
- return Id == other.entityID && other.groupID == (Type == PlayerType.Local
- ? PlayersExclusiveGroups.LocalPlayers
- : PlayersExclusiveGroups.RemotePlayers);
- }
-
- public override bool Equals(object obj)
- {
- if (ReferenceEquals(null, obj)) return false;
- if (ReferenceEquals(this, obj)) return true;
- if (obj.GetType() != this.GetType()) return false;
- return Equals((Player) obj);
- }
-
- public override int GetHashCode()
- {
- return (int) Id;
- }
-
- public override string ToString()
- {
- return $"{nameof(Type)}: {Type}, {nameof(Id)}: {Id}, {nameof(Position)}: {Position}, {nameof(Rotation)}: {Rotation}, {nameof(Mass)}: {Mass}";
- }
-
- // internal methods
-
- internal static void Init()
- {
- Utility.GameEngineManager.AddGameEngine(playerEngine);
- Utility.GameEngineManager.AddGameEngine(playerEventsEngine);
- }
- }
- }
|