Files
bl/logic/service/fight/input/fight.go

339 lines
8.3 KiB
Go
Raw Normal View History

package input
import (
element "blazing/common/data/Element"
"blazing/common/utils"
2025-11-15 22:17:43 +00:00
2025-09-29 02:40:35 +08:00
"blazing/logic/service/fight/action"
"blazing/logic/service/fight/info"
"github.com/alpacahq/alpacadecimal"
"github.com/gogf/gf/v2/util/gconv"
"github.com/gogf/gf/v2/util/grand"
)
// 计算暴击
2025-11-11 05:54:24 +00:00
func (our *Input) CalculateCrit(opp *Input, skill *info.SkillEntity) {
ourPet := our.CurrentPet()
oppPet := opp.CurrentPet()
if ourPet == nil || oppPet == nil {
return
}
skill.Crit = 0
if skill.Category() == info.Category.STATUS { //属性技能不用算暴击
return
}
CritRate := utils.Max(skill.XML.CritRate, 1)
//CritAtkFirst: 先出手时必定致命一击; 默认: 0
if skill.XML.CritAtkFirst != 0 && our.FightC.IsFirst(our.Player) {
CritRate = 16
}
//CritAtkSecond: 后出手时必定致命一击; 默认: 0
if skill.XML.CritAtkSecond != 0 && !our.FightC.IsFirst(our.Player) {
CritRate = 16
}
// CritSelfHalfHp: 自身体力低于一半时必定致命一击; 默认: 0
if skill.XML.CritSelfHalfHp != 0 && (ourPet.HP < int(ourPet.Info.MaxHp)/2) {
CritRate = 16
}
// CritFoeHalfHp: 对方体力低于一半时必定致命一击; 默认: 0
if skill.XML.CritSelfHalfHp != 0 && (oppPet.HP < int(oppPet.Info.MaxHp)/2) {
CritRate = 16
}
//todo 暴击伤害
if t := grand.Meet(CritRate, 16); t {
skill.Crit = 1
}
}
2025-09-29 02:40:35 +08:00
// 恢复血量
func (our *Input) Heal(in *Input, ac action.BattleActionI, value alpacadecimal.Decimal) {
currentPet := our.CurrentPet()
if currentPet == nil {
return
}
healValue := int(value.IntPart())
if ac != nil {
if _, ok := ac.(*action.UseItemAction); !ok {
2026-04-04 06:27:15 +08:00
our.ExecWithOpponent(in, func(t Effect) bool {
t.Heal_Pre(ac, &healValue)
return true
})
}
}
2025-09-29 02:40:35 +08:00
//使用道具回血
if _, ok := ac.(*action.UseItemAction); !ok &&
ac != nil &&
in == our &&
healValue > 0 {
our.AttackValue.GainHp += int32(healValue) //道具有专门的回血包
}
if healValue >= 0 {
currentPet.Info.ModelHP(int64(healValue))
if our.AttackValue != nil {
our.AttackValue.RemainHp = int32(currentPet.Info.Hp)
our.AttackValue.MaxHp = currentPet.Info.MaxHp
}
return
}
damage := uint32(-healValue)
if damage >= currentPet.Info.Hp {
currentPet.Info.Hp = 0
if our.AttackValue != nil {
our.AttackValue.RemainHp = 0
our.AttackValue.MaxHp = currentPet.Info.MaxHp
}
return
}
currentPet.Info.Hp -= damage
if our.AttackValue != nil {
our.AttackValue.RemainHp = int32(currentPet.Info.Hp)
our.AttackValue.MaxHp = currentPet.Info.MaxHp
}
}
2025-11-11 05:54:24 +00:00
func (our *Input) HealPP(value int) {
currentPet := our.CurrentPet()
if currentPet == nil {
return
}
currentPet.Info.HealPP(value)
}
2025-11-11 05:54:24 +00:00
func (our *Input) DelPP(value int) {
currentPet := our.CurrentPet()
if currentPet == nil {
return
}
for i := 0; i < len(currentPet.Info.SkillList); i++ {
if uint32(value) > currentPet.Info.SkillList[i].PP {
currentPet.Info.SkillList[i].PP = 0
} else {
currentPet.Info.SkillList[i].PP -= uint32(value)
}
}
2025-09-29 02:40:35 +08:00
}
// Damage 对方对我方造成伤害,处理伤害计算和扣减血量逻辑,包括各种增伤、减伤效果
2025-11-11 05:54:24 +00:00
// /红伤只允许调用一次来保持锁伤
// 这个方法是对对方造成伤害
// 伤害落实 // 血量扣减节点比如触发回神,反弹也在这里实现
func (our *Input) Damage(in *Input, sub *info.DamageZone) {
currentPet := our.CurrentPet()
if currentPet == nil {
return
}
2026-04-04 06:27:15 +08:00
attacker := in
if attacker == nil {
attacker = our
}
// if sub.Type == info.DamageType.Red { //每回合计算伤害的时候重置伤害
// our.Opp.SumDamage = sub.Damage
// }
// 对方对我方造成,需要吃到对方的加成
var ok bool
2026-04-04 06:27:15 +08:00
if our != attacker {
ok = attacker.ExecWithOpponent(our, func(t Effect) bool {
t.DamageAdd(sub) //红伤落实前,我方增伤
2025-09-28 08:13:42 +00:00
return true
})
//sub.BeforeMul = sub.Damage
if ok {
2026-04-04 06:27:15 +08:00
ok = attacker.ExecWithOpponent(our, func(t Effect) bool {
t.Damage_Mul(sub) //红伤落实前,我方增伤
return true
})
}
//sub.BeforeFloor = sub.Damage
if ok {
2026-04-04 06:27:15 +08:00
ok = attacker.ExecWithOpponent(our, func(t Effect) bool {
t.DamageFloor(sub) //红伤落实,内部有befer
return true
})
}
2025-09-28 08:13:42 +00:00
}
// sub.BeforeMul = sub.Damage
2025-09-28 08:13:42 +00:00
if ok {
2026-04-04 06:27:15 +08:00
ok = our.ExecWithOpponent(attacker, func(t Effect) bool {
t.DamageDivEx(sub) //红伤落实,内部有befer
2025-09-28 08:13:42 +00:00
return true
})
}
//sub.BeforeSUB = sub.Damage
2025-09-28 08:13:42 +00:00
if ok {
2026-04-04 06:27:15 +08:00
ok = our.ExecWithOpponent(attacker, func(t Effect) bool {
2025-09-28 08:13:42 +00:00
t.DamageSubEx(sub)
2025-09-28 08:13:42 +00:00
return true
})
}
//sub.BeforeLocked = sub.Damage
if ok {
2026-04-04 06:27:15 +08:00
our.ExecWithOpponent(attacker, func(t Effect) bool {
t.DamageLockEx(sub)
2025-09-28 08:13:42 +00:00
return true
})
}
// sub.BeforeLock = sub.Damage
2026-04-04 06:27:15 +08:00
if ok && attacker != our {
ok = attacker.ExecWithOpponent(our, func(t Effect) bool {
t.DamageLock(sub)
2025-09-29 02:40:35 +08:00
return true
})
}
2025-09-28 08:13:42 +00:00
if ok {
2026-04-04 06:27:15 +08:00
our.ExecWithOpponent(attacker, func(t Effect) bool {
2025-09-28 08:13:42 +00:00
t.Damage_Shield(sub)
2025-09-28 08:13:42 +00:00
return true
})
}
if sub.Damage.Cmp(alpacadecimal.Zero) < 0 {
sub.Damage = alpacadecimal.Zero
}
if shieldAbsorb := our.AbsorbShieldDamage(sub.Damage); shieldAbsorb.Cmp(alpacadecimal.Zero) > 0 {
sub.Damage = sub.Damage.Sub(shieldAbsorb)
}
if sub.Type == info.DamageType.Red { //红才会产生造成伤害
2026-04-04 06:27:15 +08:00
attacker.SumDamage = sub.Damage.Add(attacker.SumDamage) // 叠加总伤害 这里相当于记录红伤
}
2025-11-11 05:54:24 +00:00
if sub.Type == info.DamageType.Red { //红才会产生造成伤害
2026-04-04 06:27:15 +08:00
attacker.AttackValue.LostHp += uint32(sub.Damage.IntPart()) //红伤落实
feat(player): 新增玩家累计经验查询接口 新增 PlayerExp 控制器方法,用于返回玩家的累计经验值。同时调整了经验池字段类型为 uint32 并修复相关使用逻辑。 feat(pet): 实现宠物经验增加与升级逻辑 在 Player 结构体中新增 AddPetExp 方法,支持宠物经验增长、自动升级及进化判断。升级后会重新计算面板属性并推送更新包。 feat(fight): 重构战斗伤害计算与效果系统 引入 DamageZone 和 EnumDamageType 类型,统一红伤处理流程;移除旧有的 Pet/Skill/Prop 属性获取临时修改机制,改为直接访问真实属性。更新多个技能效果实现以适配新结构。 refactor(effect): 优化技能效果初始化和生命周期方法 统一技能效果初始化方式,明确各阶段回调函数职责,如 PreActionStart、PreAttacked 等。删除已废弃的属性修改钩子函数,并更新状态类效果实现。 refactor(input): 移除 deepcopy 依赖并替换为 go-deepcopy 将原先使用的 mohae/deepcopy 替换为 barkimedes/go-deepcopy,用于战斗节点中的 effect 拷贝逻辑,提升性能和安全性。 refactor(model): 调整玩家信息字段类型 将 PlayerInfo 中的 GoldBean 字段由 int32 改为 uint32,ExpPool 字段由 int64 改为 uint32,确保数据类型一致性与合理性。 feat(nono): 增加 Nono 跟随/收回协议结构定义 新增 NonoFollowOrHomeInInfo 和 NonoFollowOutInfo 结构体,用于处理 Nono 宠物的跟随与收回操作指令。 chore(deps): 添加 go-deepcopy 依赖 在 go.mod 中引入 github.com/barkimedes/go-deepcopy 依赖库,用于替代原有的 deepcopy 工具。
2025-09-26 13:33:55 +08:00
}
if uint32(sub.Damage.IntPart()) > currentPet.Info.Hp {
currentPet.Info.Hp = 0
} else {
currentPet.Info.Hp = currentPet.Info.Hp - uint32(sub.Damage.IntPart())
}
if our.AttackValue != nil {
our.AttackValue.RemainHp = int32(currentPet.Info.Hp)
our.AttackValue.MaxHp = currentPet.Info.MaxHp
}
//todo 待实现死亡effet
}
// 计算技能威力
func (our *Input) CalculatePower(deftype *Input, skill *info.SkillEntity) alpacadecimal.Decimal {
if deftype == nil {
return alpacadecimal.Zero
}
ourPet := our.CurrentPet()
defPet := deftype.CurrentPet()
if ourPet == nil || defPet == nil {
return alpacadecimal.Zero
}
// 1. 计算等级因子 (level * 0.4 + 2)
levelFactor := alpacadecimal.NewFromInt(int64(ourPet.Info.Level)).
Mul(alpacadecimal.NewFromFloat(0.4)).Add(alpacadecimal.NewFromInt(2))
var (
attackDec alpacadecimal.Decimal //攻击值
defenseDec alpacadecimal.Decimal //防御值
)
``` refactor(effectarg): 移动EffectArgs初始化逻辑 将EffectArgs的初始化从effectarg.go中的init函数移动到file.go的initfile函数中, 确保在使用前正确加载配置并初始化映射。 refactor(login): 更新Login方法中的Person调用 修改Login方法中对Person函数的调用,传递UserID参数以获取正确的用户信息。 refactor(user): 统一使用Person方法替代PersonOther 在UserSimInfo和UserMoreInfo方法中,将原先调用的PersonOther方法统一替换为 Person方法,保持代码一致性。 refactor(effect_damage): 简化属性获取和伤害计算逻辑 移除deepcopy相关逻辑,简化Effect0的OnSkill方法中的属性获取和伤害计算流程, 直接通过输入参数进行计算。 refactor(fightc): 优化玩家输入处理和战斗逻辑 更新GetInputByPlayer方法中的玩家判断逻辑,使用UserID比较代替对象比较; 在initplayer中添加InitAttackValue调用; 修复battleLoop中打印语句的格式问题; 调整技能攻击处理流程,增加SkillUseEnd回调调用。 refactor(attr): 改进属性获取方法和伤害计算逻辑 将GetProp方法重命名为Prop,并支持传入对方输入参数; 更新CalculatePower方法签名,使用Input类型代替BattlePetEntity; 在属性获取和伤害计算中正确处理双方属性影响。 refactor(playeraction): 简化技能使用逻辑 简化UseSkill方法中获取当前宠物信息的逻辑,去除冗余的条件判断; 在找到对应技能后添加break语句,提高执行效率。 refactor(reg): 更新Person方法实现 合并Person和PersonOther方法为统一的Person方法; 在数据库查询失败时添加错误处理,避免潜在的空指针异常。 ```
2025-09-24 12:40:13 +08:00
switch skill.Category() { //判断技能类型
case info.Category.PHYSICAL:
attackDec = our.GetProp(0)
defenseDec = deftype.GetProp(1)
case info.Category.SPECIAL:
attackDec = our.GetProp(2)
defenseDec = deftype.GetProp(3)
default:
2026-01-21 20:46:05 +00:00
return alpacadecimal.Zero
}
// 8. DmgBindLv: 使对方受到的伤害值等于等级; 默认: 0
// 9. PwrBindDv: 1,威力power取决于潜力个体值*5; 2,威力power取决于最大体力*1/3+潜力(个体值); 默认: 0
// 10. PwrDouble: 攻击时,若对方处于异常状态, 则威力翻倍;
// 11. DmgBindHpDv: 造成的伤害等于自身剩余体力*1/2+潜力(个体值); 默认: 0
if skill.XML.DmgBindLv != 0 {
2026-03-23 22:00:05 +08:00
skill.XML.Power = int(defPet.Info.Level)
}
if skill.XML.PwrBindDv != 0 {
if skill.XML.PwrBindDv == 1 {
skill.XML.Power = int(ourPet.Info.Dv * 5)
2026-02-11 11:06:28 +08:00
}
if skill.XML.PwrBindDv == 2 {
skill.XML.Power = int(ourPet.Info.Hp/3 + ourPet.Info.Dv)
}
}
if skill.XML.PwrDouble != 0 {
if deftype.StatEffect_Exist_all() {
skill.XML.Power = skill.XML.Power * 2
}
}
if skill.XML.DmgBindHpDv != 0 {
skill.XML.Power = int(ourPet.Info.Hp/2 + ourPet.Info.Dv)
}
2026-02-11 11:06:28 +08:00
// 5. 基础伤害公式:等级因子 * 威力因子 * 攻击 / 防御 / 50 + 2
baseDamage := levelFactor.
Div(alpacadecimal.NewFromInt(50)).
Mul(alpacadecimal.NewFromInt(int64(skill.XML.Power))).
Mul(attackDec.Div(defenseDec)).
2026-02-11 11:06:28 +08:00
Add(alpacadecimal.NewFromInt(2))
var typeRate alpacadecimal.Decimal
//fmt.Println(skill.Type().ID, defPet.Type().ID)
t, _ := element.Calculator.GetOffensiveMultiplier(skill.GetType().ID, defPet.GetType().ID)
2026-02-11 11:06:28 +08:00
our.AttackValue.Offensive = gconv.Float32(t)
typeRate = alpacadecimal.NewFromFloat(t)
damage := baseDamage.
Mul(skill.CriticalsameTypeBonus()). // 同属性加成
Mul(typeRate). // 克制系数
Mul(skill.Criticalrandom()) //随机波动
//println(baseDamage.IntPart(), damage.IntPart(), attackDec.IntPart(), defenseDec.IntPart(), "技能伤害")
return damage
}