using Content.Server.Access.Systems; using Content.Server.Administration.Logs; using Content.Shared.CharacterAppearance.Components; using Content.Shared.Database; using Content.Shared.Hands; using Content.Shared.IdentityManagement; using Content.Shared.IdentityManagement.Components; using Content.Shared.Inventory; using Content.Shared.Inventory.Events; using Content.Shared.Preferences; using Robust.Shared.Enums; using Robust.Shared.GameObjects.Components.Localization; namespace Content.Server.IdentityManagement; /// /// Responsible for updating the identity of an entity on init or clothing equip/unequip. /// public class IdentitySystem : SharedIdentitySystem { [Dependency] private readonly IdCardSystem _idCard = default!; [Dependency] private readonly IAdminLogManager _adminLog = default!; private HashSet _queuedIdentityUpdates = new(); public override void Initialize() { base.Initialize(); SubscribeLocalEvent((uid, _, _) => QueueIdentityUpdate(uid)); SubscribeLocalEvent((uid, _, _) => QueueIdentityUpdate(uid)); SubscribeLocalEvent((uid, _, _) => QueueIdentityUpdate(uid)); SubscribeLocalEvent((uid, _, _) => QueueIdentityUpdate(uid)); } public override void Update(float frameTime) { base.Update(frameTime); foreach (var ent in _queuedIdentityUpdates) { if (!TryComp(ent, out var identity)) continue; UpdateIdentityInfo(ent, identity); } _queuedIdentityUpdates.Clear(); } // This is where the magic happens protected override void OnComponentInit(EntityUid uid, IdentityComponent component, ComponentInit args) { base.OnComponentInit(uid, component, args); var ident = Spawn(null, Transform(uid).Coordinates); QueueIdentityUpdate(uid); component.IdentityEntitySlot.Insert(ident); } /// /// Queues an identity update to the start of the next tick. /// public void QueueIdentityUpdate(EntityUid uid) { _queuedIdentityUpdates.Add(uid); } #region Private API /// /// Updates the metadata name for the id(entity) from the current state of the character. /// private void UpdateIdentityInfo(EntityUid uid, IdentityComponent identity) { if (identity.IdentityEntitySlot.ContainedEntity is not { } ident) return; var representation = GetIdentityRepresentation(uid); var name = GetIdentityName(uid, representation); // Clone the old entity's grammar to the identity entity, for loc purposes. if (TryComp(uid, out var grammar)) { var identityGrammar = EnsureComp(ident); identityGrammar.Attributes.Clear(); foreach (var (k, v) in grammar.Attributes) { identityGrammar.Attributes.Add(k, v); } // If presumed name is null and we're using that, we set proper noun to be false ("the old woman") if (name != representation.TrueName && representation.PresumedName == null) identityGrammar.ProperNoun = false; } if (name == Name(ident)) return; MetaData(ident).EntityName = name; _adminLog.Add(LogType.Identity, LogImpact.Medium, $"{ToPrettyString(uid)} changed identity to {name}"); RaiseLocalEvent(new IdentityChangedEvent(uid, ident)); } private string GetIdentityName(EntityUid target, IdentityRepresentation representation) { var ev = new SeeIdentityAttemptEvent(); RaiseLocalEvent(target, ev); return representation.ToStringKnown(!ev.Cancelled); } /// /// Gets an 'identity representation' of an entity, with their true name being the entity name /// and their 'presumed name' and 'presumed job' being the name/job on their ID card, if they have one. /// private IdentityRepresentation GetIdentityRepresentation(EntityUid target, InventoryComponent? inventory=null, HumanoidAppearanceComponent? appearance=null) { int age = HumanoidCharacterProfile.MinimumAge; Gender gender = Gender.Neuter; // Always use their actual age and gender, since that can't really be changed by an ID. if (Resolve(target, ref appearance, false)) { gender = appearance.Gender; age = appearance.Age; } var trueName = Name(target); if (!Resolve(target, ref inventory, false)) return new(trueName, age, gender, string.Empty); string? presumedJob = null; string? presumedName = null; // Get their name and job from their ID for their presumed name. if (_idCard.TryFindIdCard(target, out var id)) { presumedName = string.IsNullOrWhiteSpace(id.FullName) ? null : id.FullName; presumedJob = id.JobTitle?.ToLowerInvariant(); } // If it didn't find a job, that's fine. return new(trueName, age, gender, presumedName, presumedJob); } #endregion } public sealed class IdentityChangedEvent : EntityEventArgs { public EntityUid CharacterEntity; public EntityUid IdentityEntity; public IdentityChangedEvent(EntityUid characterEntity, EntityUid identityEntity) { CharacterEntity = characterEntity; IdentityEntity = identityEntity; } }