- 애니메이션 이벤트 기반 방어/유지/해제 흐름과 HUD 피드백, 방어 디버그 로그를 추가했다. - 드로그 기본기1 테스트 패턴을 정리하고 공격 판정을 OnEffect 기반으로 옮기며 드로그 범위 효과의 타겟 레이어를 정상화했다. - 플레이어 퀵슬롯 테스트 세팅과 적-플레이어 겹침 방지 로직을 조정해 충돌 시 적이 수평 이동을 멈추고 최소 분리만 수행하게 했다.
105 lines
3.6 KiB
C#
105 lines
3.6 KiB
C#
using UnityEngine;
|
|
|
|
using Colosseum.Stats;
|
|
using Colosseum.Combat;
|
|
using Colosseum.Passives;
|
|
using Colosseum.Skills;
|
|
using Colosseum.Weapons;
|
|
|
|
namespace Colosseum.Skills.Effects
|
|
{
|
|
/// <summary>
|
|
/// 대미지 타입
|
|
/// </summary>
|
|
public enum DamageType
|
|
{
|
|
Physical, // 물리 대미지 (STR 기반)
|
|
Magical, // 마법 대미지 (INT 기반)
|
|
Ranged, // 원거리 대미지 (DEX 기반)
|
|
True, // 고정 대미지 (스탯 영향 없음)
|
|
}
|
|
|
|
/// <summary>
|
|
/// 데미지 효과
|
|
/// </summary>
|
|
[CreateAssetMenu(fileName = "DamageEffect", menuName = "Colosseum/Skills/Effects/Damage")]
|
|
public class DamageEffect : SkillEffect
|
|
{
|
|
[Header("Damage Settings")]
|
|
[Min(0f)] [SerializeField] private float baseDamage = 10f;
|
|
[SerializeField] private DamageType damageType = DamageType.Physical;
|
|
[Tooltip("스탯 계수 (1.0 = 100%)")]
|
|
[Min(0f)] [SerializeField] private float statScaling = 1f;
|
|
|
|
[Header("Mitigation")]
|
|
[Tooltip("이 피해가 방어/회피 규칙에서 어떤 판정으로 처리되는지 설정합니다.")]
|
|
[SerializeField] private DamageMitigationTier mitigationTier = DamageMitigationTier.Normal;
|
|
|
|
public float BaseDamage => baseDamage;
|
|
public DamageType DamageKind => damageType;
|
|
public float StatScaling => statScaling;
|
|
|
|
protected override void ApplyEffect(GameObject caster, GameObject target)
|
|
{
|
|
if (target == null) return;
|
|
|
|
// 대미지 계산
|
|
float totalDamage = CalculateDamage(caster);
|
|
|
|
// 타겟에 대미지 적용 (IDamageable 인터페이스 사용)
|
|
var damageable = target.GetComponent<IDamageable>();
|
|
if (damageable != null)
|
|
{
|
|
damageable.TakeDamage(new DamageContext(totalDamage, caster, mitigationTier));
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// 시전자 스탯 기반 대미지 계산
|
|
/// 공식: baseDamage + (statDamage * scaling)
|
|
/// </summary>
|
|
private float CalculateDamage(GameObject caster)
|
|
{
|
|
if (damageType == DamageType.True)
|
|
{
|
|
return baseDamage;
|
|
}
|
|
|
|
var stats = caster.GetComponent<CharacterStats>();
|
|
if (stats == null)
|
|
{
|
|
return baseDamage;
|
|
}
|
|
|
|
float statDamage = damageType switch
|
|
{
|
|
DamageType.Physical => stats.PhysicalDamage,
|
|
DamageType.Magical => stats.MagicDamage,
|
|
DamageType.Ranged => stats.Dexterity.FinalValue * 2f, // DEX 기반 원거리 대미지
|
|
_ => 0f,
|
|
};
|
|
|
|
float baseTotal = baseDamage + (statDamage * statScaling);
|
|
|
|
// 무기 데미지 배율 적용
|
|
float damageMultiplier = GetDamageMultiplier(caster);
|
|
float gemMultiplier = SkillRuntimeModifierUtility.GetDamageMultiplier(caster);
|
|
float passiveMultiplier = PassiveRuntimeModifierUtility.GetDamageMultiplier(caster);
|
|
return baseTotal * damageMultiplier * gemMultiplier * passiveMultiplier;
|
|
}
|
|
|
|
/// <summary>
|
|
/// 시전자의 무기 데미지 배율 조회
|
|
/// </summary>
|
|
private float GetDamageMultiplier(GameObject caster)
|
|
{
|
|
var weaponEquipment = caster.GetComponent<WeaponEquipment>();
|
|
if (weaponEquipment != null)
|
|
{
|
|
return weaponEquipment.DamageMultiplier;
|
|
}
|
|
return 1f;
|
|
}
|
|
}
|
|
}
|