Files
pixelheros/assets/script/game/hero/MonMove.ts
walkpan 9d6075be6e refactor(hero): 重构英雄属性与状态管理
- 将增益效果属性组移到武器进化属性后以优化结构
- 新增 in_stun 和 in_frost 状态标志替代 isStun/isFrost 方法
- 更新状态检查逻辑以使用新的状态标志
- 移除 HeroSkillsComp 依赖以简化移动系统
- 修改伤害计算直接使用 HeroAttrsComp 属性而非 Attrs 映射
- 简化暴击、击退等判定逻辑,移除闪避和抗性计算
- 优化 reset 方法,设置合理的默认值并重置新增状态标志
- 添加状态变化时的调试日志输出
2026-03-11 22:51:48 +08:00

339 lines
12 KiB
TypeScript
Raw Blame History

This file contains ambiguous Unicode characters
This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.
import { ecs } from "../../../../extensions/oops-plugin-framework/assets/libs/ecs/ECS";
import { HeroViewComp } from "./HeroViewComp";
import { HeroAttrsComp } from "./HeroAttrsComp";
import { smc } from "../common/SingletonModuleComp";
import { FacSet, IndexSet } from "../common/config/GameSet";
import { Attrs } from "../common/config/HeroAttrs";
import { HType } from "../common/config/heroSet";
import { SkillRange } from "../common/config/SkillSet";
/** 怪物移动组件 */
@ecs.register('MonMove')
export class MonMoveComp extends ecs.Comp {
/** 移动方向1向右-1向左 */
direction: number = 1;
/** 目标x坐标 */
targetX: number = 0;
/** 是否处于移动状态 */
moving: boolean = true;
/** 线路标识0=一线(y=120)1=二线(y=80) */
lane: number = 0;
/** 生成顺序:用于同线路内的层级排序,数值越大越晚生成,层级越前 */
spawnOrder: number = 0;
/** 目标y坐标 */
targetY: number = 0;
reset() {
this.direction = 1;
this.targetX = 0;
this.moving = true;
this.lane = 0;
this.spawnOrder = 0;
this.targetY = 0;
}
}
/** 怪物移动系统 - 专门处理怪物的移动逻辑 */
@ecs.register('MonMoveSystem')
export class MonMoveSystem extends ecs.ComblockSystem implements ecs.ISystemUpdate {
filter(): ecs.IMatcher {
return ecs.allOf(MonMoveComp, HeroViewComp, HeroAttrsComp);
}
update(e: ecs.Entity) {
// 1. 全局状态检查
if (!smc.mission.play || smc.mission.pause) return;
const view = e.get(HeroViewComp);
// 如果英雄死亡停止怪物行动标志为true则停止怪物移动
if (smc.mission.stop_mon_action) {
view.status_change("idle");
return;
}
const move = e.get(MonMoveComp);
const model = e.get(HeroAttrsComp);
// 只处理怪物
if (model.fac !== FacSet.MON) return;
if (!move.moving) return;
// 2. 异常状态检查 (死亡/复活/眩晕/冰冻)
if (model.is_stop || model.is_dead || model.in_stun|| model.in_frost) {
view.status_change("idle");
return;
}
this.updateRenderOrder(e);
// 检查是否需要y轴靠近
this.checkAndSetTargetY(e);
// 3. 核心移动逻辑分发
const nearestEnemy = this.findNearestEnemy(e);
if (nearestEnemy) {
// 战斗状态根据职业类型和rangeType执行智能战术
this.processCombatLogic(e, move, view, model, nearestEnemy);
} else {
// 非战斗状态:向目标移动
this.processMarchLogic(e, move, view, model);
model.is_atking = false;
}
}
/**
* 战斗移动逻辑分发
* 根据 rangeType 决定走位策略
*/
private processCombatLogic(e: ecs.Entity, move: MonMoveComp, view: HeroViewComp, model: HeroAttrsComp, enemy: HeroViewComp) {
// 优先使用 rangeType 判断,如果没有则回退到 type 判断
let rangeType = model.rangeType;
// 兼容性处理:如果数据未配置 rangeType根据旧的职业类型推断
if (rangeType === undefined) {
if (model.type === HType.warrior || model.type === HType.assassin) {
rangeType = SkillRange.Melee;
} else if (model.type === HType.remote) {
rangeType = SkillRange.Long;
} else {
rangeType = SkillRange.Mid;
}
}
switch (rangeType) {
case SkillRange.Melee:
this.processMeleeLogic(e, move, view, model, enemy);
break;
case SkillRange.Mid:
this.processMidLogic(e, move, view, model, enemy);
break;
case SkillRange.Long:
this.processLongLogic(e, move, view, model, enemy);
break;
default:
this.processMidLogic(e, move, view, model, enemy); // 默认中程
break;
}
}
/**
* 近战逻辑 (Melee)
* 策略:无脑突进,贴脸输出
* 范围:< 75 (攻击距离)
*/
private processMeleeLogic(e: ecs.Entity, move: MonMoveComp, view: HeroViewComp, model: HeroAttrsComp, enemy: HeroViewComp) {
const currentX = view.node.position.x;
const enemyX = enemy.node.position.x;
const dist = Math.abs(currentX - enemyX);
const attackRange = 75; // 保持原有的近战判定
move.direction = enemyX > currentX ? 1 : -1;
if (dist <= attackRange) {
view.status_change("idle");
model.is_atking = true;
} else {
const speed = model.speed / 3;
this.moveEntity(view, move.direction, speed);
model.is_atking = false;
}
}
/**
* 中程逻辑 (Mid)
* 策略:保持在中距离,灵活输出
* 范围120 - 360
*/
private processMidLogic(e: ecs.Entity, move: MonMoveComp, view: HeroViewComp, model: HeroAttrsComp, enemy: HeroViewComp) {
const currentX = view.node.position.x;
const enemyX = enemy.node.position.x;
const dist = Math.abs(currentX - enemyX);
const minRange = 120;
const maxRange = 360;
move.direction = enemyX > currentX ? 1 : -1;
if (dist < minRange) {
// 太近了,后撤
this.performRetreat(view, move, model, currentX);
} else if (dist > maxRange) {
// 太远了,追击
const speed = model.speed / 3;
this.moveEntity(view, move.direction, speed);
model.is_atking = false;
} else {
// 距离合适,站桩输出
view.status_change("idle");
model.is_atking = true;
}
}
/**
* 远程逻辑 (Long)
* 策略:保持在远距离,最大化生存
* 范围360 - 720
*/
private processLongLogic(e: ecs.Entity, move: MonMoveComp, view: HeroViewComp, model: HeroAttrsComp, enemy: HeroViewComp) {
const currentX = view.node.position.x;
const enemyX = enemy.node.position.x;
const dist = Math.abs(currentX - enemyX);
const minRange = 360;
const maxRange = 720;
move.direction = enemyX > currentX ? 1 : -1;
if (dist < minRange) {
// 太近了,后撤 (远程单位对距离更敏感)
this.performRetreat(view, move, model, currentX);
} else if (dist > maxRange) {
// 太远了,追击
const speed = model.speed / 3;
this.moveEntity(view, move.direction, speed);
model.is_atking = false;
} else {
// 距离合适,站桩输出
view.status_change("idle");
model.is_atking = true;
}
}
/** 执行后撤逻辑 */
private performRetreat(view: HeroViewComp, move: MonMoveComp, model: HeroAttrsComp, currentX: number) {
const safeRetreatX = currentX - move.direction * 50;
// 怪物活动范围通常宽一些
if (safeRetreatX >= -450 && safeRetreatX <= 450) {
const retreatSpeed = (model.speed / 3) * 0.8;
this.moveEntity(view, -move.direction, retreatSpeed);
model.is_atking = false;
} else {
// 退无可退,被迫反击
view.status_change("idle");
model.is_atking = true;
}
}
/**
* 进军逻辑 (无敌人时)
* 策略向目标X移动 (通常是屏幕左侧)
*/
private processMarchLogic(e: ecs.Entity, move: MonMoveComp, view: HeroViewComp, model: HeroAttrsComp) {
const currentX = view.node.position.x;
const targetX = move.targetX;
if (Math.abs(currentX - targetX) > 5) {
const dir = targetX > currentX ? 1 : -1;
const speed = model.speed / 3;
// 修正朝向
move.direction = dir;
this.moveEntity(view, dir, speed);
} else {
view.status_change("idle");
}
}
/** 通用移动执行 */
private moveEntity(view: HeroViewComp, direction: number, speed: number) {
const delta = speed * this.dt * direction;
const newX = view.node.position.x + delta;
// 处理Y轴移动
const move = view.ent.get(MonMoveComp);
let newY = view.node.position.y;
if (move && move.targetY !== 0) {
const deltaY = speed * this.dt * Math.sign(move.targetY - newY);
newY = newY + deltaY;
if (Math.abs(newY - move.targetY) < Math.abs(deltaY)) {
newY = move.targetY;
move.targetY = 0;
}
}
// 地图边界限制
if (newX >= -450 && newX <= 450) {
view.node.setPosition(newX, newY, 0);
view.status_change("move");
} else {
view.status_change("idle");
}
}
/** 检查并设置y轴目标位置 */
private checkAndSetTargetY(entity: ecs.Entity): void {
const move = entity.get(MonMoveComp);
const currentView = entity.get(HeroViewComp);
const heroAttrs = entity.get(HeroAttrsComp);
if (!currentView || !currentView.node || !heroAttrs) return;
if (move.targetY !== 0) return;
const currentPos = currentView.node.position;
const team = heroAttrs.fac;
let nearestHero: ecs.Entity | null = null;
let nearestDist = Infinity;
ecs.query(ecs.allOf(HeroAttrsComp, HeroViewComp)).forEach(e => {
const model = e.get(HeroAttrsComp);
const view = e.get(HeroViewComp);
if (!view || !view.node) return;
if (model.fac === team || model.is_dead) return;
const dist = Math.abs(currentPos.x - view.node.position.x);
if (dist < nearestDist) {
nearestDist = dist;
nearestHero = e;
}
});
if (!nearestHero || nearestDist > 100) return;
const heroView = nearestHero.get(HeroViewComp);
if (!heroView || !heroView.node) return;
const heroY = heroView.node.position.y;
const yDist = Math.abs(currentPos.y - heroY);
if (yDist > 50) {
const direction = heroY > currentPos.y ? 1 : -1;
move.targetY = heroY + direction * 50;
}
}
private findNearestEnemy(entity: ecs.Entity): HeroViewComp | null {
const currentView = entity.get(HeroViewComp);
if (!currentView?.node) return null;
const currentPos = currentView.node.position;
const myFac = entity.get(HeroAttrsComp).fac;
let nearest: HeroViewComp | null = null;
let minDis = Infinity;
ecs.query(ecs.allOf(HeroAttrsComp, HeroViewComp)).forEach(e => {
const m = e.get(HeroAttrsComp);
// 找对立阵营且存活的
if (m.fac !== myFac && !m.is_dead) {
const v = e.get(HeroViewComp);
if (v?.node) {
const d = Math.abs(currentPos.x - v.node.position.x);
if (d < minDis) {
minDis = d;
nearest = v;
}
}
}
});
return nearest;
}
private updateRenderOrder(entity: ecs.Entity) {
// 🔥 移除渲染层级更新:各线路固定,后召唤的天然层级更高,无需动态调整
}
}