User:Forte/解析メモ/ダメージ: Difference between revisions
< User:Forte | 解析メモ
(ダメージ計算のコード解読) |
(No difference)
|
Revision as of 13:57, 26 December 2024
public void DamageHP(
int dmg,
int ele,
int eleP = 100,
AttackSource attackSource = AttackSource.None,
Card origin = null,
bool showEffect = true)
{
if (this.hp < 0)
return;
Element e;
if (ele == 0)
{
e = Element.Void;
}
else
{
e = Element.Create(ele);
if (attackSource != AttackSource.Condition & showEffect)
ActEffect.TryDelay((Action) (() =>
{
this.PlayEffect(e.id, range: 0.25f);
EClass.Sound.Play("atk_" + e.source.alias);
}));
if (!e.source.aliasRef.IsEmpty() && attackSource != AttackSource.ManaBackfire)
{
dmg = Element.GetResistDamage(dmg, this.Evalue(e.source.aliasRef));
dmg = dmg * 100 / (100 + Mathf.Clamp(this.Evalue(961) * 5, -50, 200));
}
switch (e.id)
{
case 910:
Chara chara1 = this.Chara;
if ((chara1 != null ? (chara1.isWet ? 1 : 0) : 0) != 0)
{
dmg /= 3;
break;
}
break;
case 912:
Chara chara2 = this.Chara;
if ((chara2 != null ? (chara2.isWet ? 1 : 0) : 0) != 0)
{
dmg = dmg * 150 / 100;
break;
}
break;
}
}
if (attackSource != AttackSource.Finish)
{
if (!this.IsPCFaction && this.LV > 50)
dmg = dmg * (100 - (int) Mathf.Min(80f, Mathf.Sqrt((float) (this.LV - 50)) * 2.5f)) / 100;
dmg = dmg * Mathf.Max(0, 100 - this.Evalue(e == Element.Void ? 55 : 56)) / 100;
if (origin != null && origin.IsPC && EClass.player.codex.Has(this.id))
dmg = dmg * (100 + Mathf.Min(10, EClass.player.codex.GetOrCreate(this.id).weakspot)) / 100;
if (this.isChara && this.Chara.body.GetAttackStyle() == AttackStyle.Shield && this.elements.ValueWithoutLink(123) >= 5 && e == Element.Void)
dmg = dmg * 90 / 100;
if (this.Evalue(971) > 0)
dmg = dmg * 100 / Mathf.Clamp(100 + this.Evalue(971), 25, 1000);
if (this.HasElement(1305))
dmg = dmg * 90 / 100;
if (EClass.pc.HasElement(1207) && this.isChara)
{
int num1 = 0;
int num2 = 0;
foreach (Condition condition in this.Chara.conditions)
{
if (condition.Type == ConditionType.Buff)
++num1;
else if (condition.Type == ConditionType.Debuff)
++num2;
}
dmg = !this.IsPCParty ? dmg * Mathf.Min(100 + num2 * 5, 120) / 100 : dmg * 100 / Mathf.Min(100 + num1 * 5, 120);
}
if (this.IsPCParty && EClass.pc.ai is GoalAutoCombat)
dmg = dmg * 100 / Mathf.Min(105 + EClass.pc.Evalue(135) / 10, 110);
if (this.HasElement(1218))
{
dmg = dmg * (1000 - this.Evalue(1218)) / 1000;
if (dmg <= 0 && EClass.rnd(4) == 0)
dmg++;
}
if (dmg >= this.MaxHP / 10 && this.Evalue(68) > 0)
{
int num3 = this.MaxHP / 10;
int num4 = (dmg - num3) * 100 / (200 + this.Evalue(68) * 10);
dmg = num3 + num4;
}
}
if (origin != null && origin.IsPC && EClass.pc.Evalue(654) > 0)
dmg = 0;
if (dmg < 0)
dmg = 0;
int num5 = Mathf.Clamp(dmg * 6 / this.MaxHP, 0, 4) + (dmg > 0 ? 1 : 0);
if (this.Evalue(1421) > 0)
{
int num6 = 0;
int num7 = dmg;
if (this.hp > 0)
{
num7 = dmg - this.hp;
this.hp -= dmg;
num6 += dmg;
if (this.hp < 0 && this.Chara.mana.value >= 0)
{
num6 += this.hp;
this.hp = 0;
}
}
if (this.hp <= 0)
{
if (this.Evalue(1421) >= 2)
num7 /= 2;
dmg = num7;
if (this.Chara.mana.value > 0)
{
num7 -= this.Chara.mana.value;
this.Chara.mana.value -= dmg;
num6 += dmg;
}
if (this.Chara.mana.value <= 0)
{
this.hp -= num7;
num6 += num7;
}
}
dmg = num6;
}
else
this.hp -= dmg;
if (this.isSynced && dmg != 0)
{
float ratio = (float) dmg / (float) this.MaxHP;
Card c = this.parent is Chara ? (Card) (this.parent as Chara) : this;
ActEffect.TryDelay((Action) (() =>
{
c.PlayEffect("blood").SetParticleColor(EClass.Colors.matColors[this.material.alias].main).Emit(20 + (int) (30.0 * (double) ratio));
if (!EClass.core.config.test.showNumbers && !this.isThing)
return;
Popper popper = EClass.scene.popper.Pop(this.renderer.PositionCenter(), "DamageNum");
Color color = c.IsPC ? EClass.Colors.textColors.damagePC : (c.IsPCFaction ? EClass.Colors.textColors.damagePCParty : EClass.Colors.textColors.damage);
if (e != Element.Void)
{
color = EClass.Colors.elementColors.TryGetValue<string, Color>(e.source.alias);
float num8 = (float) (((double) color.r + (double) color.g + (double) color.b) / 3.0);
float num9 = (double) num8 > 0.5 ? 0.0f : 0.6f - num8;
color = new Color(color.r + num9, color.g + num9, color.b + num9, 1f);
}
string s = dmg.ToString() ?? "";
Color c1 = color;
popper.SetText(s, c1);
}));
}
ZoneInstanceBout instance = EClass._zone.instance as ZoneInstanceBout;
if (this.hp < 0 && Religion.recentWrath == null)
{
if (this.isRestrained && this.IsPCFaction && EClass._zone.IsPCFaction && (!this.IsPC || this.Chara.ai is AI_Torture && this.Chara.ai.IsRunning))
{
EvadeDeath();
if (this.Chara.stamina.value > 0 && (EClass.rnd(2) == 0 || !this.IsPC))
this.Chara.stamina.Mod(-1);
}
else if (this.IsInstalled && this.pos.HasBlock && this.trait.IsDoor)
EvadeDeath();
else if (!this.trait.CanBeDestroyed)
EvadeDeath();
else if (this.HasEditorTag(EditorTag.Invulnerable) || this.HasEditorTag(EditorTag.InvulnerableToMobs) && (origin == null || !origin.IsPCParty))
EvadeDeath();
else if (this.isChara)
{
if (this.Chara.HasCondition<ConInvulnerable>())
EvadeDeath();
else if (this.IsPC && EClass.debug.godMode)
EvadeDeath();
else if (this.Chara.host != null)
EvadeDeath();
else if (instance != null && (bool) (UnityEngine.Object) LayerDrama.Instance)
EvadeDeath();
else if (LayerDrama.IsActive() && this.IsPC)
{
EvadeDeath();
}
else
{
if (attackSource != AttackSource.Finish && this.IsPCParty && this.Chara.host == null && EClass.pc.ai is GoalAutoCombat)
{
if (!EClass.player.invlunerable && (EClass.pc.ai as GoalAutoCombat).listHealthy.Contains(this.Chara))
{
EClass.core.actionsNextFrame.Add((Action) (() => Msg.Say(this.IsPC ? "abort_damage" : "abort_damgeAlly")));
EClass.player.invlunerable = true;
EClass.player.TryAbortAutoCombat();
EClass.pc.stamina.Mod(-EClass.pc.stamina.max / 5);
}
if (EClass.player.invlunerable)
{
EvadeDeath();
goto label_88;
}
}
if (this.IsPC && this.Evalue(1220) > 0 && this.Chara.stamina.value >= this.Chara.stamina.max / 2)
{
this.Chara.stamina.Mod(-this.Chara.stamina.max / 2);
this.Chara.AddCondition<ConInvulnerable>();
EvadeDeath();
}
}
}
}
label_88:
if (this.trait.CanBeAttacked)
{
this.renderer.PlayAnime(AnimeID.HitObj);
this.hp = this.MaxHP;
}
if (this.hp < 0)
{
if ((attackSource == AttackSource.Melee || attackSource == AttackSource.Range) && origin != null && (origin.isSynced || this.IsPC))
{
string str = "";
if (this.IsPC && Lang.setting.combatTextStyle == 0)
{
if (e != Element.Void && e != null)
str = "dead_" + e.source.alias;
if (str == "" || !LangGame.Has(str))
str = "dead_attack";
EClass.pc.Say(str, this, "");
}
else
{
if (e != Element.Void && e != null)
str = "kill_" + e.source.alias;
if (str == "" || !LangGame.Has(str))
str = "kill_attack";
(this.IsPC ? (Card) EClass.pc : origin).Say(str, origin, this);
}
}
if (this.isChara && Religion.recentWrath == null)
{
if (this.HasElement(1410) && !this.Chara.HasCooldown(1410))
{
this.Chara.AddCooldown(1410);
this.Say("reboot", this);
this.PlaySound("reboot");
this.Chara.Cure(CureType.Boss);
this.hp = this.MaxHP / 3;
this.Chara.AddCondition<ConInvulnerable>();
return;
}
foreach (Chara chara in EClass._map.charas)
{
if (this.Chara.IsFriendOrAbove(chara) && chara.HasElement(1408) && chara.faith == EClass.game.religions.Healing && EClass.world.date.GetRawDay() != chara.GetInt(58) && (!chara.IsPCFaction || this.IsPCFaction))
{
Msg.alwaysVisible = true;
Msg.Say("layhand", (Card) chara, this);
Msg.Say("pray_heal", this);
this.hp = this.MaxHP;
this.Chara.AddCondition<ConInvulnerable>();
this.PlayEffect("revive");
this.PlaySound("revive");
chara.SetInt(58, EClass.world.date.GetRawDay());
return;
}
}
}
if (instance != null)
{
Chara target = EClass._map.FindChara(instance.uidTarget);
if (target != null)
{
if (this.IsPC)
{
EClass.pc.hp = 0;
Heal();
target.ShowDialog("_chara", "bout_lose");
return;
}
if (target == this)
{
this.hp = 0;
Heal();
target.ModAffinity(EClass.pc, 10);
target.ShowDialog("_chara", "bout_win");
return;
}
}
void Heal()
{
target.Cure(CureType.Death);
foreach (Chara member in EClass.pc.party.members)
member.Cure(CureType.Death);
}
}
if (!this.isDestroyed)
this.Die(e, origin, attackSource);
if (origin != null && origin.isChara)
{
if (origin.IsPCFactionOrMinion && this.isChara && !this.isCopy)
{
++EClass.player.stats.kills;
EClass.game.quests.list.ForeachReverse<Quest>((Action<Quest>) (q => q.OnKillChara(this.Chara)));
EClass.player.codex.AddKill(this.id);
if (Guild.Fighter.CanGiveContribution(this.Chara))
Guild.Fighter.AddContribution(5 + this.LV / 5);
if (Guild.Fighter.HasBounty(this.Chara))
{
int a = EClass.rndHalf(200 + this.LV * 20);
Msg.Say("bounty", (Card) this.Chara, a.ToString() ?? "");
EClass.pc.ModCurrency(a);
SE.Pay();
}
}
if (origin.IsPCParty && EClass.pc.Evalue(1355) > 0)
{
if (!(EClass.pc.AddCondition<ConStrife>() is ConStrife conStrife))
conStrife = EClass.pc.GetCondition<ConStrife>();
conStrife?.AddKill();
}
if (origin.GetInt(106) == 0)
origin.Chara.TalkTopic("kill");
}
Msg.SetColor();
}
else if ((attackSource == AttackSource.Melee || attackSource == AttackSource.Range) && origin != null)
(this.IsPC ? (Card) EClass.pc : origin).Say("dmgMelee" + num5.ToString() + (this.IsPC ? "pc" : ""), origin, this);
else if (this.isChara)
{
int num10 = attackSource == AttackSource.Condition || attackSource == AttackSource.WeaponEnchant ? 2 : 1;
if (num5 >= num10)
{
if (e != Element.Void)
this.Say("dmg_" + e.source.alias, this);
if (e == Element.Void || num5 >= 2)
this.Say(nameof (dmg) + num5.ToString(), this);
}
}
if (this.isChara && origin != null && origin.IsAliveInCurrentZone && origin.isChara)
{
if (e.id == 916)
origin.HealHP(Mathf.Clamp(EClass.rnd(dmg * (50 + eleP) / 500 + 5), 1, origin.MaxHP / 5 + EClass.rnd(10)));
if ((attackSource == AttackSource.Melee || attackSource == AttackSource.Range) && origin.Dist(this) <= 1)
{
if (attackSource == AttackSource.Melee && this.HasElement(1221))
{
int ele1 = this.Chara.MainElement == Element.Void ? 924 : this.Chara.MainElement.id;
if (this.id == "hedgehog_ether")
ele1 = 922;
this.Say("reflect_thorne", this, origin);
origin.DamageHP(Mathf.Clamp(dmg / 20, 1, this.MaxHP / 20), ele1, this.Power, AttackSource.Condition);
}
if (this.HasElement(1223))
{
int ele2 = this.Chara.MainElement == Element.Void ? 923 : this.Chara.MainElement.id;
this.Say("reflect_acid", this, origin);
origin.DamageHP(Mathf.Clamp(dmg / 20, 1, this.MaxHP / 20), ele2, this.Power * 2, AttackSource.Condition);
}
}
if (origin.HasElement(662) && attackSource == AttackSource.Melee && origin.isChara && this.Chara.IsHostile(origin as Chara))
{
int a = EClass.rnd(3 + Mathf.Clamp(dmg / 100, 0, origin.Evalue(662) / 10));
origin.Chara.stamina.Mod(a);
if (this.IsAliveInCurrentZone)
this.Chara.stamina.Mod(-a);
}
if (origin.HasElement(1350) && attackSource == AttackSource.Melee)
{
int a = EClass.rndHalf(2 + Mathf.Clamp(dmg / 10, 0, origin.Chara.GetPietyValue() + 10));
origin.Chara.mana.Mod(a);
if (this.IsAliveInCurrentZone)
this.Chara.mana.Mod(-a);
}
if (origin.HasElement(661) && attackSource == AttackSource.Melee)
{
int a = EClass.rnd(2 + Mathf.Clamp(dmg / 10, 0, origin.Evalue(661) + 10));
origin.Chara.mana.Mod(a);
if (this.IsAliveInCurrentZone)
this.Chara.mana.Mod(-a);
}
}
if (this.hp < 0 || !this.isChara)
return;
if (dmg > 0)
{
int a = Mathf.Min(100 * (dmg * 100 / this.MaxHP) / 100, this.Chara.isRestrained ? 15 : 200);
this.elements.ModExp(this.GetArmorSkill(), a);
if (this.Chara.body.GetAttackStyle() == AttackStyle.Shield)
this.elements.ModExp(123, a);
}
int a1 = EClass.rnd(2) == 0 ? 1 : 0;
if (attackSource == AttackSource.Condition)
a1 = 1 + EClass.rnd(2);
if (a1 > 0)
{
bool flag = this.Chara.HasCondition<ConPoison>() || (e.id == 915 || e.id == 923) && this.ResistLv(this.Evalue(955)) < 4;
this.AddBlood(a1, flag ? 6 : -1);
}
bool flag1 = true;
switch (e.id)
{
case 910:
if (Chance(30 + eleP / 5, 100))
{
this.Chara.AddCondition<ConBurning>(eleP);
break;
}
break;
case 911:
if (this.Chara.isWet)
{
if (Chance(30 + eleP / 10, 100))
{
this.Chara.AddCondition<ConFreeze>(eleP);
break;
}
break;
}
if (Chance(50 + eleP / 10, 100))
{
this.Chara.AddCondition<ConWet>(eleP);
break;
}
break;
case 912:
if (Chance(75 + eleP / 20, 100) && EClass.rnd(3) == 0)
{
this.Chara.AddCondition<ConParalyze>(1, true);
break;
}
break;
case 913:
if (Chance(30 + eleP / 5, 100))
{
this.Chara.AddCondition<ConBlind>(eleP);
break;
}
break;
case 914:
flag1 = false;
if (EClass.rnd(3) != 0)
{
if (Chance(30 + eleP / 5, 100))
{
this.Chara.AddCondition<ConConfuse>(eleP);
break;
}
break;
}
if (Chance(30 + eleP / 5, 100))
{
this.Chara.AddCondition<ConSleep>(eleP);
break;
}
break;
case 915:
if (Chance(30 + eleP / 5, 100))
{
this.Chara.AddCondition<ConPoison>(eleP);
break;
}
break;
case 917:
if (Chance(50 + eleP / 10, 100))
{
this.Chara.AddCondition<ConDim>(eleP);
break;
}
break;
case 918:
flag1 = false;
if (Chance(30 + eleP / 5, 100))
{
this.Chara.AddCondition<ConParalyze>(eleP);
break;
}
break;
case 920:
flag1 = false;
if (Chance(5 + eleP / 25, 40))
this.Chara.AddCondition<ConBlind>(eleP / 2);
if (Chance(5 + eleP / 25, 40))
this.Chara.AddCondition<ConParalyze>(eleP / 2);
if (Chance(5 + eleP / 25, 40))
this.Chara.AddCondition<ConConfuse>(eleP / 2);
if (Chance(5 + eleP / 25, 40))
this.Chara.AddCondition<ConPoison>(eleP / 2);
if (Chance(5 + eleP / 25, 40))
this.Chara.AddCondition<ConSleep>(eleP / 2);
if (Chance(5 + eleP / 25, 40))
this.Chara.AddCondition<ConDim>(eleP / 2);
if (Chance(30 + eleP / 10, 100))
{
this.Chara.SAN.Mod(EClass.rnd(2));
break;
}
break;
case 922:
this.Chara.ModCorruption(EClass.rnd(eleP / 50 + 10));
break;
case 923:
if (Chance(50 + eleP / 10, 100) && EClass.rnd(4) == 0)
{
ActEffect.Proc(EffectId.Acid, (Card) this.Chara);
break;
}
break;
case 924:
if (Chance(50 + eleP / 10, 100))
{
this.Chara.AddCondition<ConBleed>(eleP);
break;
}
break;
case 925:
if (EClass.rnd(3) == 0)
{
if (Chance(30 + eleP / 5, 100))
{
this.Chara.AddCondition<ConDim>(eleP);
break;
}
break;
}
if (EClass.rnd(2) == 0)
{
if (EClass.rnd(3) == 0)
{
this.Chara.AddCondition<ConParalyze>(1, true);
break;
}
break;
}
if (EClass.rnd(2) == 0)
{
this.Chara.AddCondition<ConConfuse>(1 + EClass.rnd(3), true);
break;
}
break;
}
if (origin != null && origin.HasElement(1411) && !this.Chara.HasCondition<ConGravity>())
{
Condition.ignoreEffect = true;
this.Chara.AddCondition<ConGravity>(2000);
Condition.ignoreEffect = false;
}
if (this.Chara.conSleep != null & flag1)
this.Chara.conSleep.Kill();
if (this.Chara.host != null && this.hp == 0 && !this.Chara.HasCondition<ConFaint>())
this.Chara.AddCondition<ConFaint>(200, true);
if (this.IsPC)
{
float num11 = (float) this.hp / (float) this.MaxHP;
if (this.Evalue(1421) > 0)
num11 = (float) this.Chara.mana.value / (float) this.Chara.mana.max;
if ((double) num11 < 0.30000001192092896)
this.PlaySound("heartbeat", (float) (1.0 - (double) num11 * 2.0));
}
if (!this.IsPC && this.hp < this.MaxHP / 5 && this.Evalue(423) <= 0 && dmg * 100 / this.MaxHP + 10 > EClass.rnd(this.IsPowerful ? 400 : 150) && !this.HasCondition<ConFear>())
this.Chara.AddCondition<ConFear>(100 + EClass.rnd(100));
if (this.Chara.ai.Current.CancelWhenDamaged && attackSource != AttackSource.Hunger && attackSource != AttackSource.Fatigue)
this.Chara.ai.Current.TryCancel(origin);
if (this.Chara.HasCondition<ConWeapon>())
{
ConWeapon condition = this.Chara.GetCondition<ConWeapon>();
if (e.source.aliasRef == condition.sourceElement.aliasRef)
condition.Mod(-1);
}
if (this.Chara.HasElement(1222) && (dmg >= this.MaxHP / 10 || EClass.rnd(20) == 0))
ActEffect.Proc(EffectId.Duplicate, this);
if (this.hp < this.MaxHP / 3 && this.HasElement(1409) && !this.Chara.HasCooldown(1409))
{
this.Chara.AddCooldown(1409);
this.Chara.AddCondition<ConBoost>(this.Power);
this.Chara.Cure(CureType.Boss);
this.Chara.HealHP(this.MaxHP / 2);
}
if (origin == null || !origin.isChara || attackSource == AttackSource.Finish)
return;
if (!AI_PlayMusic.ignoreDamage)
this.Chara.TrySetEnemy(origin.Chara);
if (origin.Evalue(428) <= 0 || this.IsPCFactionOrMinion || EClass.rnd(dmg) < EClass.rnd(this.MaxHP / 10) + this.MaxHP / 100 + 1)
return;
origin.Chara.TryNeckHunt(this.Chara, origin.Evalue(428) * 20, true);
void EvadeDeath()
{
this.hp = 0;
if (this.Evalue(1421) <= 0 || !this.isChara || this.Chara.mana.value >= 0)
return;
this.Chara.mana.value = 0;
}
bool Chance(int a, int max)
{
return (dmg > 0 || origin != null && origin.HasElement(1345)) && Mathf.Min(a, max) > EClass.rnd(100);
}
}