-
Notifications
You must be signed in to change notification settings - Fork 516
/
AOEAction.cs
89 lines (78 loc) · 3.63 KB
/
AOEAction.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
using System;
using Unity.BossRoom.Gameplay.GameplayObjects;
using Unity.BossRoom.Gameplay.GameplayObjects.Character;
using UnityEngine;
namespace Unity.BossRoom.Gameplay.Actions
{
/// <summary>
/// Area-of-effect attack Action. The attack is centered on a point provided by the client.
/// </summary>
[CreateAssetMenu(menuName = "BossRoom/Actions/AOE Action")]
public class AOEAction : Action
{
/// <summary>
/// Cheat prevention: to ensure that players don't perform AoEs outside of their attack range,
/// we ensure that the target is less than Range meters away from the player, plus this "fudge
/// factor" to accomodate miscellaneous minor movement.
/// </summary>
const float k_MaxDistanceDivergence = 1;
bool m_DidAoE;
public override bool OnStart(ServerCharacter serverCharacter)
{
float distanceAway = Vector3.Distance(serverCharacter.physicsWrapper.Transform.position, Data.Position);
if (distanceAway > Config.Range + k_MaxDistanceDivergence)
{
// Due to latency, it's possible for the client side click check to be out of date with the server driven position. Doing a final check server side to make sure.
return ActionConclusion.Stop;
}
// broadcasting to all players including myself.
// We don't know our actual targets for this attack until it triggers, so the client can't use the TargetIds list (and we clear it out for clarity).
// This means we are responsible for triggering reaction-anims ourselves, which we do in PerformAoe()
Data.TargetIds = new ulong[0];
serverCharacter.serverAnimationHandler.NetworkAnimator.SetTrigger(Config.Anim);
serverCharacter.clientCharacter.ClientPlayActionRpc(Data);
return ActionConclusion.Continue;
}
public override void Reset()
{
base.Reset();
m_DidAoE = false;
}
public override bool OnUpdate(ServerCharacter clientCharacter)
{
if (TimeRunning >= Config.ExecTimeSeconds && !m_DidAoE)
{
// actually perform the AoE attack
m_DidAoE = true;
PerformAoE(clientCharacter);
}
return ActionConclusion.Continue;
}
private void PerformAoE(ServerCharacter parent)
{
// Note: could have a non alloc version of this overlap sphere where we statically store our collider array, but since this is a self
// destroyed object, the complexity added to have a static pool of colliders that could be called by multiplayer players at the same time
// doesn't seem worth it for now.
var colliders = Physics.OverlapSphere(m_Data.Position, Config.Radius, LayerMask.GetMask("NPCs"));
for (var i = 0; i < colliders.Length; i++)
{
var enemy = colliders[i].GetComponent<IDamageable>();
if (enemy != null)
{
// actually deal the damage
enemy.ReceiveHP(parent, -Config.Amount);
}
}
}
public override bool OnStartClient(ClientCharacter clientCharacter)
{
base.OnStartClient(clientCharacter);
GameObject.Instantiate(Config.Spawns[0], Data.Position, Quaternion.identity);
return ActionConclusion.Stop;
}
public override bool OnUpdateClient(ClientCharacter clientCharacter)
{
throw new Exception("This should not execute");
}
}
}