using UnityEngine; using Unity.Netcode; namespace Colosseum.Skills { /// /// 스킬 투사체. 충돌 시 연결된 효과를 적용합니다. /// 서버에서만 이동/충돌을 처리하며, NetworkTransform으로 위치를 동기화합니다. /// [RequireComponent(typeof(Rigidbody))] public class SkillProjectile : NetworkBehaviour { [Header("이동 설정")] [Min(0f)] [SerializeField] private float speed = 15f; [Min(0f)] [SerializeField] private float lifetime = 5f; [Header("관통 설정")] [SerializeField] private bool penetrate = false; [SerializeField] private int maxPenetration = 1; [Header("충돌 이펙트")] [SerializeField] private GameObject hitEffect; [SerializeField] private float hitEffectDuration = 2f; [Header("트레일 이펙트")] [Tooltip("투사체 트레일 프리팹 (TrailRenderer가 포함된 프리팹). 미설정 시 트레일 없음.")] [SerializeField] private GameObject trailPrefab; [Tooltip("투사체 파괴 후 트레일이 남는 시간 (초)")] [Min(0.1f)] [SerializeField] private float trailDuration = 0.3f; private GameObject caster; private SkillEffect sourceEffect; private int penetrationCount; private Rigidbody rb; private bool initialized; private GameObject trailInstance; /// /// 투사체 초기화 /// public void Initialize(GameObject caster, SkillEffect sourceEffect) { this.caster = caster; this.sourceEffect = sourceEffect; initialized = true; rb = GetComponent(); if (rb != null) { rb.useGravity = false; rb.collisionDetectionMode = CollisionDetectionMode.ContinuousDynamic; } // caster 및 자식 콜라이더와의 충돌 무시 var myColliders = GetComponents(); foreach (var cc in caster.GetComponentsInChildren()) foreach (var mc in myColliders) Physics.IgnoreCollision(mc, cc); } public override void OnNetworkSpawn() { // 트레일 생성 (서버/클라이언트 양쪽에서 실행) if (trailPrefab != null) { trailInstance = Instantiate(trailPrefab, transform); trailInstance.transform.localPosition = Vector3.zero; trailInstance.transform.localRotation = Quaternion.identity; } } private void Start() { // 서버에서만 수명 관리 if (IsServer) Invoke(nameof(ServerDespawn), lifetime); } private void FixedUpdate() { // 서버에서만 이동 처리 if (!IsServer || !initialized || rb == null) return; rb.linearVelocity = transform.forward * speed; } private void OnTriggerEnter(Collider other) { // 서버에서만 충돌 처리 if (!IsServer || !initialized || sourceEffect == null) return; if (other.gameObject == caster) return; // 유효한 타겟인지 확인 if (!sourceEffect.IsValidTarget(caster, other.gameObject)) return; // 충돌 이펙트 (서버 + 클라이언트 모두 표시) if (hitEffect != null) { var effect = Instantiate(hitEffect, transform.position, transform.rotation); Destroy(effect, hitEffectDuration); HitEffectClientRpc(transform.position, transform.rotation.eulerAngles); } // 효과 적용 sourceEffect.ExecuteOnHit(caster, other.gameObject); penetrationCount++; if (!penetrate || penetrationCount >= maxPenetration) { ServerDespawn(); } } private void ServerDespawn() { if (!IsServer || !IsSpawned) return; // 트레일을 월드에 남겨서 자연스럽게 사라지게 함 if (trailInstance != null) { trailInstance.transform.SetParent(null); Destroy(trailInstance, trailDuration); trailInstance = null; } NetworkObject.Despawn(true); } public void SetDirection(Vector3 direction) { transform.rotation = Quaternion.LookRotation(direction.normalized); } /// /// 클라이언트에 충돌 이펙트 표시 /// [Rpc(SendTo.NotServer)] private void HitEffectClientRpc(Vector3 position, Vector3 eulerAngles) { if (hitEffect != null) { var effect = Instantiate(hitEffect, position, Quaternion.Euler(eulerAngles)); Destroy(effect, hitEffectDuration); } } } }