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

361 lines
9.6 KiB
Go
Raw Normal View History

package input
import (
element "blazing/common/data/Element"
2025-11-08 18:37:11 +08:00
"blazing/common/data/xmlres"
"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"
2025-11-23 23:38:03 +00:00
"github.com/gogf/gf/v2/util/grand"
)
// 计算暴击
2025-11-11 05:54:24 +00:00
func (our *Input) CalculateCrit(opp *Input, skill *info.SkillEntity) {
skill.Crit = 0
if skill.Category() == info.Category.STATUS { //属性技能不用算暴击
return
}
CritRate := utils.Max(skill.CritRate, 1)
//CritAtkFirst: 先出手时必定致命一击; 默认: 0
2025-11-11 05:54:24 +00:00
if skill.CritAtkFirst != 0 && our.FightC.IsFirst(our.Player) {
CritRate = 16
}
//CritAtkSecond: 后出手时必定致命一击; 默认: 0
2025-11-11 05:54:24 +00:00
if skill.CritAtkSecond != 0 && !our.FightC.IsFirst(our.Player) {
CritRate = 16
}
// CritSelfHalfHp: 自身体力低于一半时必定致命一击; 默认: 0
2025-11-11 05:54:24 +00:00
if skill.CritSelfHalfHp != 0 && (our.CurrentPet.HP < int(our.CurrentPet.Info.MaxHp)/2) {
CritRate = 16
}
// CritFoeHalfHp: 对方体力低于一半时必定致命一击; 默认: 0
if skill.CritSelfHalfHp != 0 && (opp.CurrentPet.HP < int(opp.CurrentPet.Info.MaxHp)/2) {
CritRate = 16
}
//todo 暴击伤害
2025-11-11 05:54:24 +00:00
if t, _, _ := our.Player.Roll(625*CritRate, 10000); t {
skill.Crit = 1
}
}
2025-09-29 02:40:35 +08:00
// 恢复血量
func (our *Input) Heal(in *Input, ac action.BattleActionI, value alpacadecimal.Decimal) {
2025-09-29 02:40:35 +08:00
//使用道具回血
if _, ok := ac.(*action.UseItemAction); !ok &&
ac != nil &&
in == our {
2025-11-11 05:54:24 +00:00
our.AttackValue.GainHp = int32(value.IntPart()) //道具有专门的回血包
}
if value.IsPositive() {
2025-11-11 05:54:24 +00:00
our.CurrentPet.Info.Hp += uint32(value.IntPart())
2025-11-11 05:54:24 +00:00
our.CurrentPet.Info.Hp = utils.Min(our.CurrentPet.Info.Hp, our.CurrentPet.Info.MaxHp)
} else {
2025-11-11 05:54:24 +00:00
if uint32(value.Abs().IntPart()) > our.CurrentPet.Info.Hp {
our.CurrentPet.Info.Hp = 0
} else {
2025-11-11 05:54:24 +00:00
our.CurrentPet.Info.Hp += uint32(value.IntPart())
}
}
}
2025-11-11 05:54:24 +00:00
func (our *Input) HealPP(value int) {
2025-11-11 05:54:24 +00:00
for i := 0; i < len(our.CurrentPet.Info.SkillList); i++ {
if value == -1 {
our.CurrentPet.Info.SkillList[i].PP = uint32(xmlres.SkillMap[int(our.CurrentPet.Info.SkillList[i].ID)].MaxPP)
} else {
our.CurrentPet.Info.SkillList[i].PP += uint32(value)
our.CurrentPet.Info.SkillList[i].PP = utils.Min(our.CurrentPet.Info.SkillList[i].PP, uint32(xmlres.SkillMap[int(our.CurrentPet.Info.SkillList[i].ID)].MaxPP))
}
}
}
2025-11-11 05:54:24 +00:00
func (our *Input) DelPP(value int) {
2025-11-11 05:54:24 +00:00
for i := 0; i < len(our.CurrentPet.Info.SkillList); i++ {
if uint32(value) > our.CurrentPet.Info.SkillList[i].PP {
our.CurrentPet.Info.SkillList[i].PP = 0
} else {
2025-11-11 05:54:24 +00:00
our.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) {
if sub.Type == info.DamageType.Red { //每回合计算伤害的时候重置伤害
2025-11-21 05:47:51 +00:00
our.Opp.SumDamage = sub.Damage
}
// 对方对我方造成,需要吃到对方的加成
var ok bool
if our != in {
ok = our.Opp.Exec(func(t Effect) bool {
t.Damage_ADD(sub) //红伤落实前,我方增伤
2025-09-28 08:13:42 +00:00
return true
})
//sub.BeforeMul = sub.Damage
if ok {
ok = our.Opp.Exec(func(t Effect) bool {
t.Damage_Mul(sub) //红伤落实前,我方增伤
return true
})
}
//sub.BeforeFloor = sub.Damage
if ok {
ok = our.Opp.Exec(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 {
ok = our.Exec(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 {
ok = our.Exec(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.BeforeLock = sub.Damage
if ok && in != our {
2025-11-11 05:54:24 +00:00
ok = our.Opp.Exec(func(t Effect) bool {
t.DamageLock(sub)
2025-09-29 02:40:35 +08:00
return true
})
}
//sub.BeforeLocked = sub.Damage
2025-09-28 08:13:42 +00:00
if ok {
our.Exec(func(t Effect) bool {
2025-09-28 08:13:42 +00:00
t.DamageLockEx(sub)
2025-09-28 08:13:42 +00:00
return true
})
}
2025-11-11 05:54:24 +00:00
if sub.Type == info.DamageType.Red { //红才会产生造成伤害
2025-11-21 05:47:51 +00:00
our.Opp.SumDamage = sub.Damage // 叠加总伤害 这里相当于记录红伤
our.Opp.AttackValue.LostHp = uint32(our.Opp.SumDamage.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()) > our.CurrentPet.Info.Hp {
our.CurrentPet.Info.Hp = 0
} else {
our.CurrentPet.Info.Hp = our.CurrentPet.Info.Hp - uint32(sub.Damage.IntPart())
}
//todo 待实现死亡effet
}
func (our *Input) GetAction() {
println("开始执行NPC动作")
next := our.Exec(func(t Effect) bool {
return t.HookAction()
})
if !next {
return
}
// 获取己方当前宠物和对方当前宠物
2025-11-11 05:54:24 +00:00
selfPet := our.FightC.GetCurrPET(our.Player)
2025-11-10 08:25:40 +00:00
//没血就切换精灵
if selfPet.Info.Hp <= 0 {
2025-11-11 05:54:24 +00:00
for _, v := range our.AllPet {
if v.Info.Hp > 0 {
our.FightC.ChangePet(our.Player, v.Info.CatchTime)
2025-11-10 08:25:40 +00:00
return
}
}
// 如果没有可用宠物,则直接返回,不执行任何操作
return
2025-11-10 08:25:40 +00:00
}
//oppPet := opp.FightC.GetCurrPET(opp.Player)
skills := selfPet.Skills
// 空技能列表直接返回,避免错误
if len(skills) == 0 {
return
}
// 步骤1计算所有技能的伤害并筛选出能秒杀对方的技能
var killableSkills []struct {
*info.SkillEntity // 技能对象假设原技能类型为skill
damage alpacadecimal.Decimal // 技能实际伤害
}
// 存储所有技能及伤害(用于后续筛选)
type skillWithDamage struct {
*info.SkillEntity
damage alpacadecimal.Decimal
}
allSkills := make([]skillWithDamage, 0, len(skills))
for _, s := range skills {
if s == nil {
continue
}
if !s.CanUse() {
continue
}
// 计算技能对对方的伤害假设CalculatePower返回伤害值或需从技能中获取
damage := our.CalculatePower(our.Opp, s)
allSkills = append(allSkills, skillWithDamage{SkillEntity: s, damage: damage})
// 判断是否能秒杀(伤害 >= 对方当前生命值)
if uint32(damage.IntPart()) >= our.Opp.CurrentPet.Info.Hp { // 假设oppPet.HP为对方当前剩余生命值
killableSkills = append(killableSkills, struct {
*info.SkillEntity
damage alpacadecimal.Decimal
}{s, damage})
}
}
// 若存在能秒杀的技能,优先使用(选伤害最高的,避免浪费高伤害技能)
if len(killableSkills) > 0 {
bestKillSkill := killableSkills[0].SkillEntity
maxDamage := killableSkills[0].damage
for _, ks := range killableSkills[1:] {
if ks.damage.Cmp(maxDamage) > 0 { // 使用decimal的比较方法比较伤害值
if ks.CanUse() {
maxDamage = ks.damage
bestKillSkill = ks.SkillEntity
}
}
}
2025-11-23 23:42:16 +00:00
our.FightC.UseSkill(our.Player, uint32(bestKillSkill.ID))
return
}
if len(allSkills) <= 0 {
our.FightC.UseSkill(our.Player, 0)
return
}
// 优化随机选择技能的逻辑,直接使用随机索引
2025-11-23 23:38:03 +00:00
randomIdx := grand.Intn(len(allSkills))
our.FightC.UseSkill(our.Player, uint32(allSkills[randomIdx].ID))
return
}
// 计算技能威力
func (our *Input) CalculatePower(deftype *Input, skill *info.SkillEntity) alpacadecimal.Decimal {
// 1. 计算等级因子 (level * 0.4 + 2)
levelFactor := alpacadecimal.NewFromInt(int64(our.CurrentPet.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 = alpacadecimal.NewFromInt(int64(our.GetProp(0, false)))
defenseDec = alpacadecimal.NewFromInt(int64(deftype.GetProp(1, false)))
case info.Category.SPECIAL:
attackDec = alpacadecimal.NewFromInt(int64(our.GetProp(2, false)))
defenseDec = alpacadecimal.NewFromInt(int64(deftype.GetProp(3, false)))
default:
return alpacadecimal.NewFromInt(0)
}
// 5. 基础伤害公式:等级因子 * 威力因子 * 攻击 / 防御 / 50 + 2
baseDamage := levelFactor.
Mul(alpacadecimal.NewFromInt(int64(skill.Power))).
Mul(attackDec).
Div(defenseDec).
Div(alpacadecimal.NewFromInt(50)).
Add(alpacadecimal.NewFromInt(2))
var typeRate alpacadecimal.Decimal
//fmt.Println(skill.Type().ID, deftype.CurrentPet.Type().ID)
t, _ := element.Calculator.GetOffensiveMultiplier(skill.GetType().ID, deftype.CurrentPet.GetType().ID)
``` 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
typeRate = alpacadecimal.NewFromFloat(t)
// 8. DmgBindLv: 使对方受到的伤害值等于等级; 默认: 0
// 9. PwrBindDv: 1,威力power取决于潜力个体值*5; 2,威力power取决于最大体力*1/3+潜力(个体值); 默认: 0
// 10. PwrDouble: 攻击时,若对方处于异常状态, 则威力翻倍;
// 11. DmgBindHpDv: 造成的伤害等于自身剩余体力*1/2+潜力(个体值); 默认: 0
if skill.DmgBindLv != 0 {
baseDamage = alpacadecimal.NewFromInt(int64(deftype.CurrentPet.Info.Level))
}
if skill.PwrBindDv != 0 {
if skill.PwrBindDv == 1 {
baseDamage = alpacadecimal.NewFromInt(int64(our.CurrentPet.Info.Dv * 5))
}
if skill.PwrBindDv == 2 {
baseDamage = alpacadecimal.NewFromInt(int64(our.CurrentPet.Info.Hp/3 + our.CurrentPet.Info.Dv))
}
}
if skill.PwrDouble != 0 {
if deftype.StatEffect_Exist_all() {
baseDamage = baseDamage.Mul(alpacadecimal.NewFromInt(2))
}
}
if skill.DmgBindHpDv != 0 {
baseDamage = alpacadecimal.NewFromInt(int64(our.CurrentPet.Info.Hp/2 + our.CurrentPet.Info.Dv))
}
damage := baseDamage.
Mul(skill.CriticalsameTypeBonus()). // 同属性加成
Mul(typeRate). // 克制系数
Mul(skill.Criticalrandom()) //随机波动
return damage
}