using System; using UnityEngine; // ============================================================================ // Health // ---------------------------------------------------------------------------- // 순수한 HP 데이터 컴포넌트. 시각 효과/넉백/사망 처리는 일절 안 함. // Enemy, Player 등 어떤 GameObject든 Health 컴포넌트만 추가하면 HP 관리 가능. // // 일반적 사용 흐름: // 1) Enemy/Player가 IDamageable.TakeDamage 받음 // 2) 시각 효과(flash) + 넉백 처리 // 3) 마지막에 _health.TakeDamage(amount) 호출 // 4) Health가 HP 감소 + OnHealthChanged 이벤트 발화 → HP바 등 UI 자동 갱신 // 5) HP가 0 되면 OnDied 이벤트 발화 → Enemy가 HandleDeath()로 Destroy 등 처리 // ============================================================================ public class Health : MonoBehaviour { [SerializeField] private int _maxHealth = 30; private int _currentHealth; // ─── 읽기 전용 프로퍼티 ────────────────────────────────────────────── public int MaxHealth => _maxHealth; public int CurrentHealth => _currentHealth; public float Ratio => _maxHealth > 0 ? (float)_currentHealth / _maxHealth : 0f; public bool IsDead => _currentHealth <= 0; // ─── 외부 구독용 이벤트 ────────────────────────────────────────────── // OnHealthChanged: (current, max). HP바, 데미지 숫자 표시 등에 사용. // OnDied: 사망 순간 1회만 발화. Enemy.HandleDeath에서 구독. public event Action OnHealthChanged; public event Action OnDied; private void Awake() { _currentHealth = _maxHealth; } // 데미지 적용. 양수 데미지만 받고, 이미 죽었으면 무시. // OnDied는 "방금 죽은 순간"에만 발화 (previous > 0 && current == 0). public void TakeDamage(int amount) { if (amount <= 0 || IsDead) return; int previous = _currentHealth; _currentHealth = Mathf.Max(_currentHealth - amount, 0); OnHealthChanged?.Invoke(_currentHealth, _maxHealth); if (_currentHealth <= 0 && previous > 0) OnDied?.Invoke(); } // 회복. 죽은 상태에서는 회복 안 됨 (부활 로직은 별도로 만들어야 함). public void Heal(int amount) { if (amount <= 0 || IsDead) return; _currentHealth = Mathf.Min(_currentHealth + amount, _maxHealth); OnHealthChanged?.Invoke(_currentHealth, _maxHealth); } // 풀체력으로 리셋. 부활/재시작 시 사용. public void ResetHealth() { _currentHealth = _maxHealth; OnHealthChanged?.Invoke(_currentHealth, _maxHealth); } // 최대 HP 변경. fill=true면 현재 HP도 풀로 채우고, false면 새 max로 클램프만. public void SetMaxHealth(int newMax, bool fill = true) { _maxHealth = Mathf.Max(newMax, 1); if (fill) _currentHealth = _maxHealth; else _currentHealth = Mathf.Min(_currentHealth, _maxHealth); OnHealthChanged?.Invoke(_currentHealth, _maxHealth); } }