223 lines
7.8 KiB
C#
223 lines
7.8 KiB
C#
using System.Collections;
|
|
using System.Collections.Generic;
|
|
using UnityEngine;
|
|
|
|
namespace BNG {
|
|
/// <summary>
|
|
/// A Grabbable object that can stick to objects and deal damage
|
|
/// </summary>
|
|
public class Arrow : MonoBehaviour {
|
|
Rigidbody rb;
|
|
Grabbable grab;
|
|
public bool Flying = false;
|
|
public float ZVel = 0;
|
|
|
|
public Collider ShaftCollider;
|
|
AudioSource impactSound;
|
|
|
|
float flightTime = 0f;
|
|
float destroyTime = 10f; // Time in seconds to destroy arrow
|
|
Coroutine queueDestroy;
|
|
|
|
public Projectile ProjectileObject;
|
|
|
|
// Get this value from the ProjectileObject
|
|
float arrowDamage;
|
|
|
|
// Start is called before the first frame update
|
|
void Start() {
|
|
rb = GetComponent<Rigidbody>();
|
|
impactSound = GetComponent<AudioSource>();
|
|
ShaftCollider = GetComponent<Collider>();
|
|
grab = GetComponent<Grabbable>();
|
|
|
|
if(ProjectileObject == null) {
|
|
ProjectileObject = gameObject.AddComponent<Projectile>();
|
|
ProjectileObject.Damage = 50;
|
|
ProjectileObject.StickToObject = true;
|
|
ProjectileObject.enabled = false;
|
|
}
|
|
|
|
arrowDamage = ProjectileObject.Damage;
|
|
}
|
|
|
|
void FixedUpdate() {
|
|
|
|
bool beingHeld = grab != null && grab.BeingHeld;
|
|
|
|
// Align arrow with velocity
|
|
if (!beingHeld && rb != null && rb.velocity != Vector3.zero && Flying && ZVel > 0.02) {
|
|
rb.rotation = Quaternion.LookRotation(rb.velocity);
|
|
}
|
|
|
|
ZVel = transform.InverseTransformDirection(rb.velocity).z;
|
|
|
|
if (Flying) {
|
|
flightTime += Time.fixedDeltaTime;
|
|
}
|
|
|
|
// Cancel Destroy if we just picked this up
|
|
if(queueDestroy != null && grab != null && grab.BeingHeld) {
|
|
StopCoroutine(queueDestroy);
|
|
}
|
|
}
|
|
|
|
public void ShootArrow(Vector3 shotForce) {
|
|
|
|
flightTime = 0f;
|
|
Flying = true;
|
|
|
|
transform.parent = null;
|
|
|
|
rb.isKinematic = false;
|
|
rb.useGravity = true;
|
|
rb.collisionDetectionMode = CollisionDetectionMode.Continuous;
|
|
rb.constraints = RigidbodyConstraints.None;
|
|
rb.AddForce(shotForce, ForceMode.VelocityChange);
|
|
StartCoroutine(ReEnableCollider());
|
|
queueDestroy = StartCoroutine(QueueDestroy());
|
|
}
|
|
|
|
IEnumerator QueueDestroy() {
|
|
yield return new WaitForSeconds(destroyTime);
|
|
|
|
if (grab != null && !grab.BeingHeld && transform.parent == null) {
|
|
Destroy(this.gameObject);
|
|
}
|
|
}
|
|
|
|
IEnumerator ReEnableCollider() {
|
|
|
|
// Wait a few frames before re-enabling collider on bow shaft
|
|
// This prevents the arrow from shooting ourselves, the bow, etc.
|
|
// If you want the arrow to still have physics while attached,
|
|
// parent a collider to the arrow near the tip
|
|
int waitFrames = 3;
|
|
for (int x = 0; x < waitFrames; x++) {
|
|
yield return new WaitForFixedUpdate();
|
|
}
|
|
|
|
ShaftCollider.enabled = true;
|
|
}
|
|
|
|
private void OnCollisionEnter(Collision collision) {
|
|
|
|
// Ignore parent collisions
|
|
if (transform.parent != null && collision.transform == transform.parent) {
|
|
return;
|
|
}
|
|
|
|
// Don't count collisions if being held
|
|
if(grab != null && grab.BeingHeld) {
|
|
return;
|
|
}
|
|
|
|
// Don't Count Triggers
|
|
if(collision.collider.isTrigger) {
|
|
return;
|
|
}
|
|
|
|
string colNameLower = collision.transform.name.ToLower();
|
|
|
|
// Ignore other very close bows and arrows
|
|
if (flightTime < 1 && (colNameLower.Contains("arrow") || colNameLower.Contains("bow"))) {
|
|
Physics.IgnoreCollision(collision.collider, ShaftCollider, true);
|
|
return;
|
|
}
|
|
|
|
// ignore player collision if quick shot
|
|
if (flightTime < 1 && collision.transform.name.ToLower().Contains("player")) {
|
|
Physics.IgnoreCollision(collision.collider, ShaftCollider, true);
|
|
return;
|
|
}
|
|
|
|
// Damage if possible
|
|
float zVel = System.Math.Abs(transform.InverseTransformDirection(rb.velocity).z);
|
|
bool doStick = true;
|
|
if (zVel > 0.02f && !rb.isKinematic) {
|
|
|
|
Damageable d = collision.gameObject.GetComponent<Damageable>();
|
|
if (d) {
|
|
d.DealDamage(arrowDamage, collision.GetContact(0).point, collision.GetContact(0).normal, true, gameObject, collision.collider.gameObject);
|
|
}
|
|
|
|
// Don't stick to dead objects
|
|
if (d != null && d.Health <= 0) {
|
|
doStick = false;
|
|
}
|
|
}
|
|
|
|
// Check to stick to object
|
|
if (!rb.isKinematic && Flying) {
|
|
if (zVel > 0.02f) {
|
|
if (grab != null && grab.BeingHeld) {
|
|
grab.DropItem(false, false);
|
|
}
|
|
if (doStick) {
|
|
tryStickArrow(collision);
|
|
}
|
|
|
|
Flying = false;
|
|
|
|
playSoundInterval(2.462f, 2.68f);
|
|
}
|
|
}
|
|
}
|
|
|
|
// Attach to collider
|
|
void tryStickArrow(Collision collision) {
|
|
|
|
Rigidbody colRigid = collision.collider.GetComponent<Rigidbody>();
|
|
transform.parent = null; // Start out with arrow being in World space
|
|
|
|
// If the collider is static then we don't need to do anything. Just stop it.
|
|
if (collision.gameObject.isStatic) {
|
|
rb.collisionDetectionMode = CollisionDetectionMode.Discrete;
|
|
rb.isKinematic = true;
|
|
}
|
|
// Next try attaching to rigidbody via FixedJoint
|
|
else if (colRigid != null && !colRigid.isKinematic) {
|
|
FixedJoint joint = gameObject.AddComponent<FixedJoint>();
|
|
joint.connectedBody = colRigid;
|
|
joint.enableCollision = false;
|
|
joint.breakForce = float.MaxValue;
|
|
joint.breakTorque = float.MaxValue;
|
|
}
|
|
else if (colRigid != null && colRigid.isKinematic && collision.transform.localScale == Vector3.one) {
|
|
transform.SetParent(collision.transform);
|
|
rb.useGravity = false;
|
|
rb.collisionDetectionMode = CollisionDetectionMode.Discrete;
|
|
rb.isKinematic = true;
|
|
rb.constraints = RigidbodyConstraints.FreezeAll;
|
|
rb.WakeUp();
|
|
}
|
|
// Finally, try parenting or just setting the arrow to kinematic
|
|
else {
|
|
if (collision.transform.localScale == Vector3.one) {
|
|
transform.SetParent(collision.transform);
|
|
rb.constraints = RigidbodyConstraints.FreezeAll;
|
|
}
|
|
else {
|
|
rb.collisionDetectionMode = CollisionDetectionMode.Discrete;
|
|
rb.useGravity = false;
|
|
rb.isKinematic = true;
|
|
}
|
|
}
|
|
}
|
|
|
|
void playSoundInterval(float fromSeconds, float toSeconds) {
|
|
if (impactSound) {
|
|
|
|
if (impactSound.isPlaying) {
|
|
impactSound.Stop();
|
|
}
|
|
|
|
impactSound.time = fromSeconds;
|
|
impactSound.pitch = Time.timeScale;
|
|
impactSound.Play();
|
|
impactSound.SetScheduledEndTime(AudioSettings.dspTime + (toSeconds - fromSeconds));
|
|
}
|
|
}
|
|
}
|
|
}
|