import { Animation, Node, BoxCollider2D, Collider2D, Component, Vec2, Vec3, _decorator, Details, Sprite, Color } from "cc"; import { delay } from "../../../Services/Utils/AsyncUtils"; import { IInput } from "../../Input/IInput"; import { UnitHealth } from "../UnitHealth"; import { UnitLevel } from "../UnitLevel"; import { Magnet } from "./Magnet"; import { PlayerRegeneration } from "./PlayerRegeneration"; import { PlayerUI } from "./PlayerUI/PlayerUI"; import { Weapon } from "./Weapon/Weapon"; const { ccclass, property } = _decorator; @ccclass("Player") export class Player extends Component { @property(BoxCollider2D) private collider: BoxCollider2D; @property(PlayerUI) private playerUI: PlayerUI; @property(Weapon) private weapon: Weapon; @property(Magnet) private magnet: Magnet; @property(Node) private playerGraphics: Node; @property(Animation) private animation: Animation; @property(Sprite) private sprite: Sprite; private input: IInput; private health: UnitHealth; private level: UnitLevel; private regeneration: PlayerRegeneration; private speed: number; private isMoveAnimationPlaying = false; public init(input: IInput, data: PlayerData): void { this.input = input; this.health = new UnitHealth(data.maxHp); this.level = new UnitLevel(data.requiredXP, data.xpMultiplier); this.regeneration = new PlayerRegeneration(this.health, data.regenerationDelay); this.speed = data.speed; this.weapon.init(data.strikeDelay, data.damage); this.magnet.init(data.magnetDuration); this.health.HealthPointsChangeEvent.on(this.animateHpChange, this); this.playerUI.init(this.health); } public get Health(): UnitHealth { return this.health; } public get Level(): UnitLevel { return this.level; } public get Weapon(): Weapon { return this.weapon; } public get Magnet(): Magnet { return this.magnet; } public get Regeneration(): PlayerRegeneration { return this.regeneration; } public get Collider(): Collider2D { return this.collider; } public gameTick(deltaTime: number): void { this.move(deltaTime); this.weapon.gameTick(deltaTime); this.magnet.gameTick(deltaTime); this.regeneration.gameTick(deltaTime); } private move(deltaTime: number): void { if (!this.health.IsAlive) return; const movement: Vec2 = this.input.getAxis(); if (!movement.equals(Vec2.ZERO)) { movement.x *= deltaTime * this.speed; movement.y *= deltaTime * this.speed; const newPosition: Vec3 = this.node.worldPosition; newPosition.x += movement.x; newPosition.y += movement.y; this.node.setWorldPosition(newPosition); if (!this.isMoveAnimationPlaying) { this.isMoveAnimationPlaying = true; this.animation.play("Move"); } if (movement.x < 0) { this.playerGraphics.setScale(new Vec3(1, 1, 1)); } else if (0 < movement.x) { this.playerGraphics.setScale(new Vec3(-1, 1, 1)); } } else { if (this.isMoveAnimationPlaying) { this.isMoveAnimationPlaying = false; this.animation.play("Idle"); } } } private async animateHpChange(hpChange: number): Promise { if (hpChange < 0) { this.sprite.color = Color.RED; } else { this.sprite.color = Color.GREEN; } await delay(100); this.sprite.color = Color.WHITE; if (!this.health.IsAlive) { this.animation.play("Die"); } } } export class PlayerData { public requiredXP: number[] = []; public speed = 0; public maxHp = 0; public regenerationDelay = 0; public xpMultiplier = 0; public goldMultiplier = 0; // Weapon public strikeDelay = 0; public damage = 0; // Magnet public magnetDuration = 0; }