Refactor MeleeWeaponComponent and related comps to be ECS (#4133)
* move everything to MeleeWeaponSystem * refactor MeleeChemicalInjector * hypospray and flash refactor * stunbaton refactor * bugfixes * flash afterinteract * resolve issues * props * playing the slots * MeleeInteractEvent + bugfixes * spear can actually use MeleeChemicalInjector
This commit is contained in:
@@ -1,46 +1,31 @@
|
||||
using System;
|
||||
using System.Collections.Generic;
|
||||
using System.Linq;
|
||||
using Content.Server.GameObjects.EntitySystems;
|
||||
using Content.Shared.Damage;
|
||||
using Content.Shared.GameObjects.Components.Damage;
|
||||
using Content.Shared.GameObjects.Components.Items;
|
||||
using Content.Shared.Interfaces.GameObjects.Components;
|
||||
using Content.Shared.Physics;
|
||||
using Robust.Server.GameObjects;
|
||||
using Robust.Shared.Audio;
|
||||
using Robust.Shared.GameObjects;
|
||||
using Robust.Shared.IoC;
|
||||
using Robust.Shared.Maths;
|
||||
using Robust.Shared.Physics;
|
||||
using Robust.Shared.Physics.Broadphase;
|
||||
using Robust.Shared.Player;
|
||||
using Robust.Shared.Serialization.Manager.Attributes;
|
||||
using Robust.Shared.Timing;
|
||||
using Robust.Shared.ViewVariables;
|
||||
|
||||
namespace Content.Server.GameObjects.Components.Weapon.Melee
|
||||
{
|
||||
[RegisterComponent]
|
||||
public class MeleeWeaponComponent : Component, IAttack, IHandSelected
|
||||
public class MeleeWeaponComponent : Component
|
||||
{
|
||||
[Dependency] private readonly IGameTiming _gameTiming = default!;
|
||||
|
||||
public override string Name => "MeleeWeapon";
|
||||
private TimeSpan _lastAttackTime;
|
||||
private TimeSpan _cooldownEnd;
|
||||
|
||||
[ViewVariables(VVAccess.ReadWrite)]
|
||||
[DataField("hitSound")]
|
||||
private string _hitSound = "/Audio/Weapons/genhit1.ogg";
|
||||
public string HitSound { get; set; } = "/Audio/Weapons/genhit1.ogg";
|
||||
|
||||
[ViewVariables(VVAccess.ReadWrite)]
|
||||
[DataField("missSound")]
|
||||
private string _missSound = "/Audio/Weapons/punchmiss.ogg";
|
||||
public string MissSound { get; set; } = "/Audio/Weapons/punchmiss.ogg";
|
||||
|
||||
[ViewVariables]
|
||||
[DataField("arcCooldownTime")]
|
||||
public float ArcCooldownTime { get; private set; } = 1f;
|
||||
public float ArcCooldownTime { get; } = 1f;
|
||||
|
||||
[ViewVariables]
|
||||
[DataField("cooldownTime")]
|
||||
public float CooldownTime { get; private set; } = 1f;
|
||||
public float CooldownTime { get; } = 1f;
|
||||
|
||||
[ViewVariables(VVAccess.ReadWrite)]
|
||||
[DataField("clickArc")]
|
||||
@@ -50,7 +35,9 @@ namespace Content.Server.GameObjects.Components.Weapon.Melee
|
||||
[DataField("arc")]
|
||||
public string Arc { get; set; } = "default";
|
||||
|
||||
[ViewVariables(VVAccess.ReadWrite)] [DataField("arcwidth")] public float ArcWidth { get; set; } = 90;
|
||||
[ViewVariables(VVAccess.ReadWrite)]
|
||||
[DataField("arcwidth")]
|
||||
public float ArcWidth { get; set; } = 90;
|
||||
|
||||
[ViewVariables(VVAccess.ReadWrite)]
|
||||
[DataField("range")]
|
||||
@@ -64,185 +51,11 @@ namespace Content.Server.GameObjects.Components.Weapon.Melee
|
||||
[DataField("damageType")]
|
||||
public DamageType DamageType { get; set; } = DamageType.Blunt;
|
||||
|
||||
[ViewVariables(VVAccess.ReadWrite)] [DataField("clickAttackEffect")] public bool ClickAttackEffect { get; set; } = true;
|
||||
[ViewVariables(VVAccess.ReadWrite)]
|
||||
[DataField("clickAttackEffect")]
|
||||
public bool ClickAttackEffect { get; set; } = true;
|
||||
|
||||
protected virtual bool OnHitEntities(IReadOnlyList<IEntity> entities, AttackEvent eventArgs)
|
||||
{
|
||||
return true;
|
||||
}
|
||||
|
||||
bool IAttack.WideAttack(AttackEvent eventArgs)
|
||||
{
|
||||
if (!eventArgs.WideAttack) return true;
|
||||
|
||||
var curTime = _gameTiming.CurTime;
|
||||
|
||||
if (curTime < _cooldownEnd)
|
||||
return true;
|
||||
|
||||
var location = eventArgs.User.Transform.Coordinates;
|
||||
var diff = eventArgs.ClickLocation.ToMapPos(Owner.EntityManager) - location.ToMapPos(Owner.EntityManager);
|
||||
var angle = Angle.FromWorldVec(diff);
|
||||
|
||||
// This should really be improved. GetEntitiesInArc uses pos instead of bounding boxes.
|
||||
var entities = ArcRayCast(eventArgs.User.Transform.WorldPosition, angle, eventArgs.User);
|
||||
|
||||
if (entities.Count != 0)
|
||||
{
|
||||
SoundSystem.Play(Filter.Pvs(Owner), _hitSound, entities.First().Transform.Coordinates);
|
||||
}
|
||||
else
|
||||
{
|
||||
SoundSystem.Play(Filter.Pvs(Owner), _missSound, eventArgs.User.Transform.Coordinates);
|
||||
}
|
||||
|
||||
var hitEntities = new List<IEntity>();
|
||||
foreach (var entity in entities)
|
||||
{
|
||||
if (!entity.Transform.IsMapTransform || entity == eventArgs.User)
|
||||
continue;
|
||||
|
||||
if (entity.TryGetComponent(out IDamageableComponent? damageComponent))
|
||||
{
|
||||
damageComponent.ChangeDamage(DamageType, Damage, false, Owner);
|
||||
hitEntities.Add(entity);
|
||||
}
|
||||
}
|
||||
SendMessage(new MeleeHitMessage(hitEntities));
|
||||
|
||||
if (!OnHitEntities(hitEntities, eventArgs)) return false;
|
||||
|
||||
if (Arc != null)
|
||||
{
|
||||
var sys = EntitySystem.Get<MeleeWeaponSystem>();
|
||||
sys.SendAnimation(Arc, angle, eventArgs.User, Owner, hitEntities);
|
||||
}
|
||||
|
||||
_lastAttackTime = curTime;
|
||||
_cooldownEnd = _lastAttackTime + TimeSpan.FromSeconds(ArcCooldownTime);
|
||||
|
||||
RefreshItemCooldown();
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
bool IAttack.ClickAttack(AttackEvent eventArgs)
|
||||
{
|
||||
if (eventArgs.WideAttack) return false;
|
||||
|
||||
var curTime = _gameTiming.CurTime;
|
||||
|
||||
if (curTime < _cooldownEnd || !eventArgs.Target.IsValid())
|
||||
return true;
|
||||
|
||||
var target = eventArgs.TargetEntity;
|
||||
|
||||
var location = eventArgs.User.Transform.Coordinates;
|
||||
var diff = eventArgs.ClickLocation.ToMapPos(Owner.EntityManager) - location.ToMapPos(Owner.EntityManager);
|
||||
var angle = Angle.FromWorldVec(diff);
|
||||
|
||||
if (target != null)
|
||||
{
|
||||
SoundSystem.Play(Filter.Pvs(Owner), _hitSound, target);
|
||||
}
|
||||
else
|
||||
{
|
||||
SoundSystem.Play(Filter.Pvs(Owner), _missSound, eventArgs.User);
|
||||
return false;
|
||||
}
|
||||
|
||||
if (target.TryGetComponent(out IDamageableComponent? damageComponent))
|
||||
{
|
||||
damageComponent.ChangeDamage(DamageType, Damage, false, Owner);
|
||||
}
|
||||
SendMessage(new MeleeHitMessage(new List<IEntity> { target }));
|
||||
|
||||
var targets = new[] { target };
|
||||
|
||||
if (!OnHitEntities(targets, eventArgs))
|
||||
return false;
|
||||
|
||||
if (ClickArc != null)
|
||||
{
|
||||
var sys = EntitySystem.Get<MeleeWeaponSystem>();
|
||||
sys.SendAnimation(ClickArc, angle, eventArgs.User, Owner, targets, ClickAttackEffect, false);
|
||||
}
|
||||
|
||||
_lastAttackTime = curTime;
|
||||
_cooldownEnd = _lastAttackTime + TimeSpan.FromSeconds(CooldownTime);
|
||||
|
||||
RefreshItemCooldown();
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
private HashSet<IEntity> ArcRayCast(Vector2 position, Angle angle, IEntity ignore)
|
||||
{
|
||||
var widthRad = Angle.FromDegrees(ArcWidth);
|
||||
var increments = 1 + 35 * (int) Math.Ceiling(widthRad / (2 * Math.PI));
|
||||
var increment = widthRad / increments;
|
||||
var baseAngle = angle - widthRad / 2;
|
||||
|
||||
var resSet = new HashSet<IEntity>();
|
||||
|
||||
var mapId = Owner.Transform.MapID;
|
||||
for (var i = 0; i < increments; i++)
|
||||
{
|
||||
var castAngle = new Angle(baseAngle + increment * i);
|
||||
var res = EntitySystem.Get<SharedBroadPhaseSystem>().IntersectRay(mapId,
|
||||
new CollisionRay(position, castAngle.ToWorldVec(),
|
||||
(int) (CollisionGroup.Impassable | CollisionGroup.MobImpassable)), Range, ignore).ToList();
|
||||
|
||||
if (res.Count != 0)
|
||||
{
|
||||
resSet.Add(res[0].HitEntity);
|
||||
}
|
||||
}
|
||||
|
||||
return resSet;
|
||||
}
|
||||
|
||||
void IHandSelected.HandSelected(HandSelectedEventArgs eventArgs)
|
||||
{
|
||||
var curTime = _gameTiming.CurTime;
|
||||
var cool = TimeSpan.FromSeconds(CooldownTime * 0.5f);
|
||||
|
||||
if (curTime < _cooldownEnd)
|
||||
{
|
||||
if (_cooldownEnd - curTime < cool)
|
||||
{
|
||||
_lastAttackTime = curTime;
|
||||
_cooldownEnd += cool;
|
||||
}
|
||||
else
|
||||
return;
|
||||
}
|
||||
else
|
||||
{
|
||||
_lastAttackTime = curTime;
|
||||
_cooldownEnd = curTime + cool;
|
||||
}
|
||||
|
||||
RefreshItemCooldown();
|
||||
}
|
||||
|
||||
private void RefreshItemCooldown()
|
||||
{
|
||||
if (Owner.TryGetComponent(out ItemCooldownComponent? cooldown))
|
||||
{
|
||||
cooldown.CooldownStart = _lastAttackTime;
|
||||
cooldown.CooldownEnd = _cooldownEnd;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
public class MeleeHitMessage : ComponentMessage
|
||||
{
|
||||
public readonly List<IEntity> HitEntities;
|
||||
|
||||
public MeleeHitMessage(List<IEntity> hitEntities)
|
||||
{
|
||||
HitEntities = hitEntities;
|
||||
}
|
||||
public TimeSpan LastAttackTime;
|
||||
public TimeSpan CooldownEnd;
|
||||
}
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user