using Content.Shared.Mind.Components;
using Robust.Shared.Containers;
using Robust.Shared.GameStates;
using Robust.Shared.Network;
using Robust.Shared.Prototypes;
namespace Content.Shared.Mind;
///
/// This component stores information about a player/mob mind. The component will be attached to a mind-entity
/// which is stored in null-space. The entity that is currently "possessed" by the mind will have a
/// .
///
///
/// Roles are attached as components on the mind-entity entity.
/// Think of it like this: if a player is supposed to have their memories,
/// their mind follows along.
///
/// Things such as respawning do not follow, because you're a new character.
/// Getting borged, cloned, turned into a catbeast, etc... will keep it following you.
///
/// Minds are stored in null-space, and are thus generally not set to players unless that player is the owner
/// of the mind. As a result it should be safe to network "secret" information like roles & objectives
///
[RegisterComponent, NetworkedComponent, AutoGenerateComponentState(true)]
public sealed partial class MindComponent : Component
{
[DataField, AutoNetworkedField]
public List Objectives = new();
///
/// The session ID of the player owning this mind.
///
[DataField, AutoNetworkedField, Access(typeof(SharedMindSystem))]
public NetUserId? UserId { get; set; }
///
/// The session ID of the original owner, if any.
/// May end up used for round-end information (as the owner may have abandoned Mind since)
///
[DataField, AutoNetworkedField, Access(typeof(SharedMindSystem))]
public NetUserId? OriginalOwnerUserId { get; set; }
///
/// The first entity that this mind controlled. Used for round end information.
/// Might be relevant if the player has ghosted since.
///
[AutoNetworkedField]
public NetEntity? OriginalOwnedEntity; // TODO WeakEntityReference make this a Datafield again
// This is a net entity, because this field currently does not get set to null when this entity is deleted.
// This is a lazy way to ensure that people check that the entity still exists.
// TODO MIND Fix this properly by adding an OriginalMindContainerComponent or something like that.
[ViewVariables]
public bool IsVisitingEntity => VisitingEntity != null;
///
/// The entity that this mind may be currently visiting. Used, for example, to allow admin ghosting to not make the owner's body catatonic, as opposed to when normally ghosting.
///
[DataField, AutoNetworkedField, Access(typeof(SharedMindSystem))]
public EntityUid? VisitingEntity { get; set; }
[ViewVariables]
public EntityUid? CurrentEntity => VisitingEntity ?? OwnedEntity;
[DataField, AutoNetworkedField, ViewVariables(VVAccess.ReadWrite)]
public string? CharacterName { get; set; }
///
/// The time of death for this Mind.
/// Can be null - will be null if the Mind is not considered "dead".
///
[DataField]
public TimeSpan? TimeOfDeath { get; set; }
///
/// The entity currently owned by this mind.
/// Can be null.
///
[DataField, AutoNetworkedField, Access(typeof(SharedMindSystem))]
public EntityUid? OwnedEntity { get; set; }
///
/// An enumerable over all the objective entities this mind has.
///
[ViewVariables, Obsolete("Use Objectives field")]
public IEnumerable AllObjectives => Objectives;
///
/// Prevents user from ghosting out
///
[DataField]
public bool PreventGhosting { get; set; }
///
/// Prevents user from suiciding
///
[DataField]
public bool PreventSuicide { get; set; }
///
/// Mind Role Entities belonging to this Mind are stored in this container.
///
[ViewVariables]
public Container MindRoleContainer = default!;
///
/// The id for the MindRoleContainer.
///
[ViewVariables]
public const string MindRoleContainerId = "mind_roles";
///
/// The mind's current antagonist/special role, or lack thereof;
///
[DataField, AutoNetworkedField]
public ProtoId RoleType = "Neutral";
///
/// The role's subtype, shown only to admins to help with antag categorization
///
[DataField]
public LocId? Subtype;
}