Files
OldThink/Content.Shared/Ghost/SharedGhostSystem.cs

289 lines
8.9 KiB
C#
Raw Permalink Normal View History

using Content.Shared._White.Antag;
using Content.Shared.Emoting;
using Content.Shared.Hands;
using Content.Shared.Interaction.Events;
using Content.Shared.Item;
2023-09-24 13:34:08 -07:00
using Content.Shared.Popups;
using Content.Shared.Roles;
using Robust.Shared.Serialization;
namespace Content.Shared.Ghost
{
2021-12-03 01:58:03 -08:00
/// <summary>
2023-08-25 18:50:46 +10:00
/// System for the <see cref="GhostComponent"/>.
/// Prevents ghosts from interacting when <see cref="GhostComponent.CanGhostInteract"/> is false.
2021-12-03 01:58:03 -08:00
/// </summary>
public abstract class SharedGhostSystem : EntitySystem
{
2023-09-24 13:34:08 -07:00
[Dependency] protected readonly SharedPopupSystem Popup = default!;
public override void Initialize()
{
base.Initialize();
2023-08-25 18:50:46 +10:00
SubscribeLocalEvent<GhostComponent, UseAttemptEvent>(OnAttempt);
SubscribeLocalEvent<GhostComponent, InteractionAttemptEvent>(OnAttempt);
SubscribeLocalEvent<GhostComponent, EmoteAttemptEvent>(OnAttempt);
SubscribeLocalEvent<GhostComponent, DropAttemptEvent>(OnAttempt);
SubscribeLocalEvent<GhostComponent, PickupAttemptEvent>(OnAttempt);
}
2023-08-25 18:50:46 +10:00
private void OnAttempt(EntityUid uid, GhostComponent component, CancellableEntityEventArgs args)
{
if (!component.CanGhostInteract)
args.Cancel();
}
2023-08-25 18:50:46 +10:00
public void SetTimeOfDeath(EntityUid uid, TimeSpan value, GhostComponent? component)
{
if (!Resolve(uid, ref component))
return;
component.TimeOfDeath = value;
}
public void SetCanReturnToBody(EntityUid uid, bool value, GhostComponent? component = null)
2023-06-21 13:04:07 +12:00
{
if (!Resolve(uid, ref component))
return;
component.CanReturnToBody = value;
}
2023-08-25 18:50:46 +10:00
public void SetCanReturnToBody(GhostComponent component, bool value)
{
2021-12-03 01:58:03 -08:00
component.CanReturnToBody = value;
}
2025-03-21 02:20:37 +05:00
public virtual void SetVisible(Entity<GhostComponent?> ghost, bool visible)
{
if (!Resolve(ghost.Owner, ref ghost.Comp))
return;
ghost.Comp.Visible = visible;
Dirty(ghost);
}
}
2021-12-03 01:58:03 -08:00
/// <summary>
/// A client to server request to get places a ghost can warp to.
/// Response is sent via <see cref="GhostWarpsResponseEvent"/>
/// </summary>
[Serializable, NetSerializable]
public sealed class GhostWarpsRequestEvent : EntityEventArgs
{
}
/// <summary>
/// An player body a ghost can warp to.
/// This is used as part of <see cref="GhostWarpsResponseEvent"/>
/// </summary>
[Serializable, NetSerializable]
public struct GhostWarpPlayer
{
public GhostWarpPlayer(NetEntity entity, string playerName, string playerJobName, string playerDepartmentID, bool isGhost, bool isLeft, bool isDead, bool isAlive)
{
Entity = entity;
Name = playerName;
JobName = playerJobName;
DepartmentID = playerDepartmentID;
IsGhost = isGhost;
IsLeft = isLeft;
IsDead = isDead;
IsAlive = isAlive;
}
/// <summary>
/// The entity representing the warp point.
/// This is passed back to the server in <see cref="GhostWarpToTargetRequestEvent"/>
/// </summary>
public NetEntity Entity { get; }
/// <summary>
/// The display player name to be surfaced in the ghost warps menu
/// </summary>
public string Name { get; }
/// <summary>
/// The display player job to be surfaced in the ghost warps menu
/// </summary>
public string JobName { get; }
/// <summary>
/// The display player department to be surfaced in the ghost warps menu
/// </summary>
public string DepartmentID { get; set; }
/// <summary>
/// Is player is ghost
/// </summary>
public bool IsGhost { get; }
/// <summary>
/// Is player body alive
/// </summary>
public bool IsAlive { get; }
/// <summary>
/// Is player body dead
/// </summary>
public bool IsDead { get; }
/// <summary>
/// Is player left from body
/// </summary>
public bool IsLeft { get; }
}
[Serializable, NetSerializable]
public struct GhostWarpGlobalAntagonist
{
public GhostWarpGlobalAntagonist(NetEntity entity, string playerName, string antagonistName, string antagonistDescription, string prototypeID)
{
Entity = entity;
Name = playerName;
AntagonistName = antagonistName;
AntagonistDescription = antagonistDescription;
PrototypeID = prototypeID;
}
/// <summary>
/// The entity representing the warp point.
/// This is passed back to the server in <see cref="GhostWarpToTargetRequestEvent"/>
/// </summary>
public NetEntity Entity { get; }
/// <summary>
/// The display player name to be surfaced in the ghost warps menu
/// </summary>
public string Name { get; }
/// <summary>
/// The display antagonist name to be surfaced in the ghost warps menu
/// </summary>
public string AntagonistName { get; }
/// <summary>
/// The display antagonist description to be surfaced in the ghost warps menu
/// </summary>
public string AntagonistDescription { get; }
/// <summary>
/// A antagonist prototype id
/// </summary>
public string PrototypeID { get; }
}
2021-12-03 01:58:03 -08:00
/// <summary>
/// An individual place a ghost can warp to.
/// This is used as part of <see cref="GhostWarpsResponseEvent"/>
2021-12-03 01:58:03 -08:00
/// </summary>
[Serializable, NetSerializable]
public struct GhostWarpPlace
{
public GhostWarpPlace(NetEntity entity, string name, string description)
{
Entity = entity;
Name = name;
Description = description;
}
2021-12-03 01:58:03 -08:00
/// <summary>
/// The entity representing the warp point.
/// This is passed back to the server in <see cref="GhostWarpToTargetRequestEvent"/>
2021-12-03 01:58:03 -08:00
/// </summary>
public NetEntity Entity { get; }
/// <summary>
/// The display name to be surfaced in the ghost warps menu
/// </summary>
public string Name { get; }
2021-12-03 01:58:03 -08:00
/// <summary>
/// The display name to be surfaced in the ghost warps menu
2021-12-03 01:58:03 -08:00
/// </summary>
public string Description { get; }
}
2021-12-03 01:58:03 -08:00
/// <summary>
/// A server to client response for a <see cref="GhostWarpsRequestEvent"/>.
/// Contains players, and locations a ghost can warp to
2021-12-03 01:58:03 -08:00
/// </summary>
[Serializable, NetSerializable]
public sealed class GhostWarpsResponseEvent : EntityEventArgs // WD edit
{
public GhostWarpsResponseEvent(List<GhostWarpPlayer> players, List<GhostWarpPlace> places, List<GhostWarpGlobalAntagonist> antagonists)
{
Players = players;
Places = places;
Antagonists = antagonists;
}
/// <summary>
/// A list of players to teleport.
/// </summary>
public List<GhostWarpPlayer> Players { get; }
/// <summary>
/// A list of warp points.
/// </summary>
public List<GhostWarpPlace> Places { get; }
/// <summary>
/// A list of antagonists to teleport.
/// </summary>
public List<GhostWarpGlobalAntagonist> Antagonists { get; }
}
2021-12-03 01:58:03 -08:00
/// <summary>
/// A client to server request for their ghost to be warped to an entity
/// </summary>
[Serializable, NetSerializable]
public sealed class GhostWarpToTargetRequestEvent : EntityEventArgs
{
public NetEntity Target { get; }
public GhostWarpToTargetRequestEvent(NetEntity target)
{
Target = target;
}
}
Cherrypicks 3 (#382) * Mobs burn to ashes on excessive heat damage (#26971) * mobs burn to ashes on excessive heat damage * remove comment, remove random lines I didn't mean to add * combine code into behavior * clean unused * fix namespace * drop next to * fix spawn entities behavior spawning entities outside container * fix burning to ash not working on all mobs (#27158) * add ghostnado button to warp menu (#27556) * add ghostnado button to warp menu * translator ops --------- Co-authored-by: deltanedas <@deltanedas:kde.org> * Make arguments and parameters wrap to one variable per line (#27766) * Fix ghosts getting spawned in nullspace (#27617) * Add tests for ghost spawn position * Make ghosts spawn immediately * Format mind system * Move ghost spawning to GhostSystem * Spawn ghost on grid or map This fixes the ghosts being attached the parent entity instead of the grid. * Move logging out of the ghost system * Make round start observer spawn using GhostSystem * Move GameTicker ghost spawning to GhostSystem Moved the more robust character name selection code over. Moved the TimeOfDeath code over. Added canReturn logic. * Add overrides and default for ghost spawn coordinates * Add warning log to ghost spawn fail * Clean up test * Dont spawn ghost on map delete * Minor changes to the role test * Fix role test failing to spawn ghost It was failing the map check due to using Nullspace * Fix ghost tests when running in parallel Not sure what happened, but it seems to be because they were running simultaneously and overwriting values. * Clean up ghost tests * Test that map deletion does not spawn ghosts * Spawn ghost on the next available map * Disallow spawning on deleted maps * Fix map deletion ghost test * Cleanup --------- Co-authored-by: Whisper <121047731+QuietlyWhisper@users.noreply.github.com> Co-authored-by: deltanedas <39013340+deltanedas@users.noreply.github.com> Co-authored-by: ShadowCommander <shadowjjt@gmail.com>
2024-06-25 21:39:44 +00:00
/// <summary>
/// A client to server request for their ghost to be warped to the most followed entity.
/// </summary>
[Serializable, NetSerializable]
public sealed class GhostnadoRequestEvent : EntityEventArgs;
2021-12-03 01:58:03 -08:00
/// <summary>
/// A client to server request for their ghost to return to body
/// </summary>
[Serializable, NetSerializable]
public sealed class GhostReturnToBodyRequest : EntityEventArgs
{
}
2021-12-03 01:58:03 -08:00
/// <summary>
/// A server to client update with the available ghost role count
/// </summary>
[Serializable, NetSerializable]
public sealed class GhostUpdateGhostRoleCountEvent : EntityEventArgs
{
public int AvailableGhostRoles { get; }
2021-12-03 01:58:03 -08:00
public GhostUpdateGhostRoleCountEvent(int availableGhostRoleCount)
{
2021-12-03 01:58:03 -08:00
AvailableGhostRoles = availableGhostRoleCount;
}
}
2023-04-28 07:13:28 +06:00
[Serializable, NetSerializable]
public sealed class GhostReturnToRoundRequest : EntityEventArgs
{
}
}