Files
Colosseum/Assets/_Game/Scripts/Skills/Effects/TauntEffect.cs
dal4segno aaa7d2d6a7 feat: 보호막 타입 분리 및 드로그 시그니처 전조 정리
- 보호막을 단일 수치에서 타입별 독립 인스턴스 구조로 리팩터링하고 같은 타입만 갱신되도록 정리
- 플레이어/보스 보호막 상태를 이상상태와 연동해 HUD 및 보스 UI에서 타입별로 식별 가능하게 보강
- 드로그 집행 개시 전조를 집행 준비 이상상태 기반으로 재구성하고 관련 데이터와 보스 컨텍스트를 정리
- 전투 밸런스 계측기와 디버그 메뉴를 추가해 피해, 치유, 보호막, 위협, 패턴 사용량 측정 경로를 마련
- 테스트용 보호막 A/B와 시그니처 전조 자산을 추가하고 기본 포트 7777 원복 후 빌드 및 런타임 검증을 완료
2026-03-26 11:19:19 +09:00

89 lines
3.4 KiB
C#

using System.Collections.Generic;
using UnityEngine;
using Colosseum.Combat;
using Colosseum.Enemy;
namespace Colosseum.Skills.Effects
{
/// <summary>
/// 주변 적의 위협 수치를 높여 주 대상을 자신에게 돌리는 도발 효과입니다.
/// 필요 시 시전자에게 임시 위협 생성 배율도 함께 부여합니다.
/// </summary>
[CreateAssetMenu(fileName = "TauntEffect", menuName = "Colosseum/Skills/Effects/Taunt")]
public class TauntEffect : SkillEffect
{
[Header("Taunt Settings")]
[Tooltip("도발 시 추가할 기본 위협 수치")]
[Min(0f)] [SerializeField] private float flatThreatAmount = 50f;
[Tooltip("현재 최고 위협보다 이 값만큼 더 높게 설정합니다.")]
[Min(0f)] [SerializeField] private float threatLeadBonus = 10f;
[Tooltip("도발과 함께 시전자에게 부여할 위협 생성 배율")]
[Min(0f)] [SerializeField] private float selfThreatMultiplier = 1.5f;
[Tooltip("위협 생성 배율 지속 시간")]
[Min(0f)] [SerializeField] private float selfThreatMultiplierDuration = 5f;
private readonly Collider[] overlapBuffer = new Collider[16];
private readonly HashSet<EnemyBase> processedEnemies = new HashSet<EnemyBase>();
protected override void ApplyEffect(GameObject caster, GameObject target)
{
if (caster == null)
return;
ApplySelfThreatMultiplier(caster);
processedEnemies.Clear();
int hitCount = Physics.OverlapSphereNonAlloc(caster.transform.position, areaRadius, overlapBuffer, targetLayers);
for (int i = 0; i < hitCount; i++)
{
Collider hit = overlapBuffer[i];
if (hit == null)
continue;
EnemyBase enemy = hit.GetComponentInParent<EnemyBase>();
if (enemy == null || processedEnemies.Contains(enemy))
continue;
if (!IsValidTarget(caster, enemy.gameObject))
continue;
processedEnemies.Add(enemy);
ApplyTauntThreat(enemy, caster);
}
}
private void ApplySelfThreatMultiplier(GameObject caster)
{
if (selfThreatMultiplier <= 0f || selfThreatMultiplierDuration <= 0f)
return;
ThreatController threatController = caster.GetComponent<ThreatController>();
if (threatController == null)
{
threatController = caster.AddComponent<ThreatController>();
}
threatController.ApplyThreatMultiplier(selfThreatMultiplier, selfThreatMultiplierDuration);
}
private void ApplyTauntThreat(EnemyBase enemy, GameObject caster)
{
if (enemy == null || caster == null || !enemy.UseThreatSystem)
return;
GameObject highestThreatTarget = enemy.GetHighestThreatTarget();
float highestThreat = highestThreatTarget != null ? enemy.GetThreat(highestThreatTarget) : 0f;
float currentCasterThreat = enemy.GetThreat(caster);
float desiredThreat = Mathf.Max(currentCasterThreat + flatThreatAmount, highestThreat + threatLeadBonus + flatThreatAmount);
enemy.SetThreat(caster, desiredThreat);
CombatBalanceTracker.RecordThreat(caster, Mathf.Max(0f, desiredThreat - currentCasterThreat));
}
}
}