-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathFXProjectileTargetedAction.cs
129 lines (113 loc) · 5.39 KB
/
FXProjectileTargetedAction.cs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
using System;
using Unity.BossRoom.Gameplay.GameplayObjects;
using Unity.BossRoom.Gameplay.GameplayObjects.Character;
using Unity.Netcode;
using UnityEngine;
namespace Unity.BossRoom.Gameplay.Actions
{
/// <summary>
/// Action that represents an always-hit raybeam-style ranged attack. A particle is shown from caster to target, and then the
/// target takes damage. (It is not possible to escape the hit; the target ALWAYS takes damage.) This is intended for fairly
/// swift particles; the time before it's applied is based on a simple distance-check at the attack's start.
/// (If no target is provided (because the user clicked on an empty spot on the map) or if the caster doesn't have line of
/// sight to the target (because it's behind a wall), we still perform an action, it just hits nothing.
/// </summary>
[CreateAssetMenu(menuName = "BossRoom/Actions/FX Projectile Targeted Action")]
public partial class FXProjectileTargetedAction : Action
{
private bool m_ImpactedTarget;
private float m_TimeUntilImpact;
private IDamageable m_DamageableTarget;
public override bool OnStart(ServerCharacter serverCharacter)
{
m_DamageableTarget = GetDamageableTarget(serverCharacter);
// figure out where the player wants us to aim at...
Vector3 targetPos = m_DamageableTarget != null ? m_DamageableTarget.transform.position : m_Data.Position;
// then make sure we can actually see that point!
if (!ActionUtils.HasLineOfSight(serverCharacter.physicsWrapper.Transform.position, targetPos, out Vector3 collidePos))
{
// we do not have line of sight to the target point. So our target instead becomes the obstruction point
m_DamageableTarget = null;
targetPos = collidePos;
// and update our action data so that when we send it to the clients, it will be up-to-date
Data.TargetIds = new ulong[0];
Data.Position = targetPos;
}
// turn to face our target!
serverCharacter.physicsWrapper.Transform.LookAt(targetPos);
// figure out how long the pretend-projectile will be flying to the target
float distanceToTargetPos = Vector3.Distance(targetPos, serverCharacter.physicsWrapper.Transform.position);
m_TimeUntilImpact = Config.ExecTimeSeconds + (distanceToTargetPos / Config.Projectiles[0].Speed_m_s);
serverCharacter.serverAnimationHandler.NetworkAnimator.SetTrigger(Config.Anim);
// tell clients to visualize this action
serverCharacter.clientCharacter.RecvDoActionClientRPC(Data);
return true;
}
public override void Reset()
{
base.Reset();
m_ImpactedTarget = false;
m_TimeUntilImpact = 0;
m_DamageableTarget = null;
m_ImpactPlayed = false;
m_ProjectileDuration = 0;
m_Projectile = null;
m_Target = null;
m_TargetTransform = null;
}
public override bool OnUpdate(ServerCharacter clientCharacter)
{
if (!m_ImpactedTarget && m_TimeUntilImpact <= TimeRunning)
{
m_ImpactedTarget = true;
if (m_DamageableTarget != null)
{
m_DamageableTarget.ReceiveHP(clientCharacter, -Config.Projectiles[0].Damage);
}
}
return true;
}
public override void Cancel(ServerCharacter serverCharacter)
{
if (!m_ImpactedTarget)
{
serverCharacter.clientCharacter.RecvCancelActionsByPrototypeIDClientRpc(ActionID);
}
}
/// <summary>
/// Returns our intended target, or null if not found/no target.
/// </summary>
private IDamageable GetDamageableTarget(ServerCharacter parent)
{
if (Data.TargetIds == null || Data.TargetIds.Length == 0)
{
return null;
}
NetworkObject obj;
if (NetworkManager.Singleton.SpawnManager.SpawnedObjects.TryGetValue(Data.TargetIds[0], out obj) && obj != null)
{
// make sure this isn't a friend (or if it is, make sure this is a friendly-fire action)
var serverChar = obj.GetComponent<ServerCharacter>();
if (serverChar && serverChar.IsNpc == (Config.IsFriendly ^ parent.IsNpc))
{
// not a valid target
return null;
}
if (PhysicsWrapper.TryGetPhysicsWrapper(Data.TargetIds[0], out var physicsWrapper))
{
return physicsWrapper.DamageCollider.GetComponent<IDamageable>();
}
else
{
return obj.GetComponent<IDamageable>();
}
}
else
{
// target could have legitimately disappeared in the time it took to queue this action... but that's pretty unlikely, so we'll log about it to ease debugging
Debug.Log($"FXProjectileTargetedAction was targeted at ID {Data.TargetIds[0]}, but that target can't be found in spawned object list! (May have just been deleted?)");
return null;
}
}
}
}