mirror of
https://github.com/WWhiteDreamProject/wwdpublic.git
synced 2026-04-17 05:27:38 +03:00
<!-- This is a semi-strict format, you can add/remove sections as needed but the order/format should be kept the same Remove these comments before submitting --> # Description the adding AI is now up to y'all because i'm not touching loadout code for name datasets, but it shouldn't be too bad from here --------- Signed-off-by: sleepyyapril <123355664+sleepyyapril@users.noreply.github.com> Signed-off-by: SolStar <44028047+ewokswagger@users.noreply.github.com> Signed-off-by: deltanedas <39013340+deltanedas@users.noreply.github.com> Co-authored-by: themias <89101928+themias@users.noreply.github.com> Co-authored-by: Verm <32827189+Vermidia@users.noreply.github.com> Co-authored-by: DrSmugleaf <10968691+DrSmugleaf@users.noreply.github.com> Co-authored-by: Sphiral <145869023+SphiraI@users.noreply.github.com> Co-authored-by: Ed <96445749+TheShuEd@users.noreply.github.com> Co-authored-by: Mr. 27 <45323883+Dutch-VanDerLinde@users.noreply.github.com> Co-authored-by: metalgearsloth <comedian_vs_clown@hotmail.com> Co-authored-by: Alzore <140123969+Blackern5000@users.noreply.github.com> Co-authored-by: ravage <142820619+ravage123321@users.noreply.github.com> Co-authored-by: slarticodefast <161409025+slarticodefast@users.noreply.github.com> Co-authored-by: Intoxicating-Innocence <188202277+Intoxicating-Innocence@users.noreply.github.com> Co-authored-by: Saphire <lattice@saphi.re> Co-authored-by: metalgearsloth <31366439+metalgearsloth@users.noreply.github.com> Co-authored-by: Errant <35878406+Errant-4@users.noreply.github.com> Co-authored-by: Tayrtahn <tayrtahn@gmail.com> Co-authored-by: CaasGit <87243814+CaasGit@users.noreply.github.com> Co-authored-by: BramvanZijp <56019239+BramvanZijp@users.noreply.github.com> Co-authored-by: Boaz1111 <149967078+Boaz1111@users.noreply.github.com> Co-authored-by: NakataRin <45946146+NakataRin@users.noreply.github.com> Co-authored-by: Kara <lunarautomaton6@gmail.com> Co-authored-by: Plykiya <58439124+Plykiya@users.noreply.github.com> Co-authored-by: SlamBamActionman <slambamactionman@gmail.com> Co-authored-by: Doomsdrayk <robotdoughnut@comcast.net> Co-authored-by: Brandon Hu <103440971+Brandon-Huu@users.noreply.github.com> Co-authored-by: SlamBamActionman <83650252+SlamBamActionman@users.noreply.github.com> Co-authored-by: ElectroJr <leonsfriedrich@gmail.com> Co-authored-by: Pieter-Jan Briers <pieterjan.briers+git@gmail.com> Co-authored-by: DrSmugleaf <DrSmugleaf@users.noreply.github.com> Co-authored-by: Julian Giebel <juliangiebel@live.de> Co-authored-by: nikthechampiongr <32041239+nikthechampiongr@users.noreply.github.com> Co-authored-by: Repo <47093363+Titian3@users.noreply.github.com> Co-authored-by: Chief-Engineer <119664036+Chief-Engineer@users.noreply.github.com> Co-authored-by: icekot8 <93311212+icekot8@users.noreply.github.com> Co-authored-by: AJCM-git <60196617+AJCM-git@users.noreply.github.com> Co-authored-by: Leon Friedrich <60421075+ElectroJr@users.noreply.github.com> Co-authored-by: no <165581243+pissdemon@users.noreply.github.com> Co-authored-by: Tornado Tech <54727692+Tornado-Technology@users.noreply.github.com> Co-authored-by: osjarw <62134478+osjarw@users.noreply.github.com> Co-authored-by: Simon <63975668+Simyon264@users.noreply.github.com> Co-authored-by: TGRCDev <tgrc@tgrc.dev> Co-authored-by: Milon <milonpl.git@proton.me> Co-authored-by: deltanedas <39013340+deltanedas@users.noreply.github.com> Co-authored-by: ShadowCommander <10494922+ShadowCommander@users.noreply.github.com> Co-authored-by: Fildrance <fildrance@gmail.com> Co-authored-by: pa.pecherskij <pa.pecherskij@interfax.ru> Co-authored-by: chavonadelal <156101927+chavonadelal@users.noreply.github.com> Co-authored-by: SolStar <44028047+ewokswagger@users.noreply.github.com> Co-authored-by: K-Dynamic <20566341+K-Dynamic@users.noreply.github.com> Co-authored-by: lzk <124214523+lzk228@users.noreply.github.com> Co-authored-by: ArchRBX <5040911+ArchRBX@users.noreply.github.com> Co-authored-by: archrbx <punk.gear5260@fastmail.com> Co-authored-by: Radezolid <snappednexus@gmail.com> Co-authored-by: Nemanja <98561806+EmoGarbage404@users.noreply.github.com> Co-authored-by: EmoGarbage404 <retron404@gmail.com> Co-authored-by: MilenVolf <63782763+MilenVolf@users.noreply.github.com> Co-authored-by: Velcroboy <107660393+IamVelcroboy@users.noreply.github.com> Co-authored-by: Velcroboy <velcroboy333@hotmail.com> Co-authored-by: neuPanda <chriseparton@gmail.com> Co-authored-by: neuPanda <spainman0@yahoo.com> Co-authored-by: Dvir <39403717+dvir001@users.noreply.github.com> Co-authored-by: Whatstone <whatston3@gmail.com> Co-authored-by: VideoKompany <135313844+VlaDOS1408@users.noreply.github.com> (cherry picked from commit 93ed70acfeda357133a701f637d3faeec02749bb)
227 lines
8.8 KiB
C#
227 lines
8.8 KiB
C#
using Content.Server.Cuffs;
|
|
using Content.Server.Forensics;
|
|
using Content.Server.Humanoid;
|
|
using Content.Server.Implants.Components;
|
|
using Content.Server.Store.Components;
|
|
using Content.Server.Store.Systems;
|
|
using Content.Shared.Cuffs.Components;
|
|
using Content.Shared.Humanoid;
|
|
using Content.Shared.Implants;
|
|
using Content.Shared.Implants.Components;
|
|
using Content.Shared.Interaction;
|
|
using Content.Shared.Physics;
|
|
using Content.Shared.Popups;
|
|
using Content.Shared.Preferences;
|
|
using Robust.Shared.Audio.Systems;
|
|
using Robust.Shared.Map;
|
|
using Robust.Shared.Physics;
|
|
using Robust.Shared.Physics.Components;
|
|
using Robust.Shared.Random;
|
|
using System.Numerics;
|
|
using Content.Shared.Movement.Pulling.Components;
|
|
using Content.Shared.Movement.Pulling.Systems;
|
|
using Content.Shared.Store.Components;
|
|
using Robust.Shared.Collections;
|
|
using Robust.Shared.Map.Components;
|
|
|
|
namespace Content.Server.Implants;
|
|
|
|
public sealed class SubdermalImplantSystem : SharedSubdermalImplantSystem
|
|
{
|
|
[Dependency] private readonly CuffableSystem _cuffable = default!;
|
|
[Dependency] private readonly HumanoidAppearanceSystem _humanoidAppearance = default!;
|
|
[Dependency] private readonly IRobustRandom _random = default!;
|
|
[Dependency] private readonly MetaDataSystem _metaData = default!;
|
|
[Dependency] private readonly StoreSystem _store = default!;
|
|
[Dependency] private readonly SharedAudioSystem _audio = default!;
|
|
[Dependency] private readonly SharedPopupSystem _popup = default!;
|
|
[Dependency] private readonly SharedTransformSystem _xform = default!;
|
|
[Dependency] private readonly ForensicsSystem _forensicsSystem = default!;
|
|
[Dependency] private readonly PullingSystem _pullingSystem = default!;
|
|
[Dependency] private readonly EntityLookupSystem _lookupSystem = default!;
|
|
[Dependency] private readonly SharedMapSystem _mapSystem = default!;
|
|
|
|
private EntityQuery<PhysicsComponent> _physicsQuery;
|
|
private HashSet<Entity<MapGridComponent>> _targetGrids = [];
|
|
|
|
public override void Initialize()
|
|
{
|
|
base.Initialize();
|
|
|
|
_physicsQuery = GetEntityQuery<PhysicsComponent>();
|
|
|
|
SubscribeLocalEvent<SubdermalImplantComponent, UseFreedomImplantEvent>(OnFreedomImplant);
|
|
SubscribeLocalEvent<StoreComponent, ImplantRelayEvent<AfterInteractUsingEvent>>(OnStoreRelay);
|
|
SubscribeLocalEvent<SubdermalImplantComponent, ActivateImplantEvent>(OnActivateImplantEvent);
|
|
SubscribeLocalEvent<SubdermalImplantComponent, UseScramImplantEvent>(OnScramImplant);
|
|
SubscribeLocalEvent<SubdermalImplantComponent, UseDnaScramblerImplantEvent>(OnDnaScramblerImplant);
|
|
|
|
}
|
|
|
|
private void OnStoreRelay(EntityUid uid, StoreComponent store, ImplantRelayEvent<AfterInteractUsingEvent> implantRelay)
|
|
{
|
|
var args = implantRelay.Event;
|
|
|
|
if (args.Handled)
|
|
return;
|
|
|
|
// can only insert into yourself to prevent uplink checking with renault
|
|
if (args.Target != args.User)
|
|
return;
|
|
|
|
if (!TryComp<CurrencyComponent>(args.Used, out var currency))
|
|
return;
|
|
|
|
// same as store code, but message is only shown to yourself
|
|
args.Handled = _store.TryAddCurrency(_store.GetCurrencyValue(args.Used, currency), uid, store);
|
|
|
|
if (!args.Handled)
|
|
return;
|
|
|
|
var msg = Loc.GetString("store-currency-inserted-implant", ("used", args.Used));
|
|
_popup.PopupEntity(msg, args.User, args.User);
|
|
QueueDel(args.Used);
|
|
}
|
|
|
|
private void OnFreedomImplant(EntityUid uid, SubdermalImplantComponent component, UseFreedomImplantEvent args)
|
|
{
|
|
if (!TryComp<CuffableComponent>(component.ImplantedEntity, out var cuffs) || cuffs.Container.ContainedEntities.Count < 1)
|
|
return;
|
|
|
|
_cuffable.Uncuff(component.ImplantedEntity.Value, cuffs.LastAddedCuffs, cuffs.LastAddedCuffs);
|
|
args.Handled = true;
|
|
}
|
|
|
|
private void OnActivateImplantEvent(EntityUid uid, SubdermalImplantComponent component, ActivateImplantEvent args)
|
|
{
|
|
args.Handled = true;
|
|
}
|
|
|
|
private void OnScramImplant(EntityUid uid, SubdermalImplantComponent component, UseScramImplantEvent args)
|
|
{
|
|
if (component.ImplantedEntity is not { } ent)
|
|
return;
|
|
|
|
if (!TryComp<ScramImplantComponent>(uid, out var implant))
|
|
return;
|
|
|
|
// We need stop the user from being pulled so they don't just get "attached" with whoever is pulling them.
|
|
// This can for example happen when the user is cuffed and being pulled.
|
|
if (TryComp<PullableComponent>(ent, out var pull) && _pullingSystem.IsPulled(ent, pull))
|
|
_pullingSystem.TryStopPull(ent, pull);
|
|
|
|
var xform = Transform(ent);
|
|
var targetCoords = SelectRandomTileInRange(xform, implant.TeleportRadius);
|
|
|
|
if (targetCoords != null)
|
|
{
|
|
_xform.SetCoordinates(ent, targetCoords.Value);
|
|
_audio.PlayPvs(implant.TeleportSound, ent);
|
|
args.Handled = true;
|
|
}
|
|
}
|
|
|
|
private EntityCoordinates? SelectRandomTileInRange(TransformComponent userXform, float radius)
|
|
{
|
|
var userCoords = userXform.Coordinates.ToMap(EntityManager, _xform);
|
|
_targetGrids.Clear();
|
|
_lookupSystem.GetEntitiesInRange(userCoords, radius, _targetGrids);
|
|
Entity<MapGridComponent>? targetGrid = null;
|
|
|
|
if (_targetGrids.Count == 0)
|
|
return null;
|
|
|
|
// Give preference to the grid the entity is currently on.
|
|
// This does not guarantee that if the probability fails that the owner's grid won't be picked.
|
|
// In reality the probability is higher and depends on the number of grids.
|
|
if (userXform.GridUid != null && TryComp<MapGridComponent>(userXform.GridUid, out var gridComp))
|
|
{
|
|
var userGrid = new Entity<MapGridComponent>(userXform.GridUid.Value, gridComp);
|
|
if (_random.Prob(0.5f))
|
|
{
|
|
_targetGrids.Remove(userGrid);
|
|
targetGrid = userGrid;
|
|
}
|
|
}
|
|
|
|
if (targetGrid == null)
|
|
targetGrid = _random.GetRandom().PickAndTake(_targetGrids);
|
|
|
|
EntityCoordinates? targetCoords = null;
|
|
|
|
do
|
|
{
|
|
var valid = false;
|
|
|
|
var range = (float) Math.Sqrt(radius);
|
|
var box = Box2.CenteredAround(userCoords.Position, new Vector2(range, range));
|
|
var tilesInRange = _mapSystem.GetTilesEnumerator(targetGrid.Value.Owner, targetGrid.Value.Comp, box, false);
|
|
var tileList = new ValueList<Vector2i>();
|
|
|
|
while (tilesInRange.MoveNext(out var tile))
|
|
{
|
|
tileList.Add(tile.GridIndices);
|
|
}
|
|
|
|
while (tileList.Count != 0)
|
|
{
|
|
var tile = tileList.RemoveSwap(_random.Next(tileList.Count));
|
|
valid = true;
|
|
foreach (var entity in _mapSystem.GetAnchoredEntities(targetGrid.Value.Owner, targetGrid.Value.Comp,
|
|
tile))
|
|
{
|
|
if (!_physicsQuery.TryGetComponent(entity, out var body))
|
|
continue;
|
|
|
|
if (body.BodyType != BodyType.Static ||
|
|
!body.Hard ||
|
|
(body.CollisionLayer & (int) CollisionGroup.MobMask) == 0)
|
|
continue;
|
|
|
|
valid = false;
|
|
break;
|
|
}
|
|
|
|
if (valid)
|
|
{
|
|
targetCoords = new EntityCoordinates(targetGrid.Value.Owner,
|
|
_mapSystem.TileCenterToVector(targetGrid.Value, tile));
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (valid || _targetGrids.Count == 0) // if we don't do the check here then PickAndTake will blow up on an empty set.
|
|
break;
|
|
|
|
targetGrid = _random.GetRandom().PickAndTake(_targetGrids);
|
|
} while (true);
|
|
|
|
return targetCoords;
|
|
}
|
|
|
|
private void OnDnaScramblerImplant(EntityUid uid, SubdermalImplantComponent component, UseDnaScramblerImplantEvent args)
|
|
{
|
|
if (component.ImplantedEntity is not { } ent)
|
|
return;
|
|
|
|
if (TryComp<HumanoidAppearanceComponent>(ent, out var humanoid))
|
|
{
|
|
var newProfile = HumanoidCharacterProfile.RandomWithSpecies(humanoid.Species);
|
|
_humanoidAppearance.LoadProfile(ent, newProfile, humanoid);
|
|
_metaData.SetEntityName(ent, newProfile.Name);
|
|
if (TryComp<DnaComponent>(ent, out var dna))
|
|
{
|
|
dna.DNA = _forensicsSystem.GenerateDNA();
|
|
}
|
|
if (TryComp<FingerprintComponent>(ent, out var fingerprint))
|
|
{
|
|
fingerprint.Fingerprint = _forensicsSystem.GenerateFingerprint();
|
|
}
|
|
_popup.PopupEntity(Loc.GetString("scramble-implant-activated-popup"), ent, ent);
|
|
}
|
|
|
|
args.Handled = true;
|
|
QueueDel(uid);
|
|
}
|
|
}
|