You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

130 lines
5.4 KiB
C#

1 week ago
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.ClientPlayActionRpc(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.ClientCancelActionsByPrototypeIDRpc(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;
}
}
}
}