* Init testing * copyright * oops * Tracking the embed entity uid * testing stuff for gradual injection * work * weh * god save me * bleh * Yippee! * Again * Mini syringe ammo * cleaning up * mini syringes have a texture for fill amount * -3 cool points :( * hitboxes * init cleanup * much needed fixes * Fixes
186 lines
6.6 KiB
C#
186 lines
6.6 KiB
C#
using System.Numerics;
|
|
using Content.Shared.CombatMode.Pacification;
|
|
using Content.Shared.Damage;
|
|
using Content.Shared.DoAfter;
|
|
using Content.Shared.Hands.EntitySystems;
|
|
using Content.Shared.Interaction;
|
|
using Content.Shared.Mobs.Components;
|
|
using Content.Shared.Throwing;
|
|
using Robust.Shared.Audio.Systems;
|
|
using Robust.Shared.Map;
|
|
using Robust.Shared.Network;
|
|
using Robust.Shared.Physics;
|
|
using Robust.Shared.Physics.Components;
|
|
using Robust.Shared.Physics.Dynamics;
|
|
using Robust.Shared.Physics.Events;
|
|
using Robust.Shared.Physics.Systems;
|
|
using Robust.Shared.Serialization;
|
|
|
|
namespace Content.Shared.Projectiles;
|
|
|
|
public abstract partial class SharedProjectileSystem : EntitySystem
|
|
{
|
|
public const string ProjectileFixture = "projectile";
|
|
|
|
[Dependency] private readonly INetManager _netManager = default!;
|
|
[Dependency] private readonly SharedAudioSystem _audio = default!;
|
|
[Dependency] private readonly SharedDoAfterSystem _doAfter = default!;
|
|
[Dependency] private readonly SharedHandsSystem _hands = default!;
|
|
[Dependency] private readonly SharedPhysicsSystem _physics = default!;
|
|
[Dependency] private readonly SharedTransformSystem _transform = default!;
|
|
|
|
public override void Initialize()
|
|
{
|
|
base.Initialize();
|
|
|
|
SubscribeLocalEvent<ProjectileComponent, PreventCollideEvent>(PreventCollision);
|
|
SubscribeLocalEvent<EmbeddableProjectileComponent, ProjectileHitEvent>(OnEmbedProjectileHit);
|
|
SubscribeLocalEvent<EmbeddableProjectileComponent, ThrowDoHitEvent>(OnEmbedThrowDoHit);
|
|
SubscribeLocalEvent<EmbeddableProjectileComponent, ActivateInWorldEvent>(OnEmbedActivate);
|
|
SubscribeLocalEvent<EmbeddableProjectileComponent, RemoveEmbeddedProjectileEvent>(OnEmbedRemove);
|
|
}
|
|
|
|
private void OnEmbedActivate(EntityUid uid, EmbeddableProjectileComponent component, ActivateInWorldEvent args)
|
|
{
|
|
// Nuh uh
|
|
if (component.RemovalTime == null)
|
|
return;
|
|
|
|
if (args.Handled || !args.Complex || !TryComp<PhysicsComponent>(uid, out var physics) || physics.BodyType != BodyType.Static)
|
|
return;
|
|
|
|
args.Handled = true;
|
|
|
|
_doAfter.TryStartDoAfter(new DoAfterArgs(EntityManager, args.User, component.RemovalTime.Value,
|
|
new RemoveEmbeddedProjectileEvent(), eventTarget: uid, target: uid));
|
|
}
|
|
|
|
private void OnEmbedRemove(EntityUid uid, EmbeddableProjectileComponent component, RemoveEmbeddedProjectileEvent args)
|
|
{
|
|
// Whacky prediction issues.
|
|
if (args.Cancelled || _netManager.IsClient)
|
|
return;
|
|
|
|
if (component.DeleteOnRemove)
|
|
{
|
|
QueueDel(uid);
|
|
return;
|
|
}
|
|
|
|
var xform = Transform(uid);
|
|
TryComp<PhysicsComponent>(uid, out var physics);
|
|
_physics.SetBodyType(uid, BodyType.Dynamic, body: physics, xform: xform);
|
|
_transform.AttachToGridOrMap(uid, xform);
|
|
component.EmbeddedIntoUid = null;
|
|
Dirty(uid, component);
|
|
|
|
// Reset whether the projectile has damaged anything if it successfully was removed
|
|
if (TryComp<ProjectileComponent>(uid, out var projectile))
|
|
{
|
|
projectile.Shooter = null;
|
|
projectile.Weapon = null;
|
|
projectile.DamagedEntity = false;
|
|
}
|
|
|
|
// Land it just coz uhhh yeah
|
|
var landEv = new LandEvent(args.User, true);
|
|
RaiseLocalEvent(uid, ref landEv);
|
|
_physics.WakeBody(uid, body: physics);
|
|
|
|
// try place it in the user's hand
|
|
_hands.TryPickupAnyHand(args.User, uid);
|
|
}
|
|
|
|
private void OnEmbedThrowDoHit(EntityUid uid, EmbeddableProjectileComponent component, ThrowDoHitEvent args)
|
|
{
|
|
if (!component.EmbedOnThrow)
|
|
return;
|
|
|
|
Embed(uid, args.Target, null, component);
|
|
}
|
|
|
|
private void OnEmbedProjectileHit(EntityUid uid, EmbeddableProjectileComponent component, ref ProjectileHitEvent args)
|
|
{
|
|
Embed(uid, args.Target, args.Shooter, component);
|
|
|
|
// Raise a specific event for projectiles.
|
|
if (TryComp(uid, out ProjectileComponent? projectile))
|
|
{
|
|
var ev = new ProjectileEmbedEvent(projectile.Shooter!.Value, projectile.Weapon!.Value, args.Target);
|
|
RaiseLocalEvent(uid, ref ev);
|
|
}
|
|
}
|
|
|
|
private void Embed(EntityUid uid, EntityUid target, EntityUid? user, EmbeddableProjectileComponent component)
|
|
{
|
|
TryComp<PhysicsComponent>(uid, out var physics);
|
|
_physics.SetLinearVelocity(uid, Vector2.Zero, body: physics);
|
|
_physics.SetBodyType(uid, BodyType.Static, body: physics);
|
|
var xform = Transform(uid);
|
|
_transform.SetParent(uid, xform, target);
|
|
|
|
if (component.Offset != Vector2.Zero)
|
|
{
|
|
var rotation = xform.LocalRotation;
|
|
if (TryComp<ThrowingAngleComponent>(uid, out var throwingAngleComp))
|
|
rotation += throwingAngleComp.Angle;
|
|
_transform.SetLocalPosition(uid, xform.LocalPosition + rotation.RotateVec(component.Offset),
|
|
xform);
|
|
}
|
|
|
|
_audio.PlayPredicted(component.Sound, uid, null);
|
|
component.EmbeddedIntoUid = target;
|
|
var ev = new EmbedEvent(user, target);
|
|
RaiseLocalEvent(uid, ref ev);
|
|
Dirty(uid, component);
|
|
}
|
|
|
|
private void PreventCollision(EntityUid uid, ProjectileComponent component, ref PreventCollideEvent args)
|
|
{
|
|
if (component.IgnoreShooter && (args.OtherEntity == component.Shooter || args.OtherEntity == component.Weapon))
|
|
{
|
|
args.Cancelled = true;
|
|
}
|
|
}
|
|
|
|
public void SetShooter(EntityUid id, ProjectileComponent component, EntityUid shooterId)
|
|
{
|
|
if (component.Shooter == shooterId)
|
|
return;
|
|
|
|
component.Shooter = shooterId;
|
|
Dirty(id, component);
|
|
}
|
|
|
|
[Serializable, NetSerializable]
|
|
private sealed partial class RemoveEmbeddedProjectileEvent : DoAfterEvent
|
|
{
|
|
public override DoAfterEvent Clone() => this;
|
|
}
|
|
}
|
|
|
|
[Serializable, NetSerializable]
|
|
public sealed class ImpactEffectEvent : EntityEventArgs
|
|
{
|
|
public string Prototype;
|
|
public NetCoordinates Coordinates;
|
|
|
|
public ImpactEffectEvent(string prototype, NetCoordinates coordinates)
|
|
{
|
|
Prototype = prototype;
|
|
Coordinates = coordinates;
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Raised when an entity is just about to be hit with a projectile but can reflect it
|
|
/// </summary>
|
|
[ByRefEvent]
|
|
public record struct ProjectileReflectAttemptEvent(EntityUid ProjUid, ProjectileComponent Component, bool Cancelled);
|
|
|
|
/// <summary>
|
|
/// Raised when a projectile hits an entity
|
|
/// </summary>
|
|
[ByRefEvent]
|
|
public record struct ProjectileHitEvent(DamageSpecifier Damage, EntityUid Target, EntityUid? Shooter = null);
|