2023-04-07 11:21:12 -07:00
using System.Diagnostics.CodeAnalysis ;
2023-07-08 14:08:32 +10:00
using System.Numerics ;
2023-06-01 00:57:31 -05:00
using Content.Shared.Administration.Logs ;
using Content.Shared.Database ;
2023-08-25 12:48:27 +10:00
using Content.Shared.Hands ;
2023-07-30 18:07:45 -07:00
using Content.Shared.Inventory ;
using Content.Shared.Inventory.Events ;
2024-07-11 05:55:56 +00:00
using Content.Shared.Item.ItemToggle ;
2023-04-07 11:21:12 -07:00
using Content.Shared.Popups ;
2023-05-15 15:21:05 +10:00
using Content.Shared.Projectiles ;
using Content.Shared.Weapons.Ranged.Components ;
2023-09-11 02:04:02 -07:00
using Content.Shared.Weapons.Ranged.Events ;
2023-11-27 22:12:34 +11:00
using Robust.Shared.Audio.Systems ;
2023-05-15 15:21:05 +10:00
using Robust.Shared.Network ;
2023-09-11 02:04:02 -07:00
using Robust.Shared.Physics.Components ;
2023-04-07 11:21:12 -07:00
using Robust.Shared.Physics.Systems ;
using Robust.Shared.Random ;
2025-06-18 03:50:01 +03:00
using Content.Shared.Examine ;
using Content.Shared.Localizations ;
2023-04-02 16:48:32 +03:00
namespace Content.Shared.Weapons.Reflect ;
/// <summary>
/// This handles reflecting projectiles and hitscan shots.
/// </summary>
2024-01-03 17:24:02 +11:00
public sealed class ReflectSystem : EntitySystem
2023-04-02 16:48:32 +03:00
{
2023-05-15 15:21:05 +10:00
[Dependency] private readonly INetManager _netManager = default ! ;
2023-04-02 16:48:32 +03:00
[Dependency] private readonly IRobustRandom _random = default ! ;
2023-06-01 00:57:31 -05:00
[Dependency] private readonly ISharedAdminLogManager _adminLogger = default ! ;
2024-07-11 05:55:56 +00:00
[Dependency] private readonly ItemToggleSystem _toggle = default ! ;
2023-04-02 16:48:32 +03:00
[Dependency] private readonly SharedPopupSystem _popup = default ! ;
[Dependency] private readonly SharedPhysicsSystem _physics = default ! ;
[Dependency] private readonly SharedAudioSystem _audio = default ! ;
[Dependency] private readonly SharedTransformSystem _transform = default ! ;
2024-05-23 22:43:04 -04:00
2023-04-02 16:48:32 +03:00
public override void Initialize ( )
{
base . Initialize ( ) ;
2023-07-30 18:07:45 -07:00
2025-04-30 16:10:54 +02:00
Subs . SubscribeWithRelay < ReflectComponent , ProjectileReflectAttemptEvent > ( OnReflectUserCollide , baseEvent : false ) ;
Subs . SubscribeWithRelay < ReflectComponent , HitScanReflectAttemptEvent > ( OnReflectUserHitscan , baseEvent : false ) ;
2024-06-23 12:16:08 +10:00
SubscribeLocalEvent < ReflectComponent , ProjectileReflectAttemptEvent > ( OnReflectCollide ) ;
SubscribeLocalEvent < ReflectComponent , HitScanReflectAttemptEvent > ( OnReflectHitscan ) ;
2025-04-30 16:10:54 +02:00
2023-07-30 18:07:45 -07:00
SubscribeLocalEvent < ReflectComponent , GotEquippedEvent > ( OnReflectEquipped ) ;
SubscribeLocalEvent < ReflectComponent , GotUnequippedEvent > ( OnReflectUnequipped ) ;
2023-08-25 12:48:27 +10:00
SubscribeLocalEvent < ReflectComponent , GotEquippedHandEvent > ( OnReflectHandEquipped ) ;
SubscribeLocalEvent < ReflectComponent , GotUnequippedHandEvent > ( OnReflectHandUnequipped ) ;
2025-06-18 03:50:01 +03:00
SubscribeLocalEvent < ReflectComponent , ExaminedEvent > ( OnExamine ) ;
2025-04-30 16:10:54 +02:00
}
private void OnReflectUserCollide ( Entity < ReflectComponent > ent , ref ProjectileReflectAttemptEvent args )
{
if ( args . Cancelled )
return ;
2024-01-03 17:24:02 +11:00
2025-04-30 16:10:54 +02:00
if ( ! ent . Comp . InRightPlace )
return ; // only reflect when equipped correctly
if ( TryReflectProjectile ( ent , ent . Owner , args . ProjUid ) )
args . Cancelled = true ;
2023-04-02 16:48:32 +03:00
}
2025-04-30 16:10:54 +02:00
private void OnReflectUserHitscan ( Entity < ReflectComponent > ent , ref HitScanReflectAttemptEvent args )
2023-04-02 16:48:32 +03:00
{
2023-08-25 12:48:27 +10:00
if ( args . Reflected )
2023-04-19 20:02:30 +10:00
return ;
2023-08-25 12:48:27 +10:00
2025-04-30 16:10:54 +02:00
if ( ! ent . Comp . InRightPlace )
return ; // only reflect when equipped correctly
2024-06-04 14:26:19 +01:00
2025-04-30 16:10:54 +02:00
if ( TryReflectHitscan ( ent , ent . Owner , args . Shooter , args . SourceItem , args . Direction , args . Reflective , out var dir ) )
{
2024-06-23 12:16:08 +10:00
args . Direction = dir . Value ;
args . Reflected = true ;
}
2024-06-04 14:26:19 +01:00
}
2025-04-30 16:10:54 +02:00
private void OnReflectCollide ( Entity < ReflectComponent > ent , ref ProjectileReflectAttemptEvent args )
2024-06-04 14:26:19 +01:00
{
2025-04-30 16:10:54 +02:00
if ( args . Cancelled )
return ;
2024-06-04 14:26:19 +01:00
2025-04-30 16:10:54 +02:00
if ( TryReflectProjectile ( ent , ent . Owner , args . ProjUid ) )
2024-06-23 12:16:08 +10:00
args . Cancelled = true ;
2023-04-02 16:48:32 +03:00
}
2025-04-30 16:10:54 +02:00
private void OnReflectHitscan ( Entity < ReflectComponent > ent , ref HitScanReflectAttemptEvent args )
2023-04-02 16:48:32 +03:00
{
2025-04-30 16:10:54 +02:00
if ( args . Reflected )
2023-04-02 16:48:32 +03:00
return ;
2023-04-19 20:02:30 +10:00
2025-04-30 16:10:54 +02:00
if ( TryReflectHitscan ( ent , ent . Owner , args . Shooter , args . SourceItem , args . Direction , args . Reflective , out var dir ) )
{
args . Direction = dir . Value ;
args . Reflected = true ;
}
2024-06-04 14:26:19 +01:00
}
2025-04-30 16:10:54 +02:00
private bool TryReflectProjectile ( Entity < ReflectComponent > reflector , EntityUid user , Entity < ProjectileComponent ? > projectile )
2024-06-04 14:26:19 +01:00
{
2025-04-30 16:10:54 +02:00
if ( ! TryComp < ReflectiveComponent > ( projectile , out var reflective ) | |
( reflector . Comp . Reflects & reflective . Reflective ) = = 0x0 | |
! _toggle . IsActivated ( reflector . Owner ) | |
! _random . Prob ( reflector . Comp . ReflectProb ) | |
2024-06-23 12:16:08 +10:00
! TryComp < PhysicsComponent > ( projectile , out var physics ) )
{
2023-04-19 20:02:30 +10:00
return false ;
2024-06-23 12:16:08 +10:00
}
2023-04-02 16:48:32 +03:00
2025-04-30 16:10:54 +02:00
var rotation = _random . NextAngle ( - reflector . Comp . Spread / 2 , reflector . Comp . Spread / 2 ) . Opposite ( ) ;
2023-04-19 20:02:30 +10:00
var existingVelocity = _physics . GetMapLinearVelocity ( projectile , component : physics ) ;
2023-08-25 12:48:27 +10:00
var relativeVelocity = existingVelocity - _physics . GetMapLinearVelocity ( user ) ;
2023-04-19 20:02:30 +10:00
var newVelocity = rotation . RotateVec ( relativeVelocity ) ;
2023-04-02 16:48:32 +03:00
2023-04-19 20:02:30 +10:00
// Have the velocity in world terms above so need to convert it back to local.
var difference = newVelocity - existingVelocity ;
2023-04-02 16:48:32 +03:00
2023-04-19 20:02:30 +10:00
_physics . SetLinearVelocity ( projectile , physics . LinearVelocity + difference , body : physics ) ;
var locRot = Transform ( projectile ) . LocalRotation ;
var newRot = rotation . RotateVec ( locRot . ToVec ( ) ) ;
_transform . SetLocalRotation ( projectile , newRot . ToAngle ( ) ) ;
2025-04-30 16:10:54 +02:00
PlayAudioAndPopup ( reflector . Comp , user ) ;
2023-05-15 15:21:05 +10:00
2025-04-30 16:10:54 +02:00
if ( Resolve ( projectile , ref projectile . Comp , false ) )
2024-06-23 12:16:08 +10:00
{
2025-04-30 16:10:54 +02:00
_adminLogger . Add ( LogType . BulletHit , LogImpact . Medium , $"{ToPrettyString(user)} reflected {ToPrettyString(projectile)} from {ToPrettyString(projectile.Comp.Weapon)} shot by {projectile.Comp.Shooter}" ) ;
2023-06-01 00:57:31 -05:00
2025-04-30 16:10:54 +02:00
projectile . Comp . Shooter = user ;
projectile . Comp . Weapon = user ;
Dirty ( projectile , projectile . Comp ) ;
2024-06-23 12:16:08 +10:00
}
else
{
_adminLogger . Add ( LogType . BulletHit , LogImpact . Medium , $"{ToPrettyString(user)} reflected {ToPrettyString(projectile)}" ) ;
}
2023-05-15 15:21:05 +10:00
2023-04-19 20:02:30 +10:00
return true ;
2023-04-02 16:48:32 +03:00
}
2023-08-25 12:48:27 +10:00
private bool TryReflectHitscan (
2025-04-30 16:10:54 +02:00
Entity < ReflectComponent > reflector ,
2023-08-25 12:48:27 +10:00
EntityUid user ,
EntityUid ? shooter ,
EntityUid shotSource ,
Vector2 direction ,
2025-04-13 12:42:03 +01:00
ReflectType hitscanReflectType ,
2023-05-15 15:21:05 +10:00
[NotNullWhen(true)] out Vector2 ? newDirection )
2023-04-02 16:48:32 +03:00
{
2025-04-30 16:10:54 +02:00
if ( ( reflector . Comp . Reflects & hitscanReflectType ) = = 0x0 | |
! _toggle . IsActivated ( reflector . Owner ) | |
! _random . Prob ( reflector . Comp . ReflectProb ) )
Weapon Reflection Movement Mechanic (#27219)
* Weapon Reflection Movement Mechanic
Adds a movement mechanic to deflection.
Standing still gives you your best chance of deflecting a shot.
Moving lowers this to 2/3rds. Sprinting to 1/3rd.
This allows for robust players to express better and provides
counterplay to someone finding a goober-strong deflection
weapon, giving more design space.
As part of this PR I've also touched the numbers of a few swords,
shields, etc. and modified some descriptions to make them read
better. The balance numbers are not remotely final, but as intent:
1. All the sidearm swords (katana, cutlass, captain's sabre) have the same damage. There's no good reason the "ceremonial" blade the captain has doing more damage than a katana.
2. The Captain's Sabre has a 30% reflect chance, dropping to 20% when moving and 10% when sprinting. This one is controversial due to the recent nerf, I suspect: This could easily be 15->10->5?
3. The Energy Katana has a flat 30% reflect chance.
4. The meme Throngler has a 30% reflect chance, dropping to 20% when moving and 10% when sprinting.
5. The E-Sword has a 30% reflect chance, dropping to 20% when moving and 10% when sprinting.
6. The Double E-Sword has a mighty 75% reflect chance, dropping to 50% and then 25%.
7. Both reflective shields - Mirror and Energy - have a 95% deflect chance, dropping to 63% then 31%.
* Resolve PR comments.
* Weh?
* Reign in double esword a tad
* Shield nerfs no longer real
* Improve Mirror Cult desc
* Simple alert for deflection! No art yet.
* Added a new icon for deflecting
2024-05-07 19:14:58 +01:00
{
newDirection = null ;
return false ;
}
2025-04-30 16:10:54 +02:00
PlayAudioAndPopup ( reflector . Comp , user ) ;
2023-04-19 20:02:30 +10:00
2025-04-30 16:10:54 +02:00
var spread = _random . NextAngle ( - reflector . Comp . Spread / 2 , reflector . Comp . Spread / 2 ) ;
2023-05-15 15:21:05 +10:00
newDirection = - spread . RotateVec ( direction ) ;
2023-06-01 00:57:31 -05:00
if ( shooter ! = null )
2023-08-25 12:48:27 +10:00
_adminLogger . Add ( LogType . HitScanHit , LogImpact . Medium , $"{ToPrettyString(user)} reflected hitscan from {ToPrettyString(shotSource)} shot by {ToPrettyString(shooter.Value)}" ) ;
2023-06-01 00:57:31 -05:00
else
2023-08-25 12:48:27 +10:00
_adminLogger . Add ( LogType . HitScanHit , LogImpact . Medium , $"{ToPrettyString(user)} reflected hitscan from {ToPrettyString(shotSource)}" ) ;
2023-06-01 00:57:31 -05:00
2023-05-15 15:21:05 +10:00
return true ;
2023-04-02 16:48:32 +03:00
}
2023-07-30 18:07:45 -07:00
2025-04-30 16:10:54 +02:00
private void PlayAudioAndPopup ( ReflectComponent reflect , EntityUid user )
2023-07-30 18:07:45 -07:00
{
2025-04-30 16:10:54 +02:00
// Can probably be changed for prediction
if ( _netManager . IsServer )
{
_popup . PopupEntity ( Loc . GetString ( "reflect-shot" ) , user ) ;
_audio . PlayPvs ( reflect . SoundOnReflect , user ) ;
}
2023-08-25 12:48:27 +10:00
}
2023-07-30 18:07:45 -07:00
2025-04-30 16:10:54 +02:00
private void OnReflectEquipped ( Entity < ReflectComponent > ent , ref GotEquippedEvent args )
2023-08-25 12:48:27 +10:00
{
2025-04-30 16:10:54 +02:00
ent . Comp . InRightPlace = ( ent . Comp . SlotFlags & args . SlotFlags ) = = args . SlotFlags ;
Dirty ( ent ) ;
2023-08-25 12:48:27 +10:00
}
2023-07-30 18:07:45 -07:00
2025-04-30 16:10:54 +02:00
private void OnReflectUnequipped ( Entity < ReflectComponent > ent , ref GotUnequippedEvent args )
2023-08-25 12:48:27 +10:00
{
2025-04-30 16:10:54 +02:00
ent . Comp . InRightPlace = false ;
Dirty ( ent ) ;
2023-08-25 12:48:27 +10:00
}
2023-07-30 18:07:45 -07:00
2025-04-30 16:10:54 +02:00
private void OnReflectHandEquipped ( Entity < ReflectComponent > ent , ref GotEquippedHandEvent args )
2024-01-03 17:24:02 +11:00
{
2025-04-30 16:10:54 +02:00
ent . Comp . InRightPlace = ent . Comp . ReflectingInHands ;
Dirty ( ent ) ;
2024-01-03 17:24:02 +11:00
}
2025-04-30 16:10:54 +02:00
private void OnReflectHandUnequipped ( Entity < ReflectComponent > ent , ref GotUnequippedHandEvent args )
2023-08-25 12:48:27 +10:00
{
2025-04-30 16:10:54 +02:00
ent . Comp . InRightPlace = false ;
Dirty ( ent ) ;
2023-07-30 18:07:45 -07:00
}
2025-06-18 03:50:01 +03:00
#region Examine
private void OnExamine ( Entity < ReflectComponent > ent , ref ExaminedEvent args )
{
// This isn't examine verb or something just because it looks too much bad.
// Trust me, universal verb for the potential weapons, armor and walls looks awful.
var value = MathF . Round ( ent . Comp . ReflectProb * 100 , 1 ) ;
if ( ! _toggle . IsActivated ( ent . Owner ) | | value = = 0 | | ent . Comp . Reflects = = ReflectType . None )
return ;
var compTypes = ent . Comp . Reflects . ToString ( ) . Split ( ", " ) ;
List < string > typeList = new ( compTypes . Length ) ;
for ( var i = 0 ; i < compTypes . Length ; i + + )
{
var type = Loc . GetString ( ( "reflect-component-" + compTypes [ i ] ) . ToLower ( ) ) ;
typeList . Add ( type ) ;
}
var msg = ContentLocalizationManager . FormatList ( typeList ) ;
args . PushMarkup ( Loc . GetString ( "reflect-component-examine" , ( "value" , value ) , ( "type" , msg ) ) ) ;
}
#endregion
2023-04-02 16:48:32 +03:00
}