GroundBurnArea.ts 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433
  1. import { _decorator, Component, Node, Vec3, find, BoxCollider2D, RigidBody2D, Collider2D, Contact2DType, IPhysics2DContact, ERigidBody2DType } from 'cc';
  2. import { sp } from 'cc';
  3. import EventBus, { GameEvents } from '../../Core/EventBus';
  4. import { EnemyAttackStateManager } from '../EnemyAttackStateManager';
  5. import { GroundBurnAreaManager } from './GroundBurnAreaManager';
  6. import { PhysicsManager } from '../../Core/PhysicsManager';
  7. const { ccclass, property } = _decorator;
  8. /**
  9. * 地面燃烧区域组件
  10. * 用于在地面创建固定范围的燃烧区域,对进入区域的敌人造成持续伤害
  11. */
  12. @ccclass('GroundBurnArea')
  13. export class GroundBurnArea extends Component {
  14. private _duration: number = 0;
  15. private _damage: number = 0;
  16. private _tickInterval: number = 0;
  17. private _isActive: boolean = false;
  18. private _burnEffectNode: Node | null = null; // 燃烧特效节点
  19. private _enemiesInArea: Set<Node> = new Set(); // 区域内的敌人
  20. private _damageTimer: number = 0; // 伤害计时器
  21. /**
  22. * 组件加载时的初始化
  23. */
  24. protected onLoad(): void {
  25. // 立即禁用RigidBody2D组件,防止在物理世界未准备好时激活
  26. const rigidBody = this.node.getComponent(RigidBody2D);
  27. if (rigidBody) {
  28. rigidBody.enabled = false;
  29. }
  30. // 也禁用BoxCollider2D组件,确保完全避免物理系统访问
  31. const collider = this.node.getComponent(BoxCollider2D);
  32. if (collider) {
  33. collider.enabled = false;
  34. }
  35. }
  36. /**
  37. * 初始化地面燃烧区域
  38. * @param position 燃烧区域位置
  39. * @param duration 持续时间(秒)
  40. * @param damage 每次伤害值
  41. * @param tickInterval 伤害间隔(秒)
  42. */
  43. public initGroundBurnArea(position: Vec3, duration: number, damage: number, tickInterval: number): void {
  44. this._duration = duration;
  45. this._damage = damage;
  46. this._tickInterval = tickInterval;
  47. this._isActive = true;
  48. this._damageTimer = 0;
  49. // 设置节点位置
  50. this.node.setWorldPosition(position);
  51. // 获取预制体中的火焰特效节点
  52. this.setupFlameEffect();
  53. // 先禁用RigidBody2D组件,避免在物理世界未准备好时激活
  54. const rigidBody = this.node.getComponent(RigidBody2D);
  55. if (rigidBody) {
  56. rigidBody.enabled = false;
  57. }
  58. // 设置碰撞器
  59. this.setupCollider();
  60. console.log(`[GroundBurnArea] 初始化地面燃烧区域 - 位置: (${position.x.toFixed(1)}, ${position.y.toFixed(1)}), 持续时间: ${duration}秒, 伤害: ${damage}, 间隔: ${tickInterval}秒`);
  61. // 启动燃烧区域
  62. this.startBurnArea();
  63. }
  64. /**
  65. * 设置碰撞器
  66. */
  67. private setupCollider(): void {
  68. // 延迟设置碰撞器,确保物理系统已初始化
  69. this.scheduleOnce(() => {
  70. this.trySetupCollider();
  71. }, 0.2); // 增加延迟时间到0.2秒
  72. }
  73. private trySetupCollider(): void {
  74. // 检查物理系统是否已初始化
  75. const physicsManager = PhysicsManager.getInstance();
  76. if (!physicsManager) {
  77. console.warn('[GroundBurnArea] 物理系统未初始化,延迟重试');
  78. // 延迟重试
  79. this.scheduleOnce(() => {
  80. this.trySetupCollider();
  81. }, 0.1);
  82. return;
  83. }
  84. const physicsSystem = physicsManager.getSystem();
  85. if (!physicsSystem || !physicsSystem.enable) {
  86. console.warn('[GroundBurnArea] 物理系统未启用,延迟重试');
  87. // 延迟重试
  88. this.scheduleOnce(() => {
  89. this.trySetupCollider();
  90. }, 0.1);
  91. return;
  92. }
  93. // 获取碰撞器组件
  94. const collider = this.node.getComponent(BoxCollider2D);
  95. if (!collider) {
  96. console.error('[GroundBurnArea] 未找到 BoxCollider2D 组件');
  97. return;
  98. }
  99. // 重新启用碰撞器组件
  100. collider.enabled = true;
  101. // 获取刚体组件并确保启用碰撞监听
  102. const rigidBody = this.node.getComponent(RigidBody2D);
  103. if (rigidBody) {
  104. try {
  105. rigidBody.type = ERigidBody2DType.Static;
  106. rigidBody.enabledContactListener = true;
  107. // 重新启用RigidBody2D组件
  108. rigidBody.enabled = true;
  109. } catch (error) {
  110. console.error('[GroundBurnArea] 设置刚体组件时出错:', error);
  111. // 如果设置失败,再次延迟重试
  112. this.scheduleOnce(() => {
  113. this.trySetupCollider();
  114. }, 0.1);
  115. return;
  116. }
  117. }
  118. // 监听碰撞事件(移除之前的监听器避免重复)
  119. collider.off(Contact2DType.BEGIN_CONTACT, this.onBeginContact, this);
  120. collider.off(Contact2DType.END_CONTACT, this.onEndContact, this);
  121. collider.on(Contact2DType.BEGIN_CONTACT, this.onBeginContact, this);
  122. collider.on(Contact2DType.END_CONTACT, this.onEndContact, this);
  123. console.log(`[GroundBurnArea] 碰撞器设置完成,碰撞器启用: ${collider.enabled}, 刚体启用: ${rigidBody?.enabled}`);
  124. // 验证碰撞器设置
  125. if (!collider.enabled) {
  126. console.error('[GroundBurnArea] 碰撞器未启用!');
  127. }
  128. if (rigidBody && !rigidBody.enabled) {
  129. console.error('[GroundBurnArea] 刚体未启用!');
  130. }
  131. if (rigidBody && !rigidBody.enabledContactListener) {
  132. console.error('[GroundBurnArea] 刚体碰撞监听未启用!');
  133. }
  134. }
  135. /**
  136. * 敌人进入燃烧区域
  137. */
  138. private onBeginContact(selfCollider: Collider2D, otherCollider: Collider2D, contact: IPhysics2DContact | null): void {
  139. const enemyNode = otherCollider.node;
  140. console.log(`[GroundBurnArea] 碰撞检测 - 节点名称: ${enemyNode.name}, 父节点: ${enemyNode.parent?.name || 'null'}`);
  141. // 检查是否是敌人
  142. if (this.isEnemyNode(enemyNode)) {
  143. // 仅记录可攻击的敌人进入燃烧区域(隐身等状态不记录)
  144. const attackStateManager = EnemyAttackStateManager.getInstance();
  145. if (attackStateManager && !attackStateManager.isEnemyAttackable(enemyNode)) {
  146. console.log(`[GroundBurnArea] 敌人不可攻击(隐身等),不加入区域: ${enemyNode.name}`);
  147. return;
  148. }
  149. this._enemiesInArea.add(enemyNode);
  150. console.log(`[GroundBurnArea] 敌人进入燃烧区域: ${enemyNode.name}, 区域内敌人数量: ${this._enemiesInArea.size}`);
  151. } else {
  152. console.log(`[GroundBurnArea] 非敌人节点进入燃烧区域: ${enemyNode.name}`);
  153. }
  154. }
  155. /**
  156. * 敌人离开燃烧区域
  157. */
  158. private onEndContact(selfCollider: Collider2D, otherCollider: Collider2D, contact: IPhysics2DContact | null): void {
  159. const enemyNode = otherCollider.node;
  160. // 检查是否是敌人
  161. if (this.isEnemyNode(enemyNode)) {
  162. this._enemiesInArea.delete(enemyNode);
  163. console.log(`[GroundBurnArea] 敌人离开燃烧区域: ${enemyNode.name}, 区域内敌人数量: ${this._enemiesInArea.size}`);
  164. }
  165. }
  166. /**
  167. * 检查是否是敌人节点
  168. */
  169. private isEnemyNode(node: Node): boolean {
  170. if (!node || !node.name) {
  171. return false;
  172. }
  173. // 检查是否为EnemySprite子节点
  174. if (node.name === 'EnemySprite' && node.parent) {
  175. const hasEnemyComponent = node.parent.getComponent('EnemyInstance') !== null;
  176. console.log(`[GroundBurnArea] EnemySprite检测 - 节点: ${node.name}, 父节点: ${node.parent.name}, 有EnemyInstance组件: ${hasEnemyComponent}`);
  177. return hasEnemyComponent;
  178. }
  179. // 检查节点名称
  180. const nameCheck = (
  181. node.name.includes('Enemy') ||
  182. node.name.includes('enemy') ||
  183. node.name.includes('僵尸') ||
  184. node.name.includes('Zombie') ||
  185. node.name.includes('zombie')
  186. );
  187. // 检查父节点名称
  188. const parentCheck = (
  189. node.parent?.name === 'enemyContainer' ||
  190. node.parent?.name === 'EnemyContainer' ||
  191. node.parent?.name?.includes('enemy') ||
  192. node.parent?.name?.includes('Enemy')
  193. );
  194. // 检查是否有EnemyInstance组件
  195. const componentCheck = node.getComponent('EnemyInstance') !== null;
  196. const isEnemy = nameCheck || parentCheck || componentCheck;
  197. console.log(`[GroundBurnArea] 敌人检测 - 节点: ${node.name}, 父节点: ${node.parent?.name || 'null'}, 名称检查: ${nameCheck}, 父节点检查: ${parentCheck}, 组件检查: ${componentCheck}, 是敌人: ${isEnemy}`);
  198. return isEnemy;
  199. }
  200. /**
  201. * 启动燃烧区域
  202. */
  203. private startBurnArea(): void {
  204. if (!this._isActive) {
  205. return;
  206. }
  207. console.log(`[GroundBurnArea] 启动地面燃烧区域 - 持续时间: ${this._duration}秒`);
  208. // 启动更新循环
  209. this.schedule(this.updateBurnArea, 0.1); // 每0.1秒更新一次
  210. // 设置区域销毁定时器
  211. this.scheduleOnce(() => {
  212. this.destroyBurnArea();
  213. }, this._duration);
  214. }
  215. /**
  216. * 更新燃烧区域
  217. */
  218. private updateBurnArea(deltaTime: number): void {
  219. if (!this._isActive) {
  220. return;
  221. }
  222. // 更新伤害计时器
  223. this._damageTimer += deltaTime;
  224. // 检查是否到了造成伤害的时间
  225. if (this._damageTimer >= this._tickInterval) {
  226. console.log(`[GroundBurnArea] 燃烧区域伤害检查 - 区域内敌人数量: ${this._enemiesInArea.size}, 伤害值: ${this._damage}`);
  227. this.dealDamageToEnemiesInArea();
  228. this._damageTimer = 0;
  229. }
  230. // 清理无效的敌人节点
  231. this.cleanupInvalidEnemies();
  232. }
  233. /**
  234. * 对区域内的敌人造成伤害
  235. */
  236. private dealDamageToEnemiesInArea(): void {
  237. if (this._enemiesInArea.size === 0) {
  238. return;
  239. }
  240. console.log(`[GroundBurnArea] 对区域内 ${this._enemiesInArea.size} 个敌人造成燃烧伤害: ${this._damage}`);
  241. const eventBus = EventBus.getInstance();
  242. const attackStateManager = EnemyAttackStateManager.getInstance();
  243. // 对每个区域内的敌人造成伤害
  244. for (const enemyNode of this._enemiesInArea) {
  245. if (enemyNode && enemyNode.isValid) {
  246. // 检查敌人是否处于漂移状态
  247. const enemyInstance = enemyNode.getComponent('EnemyInstance') as any;
  248. if (enemyInstance && enemyInstance.isDrifting()) {
  249. console.log(`[GroundBurnArea] 敌人 ${enemyNode.name} 处于漂移状态,跳过燃烧伤害`);
  250. continue;
  251. }
  252. // 检查敌人是否可攻击(隐身等状态不可受伤害)
  253. if (attackStateManager && !attackStateManager.isEnemyAttackable(enemyNode)) {
  254. console.log(`[GroundBurnArea] 敌人不可攻击,跳过燃烧伤害: ${enemyNode.name}`);
  255. continue;
  256. }
  257. const damageData = {
  258. enemyNode: enemyNode,
  259. damage: this._damage,
  260. isCritical: false,
  261. source: 'GroundBurnArea'
  262. };
  263. eventBus.emit(GameEvents.APPLY_DAMAGE_TO_ENEMY, damageData);
  264. }
  265. }
  266. }
  267. /**
  268. * 清理无效的敌人节点
  269. */
  270. private cleanupInvalidEnemies(): void {
  271. const invalidEnemies: Node[] = [];
  272. for (const enemyNode of this._enemiesInArea) {
  273. if (!enemyNode || !enemyNode.isValid) {
  274. invalidEnemies.push(enemyNode);
  275. }
  276. }
  277. // 移除无效的敌人
  278. for (const invalidEnemy of invalidEnemies) {
  279. this._enemiesInArea.delete(invalidEnemy);
  280. }
  281. }
  282. /**
  283. * 设置火焰特效
  284. */
  285. private setupFlameEffect(): void {
  286. // 查找预制体中的 FlameEffect 子节点
  287. const flameEffectNode = this.node.getChildByName('FlameEffect');
  288. if (flameEffectNode) {
  289. this._burnEffectNode = flameEffectNode;
  290. // 启动火焰特效动画
  291. const skeleton = flameEffectNode.getComponent(sp.Skeleton);
  292. if (skeleton) {
  293. skeleton.setAnimation(0, 'animation', true);
  294. console.log('[GroundBurnArea] 火焰特效节点已找到并开始播放动画');
  295. } else {
  296. console.warn('[GroundBurnArea] FlameEffect 节点缺少 sp.Skeleton 组件');
  297. }
  298. } else {
  299. console.warn('[GroundBurnArea] 未找到 FlameEffect 子节点,请检查预制体配置');
  300. }
  301. }
  302. /**
  303. * 设置燃烧特效节点
  304. */
  305. public setBurnEffectNode(effectNode: Node): void {
  306. this._burnEffectNode = effectNode;
  307. }
  308. /**
  309. * 销毁燃烧区域
  310. */
  311. public destroyBurnArea(): void {
  312. console.log(`[GroundBurnArea] 销毁地面燃烧区域`);
  313. this._isActive = false;
  314. this.unscheduleAllCallbacks();
  315. // 清空敌人列表
  316. this._enemiesInArea.clear();
  317. // 销毁燃烧特效节点
  318. if (this._burnEffectNode && this._burnEffectNode.isValid) {
  319. console.log(`[GroundBurnArea] 销毁燃烧特效节点`);
  320. this._burnEffectNode.destroy();
  321. this._burnEffectNode = null;
  322. }
  323. // 通知管理器移除此燃烧区域
  324. const manager = GroundBurnAreaManager.getInstance();
  325. if (manager) {
  326. manager.removeBurnArea(this.node);
  327. }
  328. // 销毁节点
  329. if (this.node && this.node.isValid) {
  330. this.node.destroy();
  331. }
  332. }
  333. /**
  334. * 获取燃烧区域是否激活
  335. */
  336. public isActive(): boolean {
  337. return this._isActive;
  338. }
  339. /**
  340. * 获取区域内敌人数量
  341. */
  342. public getEnemyCount(): number {
  343. return this._enemiesInArea.size;
  344. }
  345. /**
  346. * 组件销毁时清理资源
  347. */
  348. protected onDestroy(): void {
  349. this._isActive = false;
  350. this.unscheduleAllCallbacks();
  351. // 清空敌人列表
  352. this._enemiesInArea.clear();
  353. // 清理燃烧特效节点
  354. if (this._burnEffectNode && this._burnEffectNode.isValid) {
  355. console.log(`[GroundBurnArea] 组件销毁时清理燃烧特效节点`);
  356. this._burnEffectNode.destroy();
  357. this._burnEffectNode = null;
  358. }
  359. // 通知管理器移除此燃烧区域
  360. const manager = GroundBurnAreaManager.getInstance();
  361. if (manager) {
  362. manager.removeBurnArea(this.node);
  363. }
  364. console.log(`[GroundBurnArea] 地面燃烧区域组件已销毁`);
  365. }
  366. }