#pragma once
#include "olcUTIL_Animate2D.h"
#include "Animation.h"
#include "Monster.h"
#include "State.h"
#include "Ability.h"
#include "Class.h"
#include "Buff.h"
#include "Pathfinding.h"

struct CastInfo{
	std::string name;
	float castTimer;
	float castTotalTime;
	vf2d castPos;
};

struct Player{
	friend class Crawler;
	friend class sig::Animation;
	friend class Warrior;
	friend class Thief;
	friend class Ranger;
	friend class Trapper;
	friend class Wizard;
	friend class Witch;
	private:
	int hp=100,maxhp=hp;
	int mana=100,maxmana=mana;
	int atk=10;
	vf2d pos;
	float z=0;
	float moveSpd=1.0f;
	float size=1.0f;
	float spin_attack_timer=0;
	float spin_spd=0;
	float spin_angle=0;
	float lastAnimationFlip=0;
	float manaTickTimer=0;
	std::pair<std::string,float> notEnoughManaDisplay={"",0};
	float teleportAttemptWaitTime=0; //If a teleport fails, we wait awhile before trying again, it's expensive.
	State state=State::NORMAL;
	Animate2D::Animation<AnimationState>animation;
	Animate2D::AnimationState internal_animState;
	Key lastReleasedMovementKey;
	void Update(float fElapsedTime);
	void AddAnimation(AnimationState state);
	std::vector<Buff>buffList;
	CastInfo castInfo={"",0};
	vf2d movementVelocity={};//This tells us if the player is moving (mostly controlled by user input) since their velocity is not used for regular movement.
protected:
	const float ATTACK_COOLDOWN=0.35f;
	const float MAGIC_ATTACK_COOLDOWN=0.85f;
	const float ARROW_ATTACK_COOLDOWN=0.6f;
	void SetSwordSwingTimer(float val);
	void SetState(State newState);
	void SetFacingDirection(Key direction);
	void SetLastReleasedMovementKey(Key k);
	void Spin(float duration,float spinSpd);
	//Returns true if the move was valid and successful.
	bool SetX(float x);
	//Returns true if the move was valid and successful.
	bool SetY(float y);
	void SetZ(float z);
	//Returns true if the move was valid and successful.
	bool SetPos(vf2d pos);
	float friction=400;
	float attack_cooldown_timer=0;
	float iframe_time=0;
	float teleportAnimationTimer=0;
	vf2d teleportTarget={};
	vf2d teleportStartPosition={};
	std::pair<std::string,float> notificationDisplay={"",0};
	bool upperLevel=false;
	vf2d vel={0,0};
	float attack_range=1.5f;
	Key facingDirection=DOWN;
	float swordSwingTimer=0;
	void CastSpell(Ability&ability);
	Ability*castPrepAbility;
	void PrepareCast(Ability&ability);
	vf2d precastLocation={};
	void SetVelocity(vf2d vel);
	const float RETREAT_DISTANCE=24*2.5;
	const float RETREAT_TIME=0.2; //How long the Retreat ability takes.
	const int RETREAT_GHOST_FRAMES=8;
	const float RETREAT_GHOST_FRAME_DELAY=0.025;
	float ghostFrameTimer=0;
	float ghostRemoveTimer=0;
	float retreatTimer=0;
	std::vector<vf2d>ghostPositions;
	float rapidFireTimer=0;
	int remainingRapidFireShots=0;
	const float RAPID_FIRE_SHOOT_DELAY=0.1;
	const int RAPID_FIRE_SHOOT_AMOUNT=4;
public:
	Player();
	//So this is rather fascinating and only exists because we have the ability to change classes which means we need to initialize a class
	//using a new object type... Because of that we'll take the pointer reference to the old object and copy some of its properties to this new
	//one. It's hackish but it means we can reduce the amount of extra boilerplate when class changing...I don't know how to feel about this.
	Player(Player*player);
	const static float GROUND_SLAM_SPIN_TIME;
	vf2d&GetPos();
	float GetX();
	float GetY();
	float GetZ();
	int GetHealth();
	int GetMaxHealth();
	int GetMana();
	int GetMaxMana();
	int GetAttack();
	float GetMoveSpdMult();
	float GetSizeMult();
	float GetAttackRangeMult();
	float GetSpinAngle();
	State GetState();
	Key GetFacingDirection();
	vf2d GetVelocity();
	bool HasIframes();
	void UpdateWalkingAnimation(Key direction);
	void UpdateIdleAnimation(Key direction);
	//The range is the search range in tiles.
	bool CanPathfindTo(vf2d pos,vf2d targetPos,float range=8);
	bool CanMove();

	void AddBuff(BuffType type,float duration,float intensity);
	std::vector<Buff>GetBuffs(BuffType buff);

	bool Hurt(int damage,bool onUpperLevel);
	//specificClass is a bitwise-combination of classes from the Class enum. It makes sure certain animations only play if you are a certain class.
	void UpdateAnimation(AnimationState animState,int specificClass=ANY);
	Animate2D::Frame GetFrame();
	Key GetLastReleasedMovementKey();
	float GetSwordSwingTimer();
	bool OnUpperLevel();
	//Triggers when the player has moved.
	void Moved();
	virtual ~Player()=default;
	virtual Class GetClass()=0;
	virtual bool AutoAttack()=0;
	virtual void OnUpdate(float fElapsedTime)=0;
	virtual std::string GetClassName()=0;
	virtual Ability&GetRightClickAbility()=0;
	virtual Ability&GetAbility1()=0;
	virtual Ability&GetAbility2()=0;
	virtual Ability&GetAbility3()=0;
	virtual Ability&GetAbility4()=0;
	virtual AnimationState&GetWalkNAnimation()=0;
	virtual AnimationState&GetWalkEAnimation()=0;
	virtual AnimationState&GetWalkSAnimation()=0;
	virtual AnimationState&GetWalkWAnimation()=0;
	virtual AnimationState&GetIdleNAnimation()=0;
	virtual AnimationState&GetIdleEAnimation()=0;
	virtual AnimationState&GetIdleSAnimation()=0;
	virtual AnimationState&GetIdleWAnimation()=0;

	CastInfo&GetCastInfo();
	void SetAnimationBasedOnTargetingDirection(float targetDirection);
};

struct Warrior:Player{
	static std::string name;
	static Class cl;
	static Ability rightClickAbility,ability1,ability2,ability3,ability4;
	static AnimationState walk_n,walk_e,walk_s,walk_w,idle_n,idle_e,idle_s,idle_w;
	Warrior();
	Warrior(Player*player);
	Class GetClass()override;
	bool AutoAttack()override;
	//Include only WARRIOR-specific implementations!
	void OnUpdate(float fElapsedTime)override;
	static void InitializeClassAbilities();
	std::string GetClassName()override;
	Ability&GetRightClickAbility()override;
	Ability&GetAbility1()override;
	Ability&GetAbility2()override;
	Ability&GetAbility3()override;
	Ability&GetAbility4()override;
	AnimationState&GetWalkNAnimation()override;
	AnimationState&GetWalkEAnimation()override;
	AnimationState&GetWalkSAnimation()override;
	AnimationState&GetWalkWAnimation()override;
	AnimationState&GetIdleNAnimation()override;
	AnimationState&GetIdleEAnimation()override;
	AnimationState&GetIdleSAnimation()override;
	AnimationState&GetIdleWAnimation()override;
};

struct Thief:Player{
	static std::string name;
	static Class cl;
	static Ability rightClickAbility,ability1,ability2,ability3,ability4;
	static AnimationState walk_n,walk_e,walk_s,walk_w,idle_n,idle_e,idle_s,idle_w;
	Thief();
	Thief(Player*player);
	Class GetClass()override;
	bool AutoAttack()override;
	//Include only THIEF-specific implementations!
	void OnUpdate(float fElapsedTime)override;
	static void InitializeClassAbilities();
	std::string GetClassName()override;
	Ability&GetRightClickAbility()override;
	Ability&GetAbility1()override;
	Ability&GetAbility2()override;
	Ability&GetAbility3()override;
	Ability&GetAbility4()override;
	AnimationState&GetWalkNAnimation()override;
	AnimationState&GetWalkEAnimation()override;
	AnimationState&GetWalkSAnimation()override;
	AnimationState&GetWalkWAnimation()override;
	AnimationState&GetIdleNAnimation()override;
	AnimationState&GetIdleEAnimation()override;
	AnimationState&GetIdleSAnimation()override;
	AnimationState&GetIdleWAnimation()override;
};

struct Ranger:Player{
	static std::string name;
	static Class cl;
	static Ability rightClickAbility,ability1,ability2,ability3,ability4;
	static AnimationState walk_n,walk_e,walk_s,walk_w,idle_n,idle_e,idle_s,idle_w;
	Ranger();
	Ranger(Player*player);
	Class GetClass()override;
	bool AutoAttack()override;
	//Include only RANGER-specific implementations!
	void OnUpdate(float fElapsedTime)override;
	static void InitializeClassAbilities();
	std::string GetClassName()override;
	Ability&GetRightClickAbility()override;
	Ability&GetAbility1()override;
	Ability&GetAbility2()override;
	Ability&GetAbility3()override;
	Ability&GetAbility4()override;
	AnimationState&GetWalkNAnimation()override;
	AnimationState&GetWalkEAnimation()override;
	AnimationState&GetWalkSAnimation()override;
	AnimationState&GetWalkWAnimation()override;
	AnimationState&GetIdleNAnimation()override;
	AnimationState&GetIdleEAnimation()override;
	AnimationState&GetIdleSAnimation()override;
	AnimationState&GetIdleWAnimation()override;
};

struct Trapper:Player{
	static std::string name;
	static Class cl;
	static Ability rightClickAbility,ability1,ability2,ability3,ability4;
	static AnimationState walk_n,walk_e,walk_s,walk_w,idle_n,idle_e,idle_s,idle_w;
	Trapper();
	Trapper(Player*player);
	Class GetClass()override;
	bool AutoAttack()override;
	//Include only TRAPPER-specific implementations!
	void OnUpdate(float fElapsedTime)override;
	static void InitializeClassAbilities();
	std::string GetClassName()override;
	Ability&GetRightClickAbility()override;
	Ability&GetAbility1()override;
	Ability&GetAbility2()override;
	Ability&GetAbility3()override;
	Ability&GetAbility4()override;
	AnimationState&GetWalkNAnimation()override;
	AnimationState&GetWalkEAnimation()override;
	AnimationState&GetWalkSAnimation()override;
	AnimationState&GetWalkWAnimation()override;
	AnimationState&GetIdleNAnimation()override;
	AnimationState&GetIdleEAnimation()override;
	AnimationState&GetIdleSAnimation()override;
	AnimationState&GetIdleWAnimation()override;
};

struct Wizard:Player{
	static std::string name;
	static Class cl;
	static Ability rightClickAbility,ability1,ability2,ability3,ability4;
	static AnimationState walk_n,walk_e,walk_s,walk_w,idle_n,idle_e,idle_s,idle_w;
	Wizard();
	Wizard(Player*player);
	Class GetClass()override;
	bool AutoAttack()override;
	//Include only WIZARD-specific implementations!
	void OnUpdate(float fElapsedTime)override;
	static void InitializeClassAbilities();
	std::string GetClassName()override;
	Ability&GetRightClickAbility()override;
	Ability&GetAbility1()override;
	Ability&GetAbility2()override;
	Ability&GetAbility3()override;
	Ability&GetAbility4()override;
	AnimationState&GetWalkNAnimation()override;
	AnimationState&GetWalkEAnimation()override;
	AnimationState&GetWalkSAnimation()override;
	AnimationState&GetWalkWAnimation()override;
	AnimationState&GetIdleNAnimation()override;
	AnimationState&GetIdleEAnimation()override;
	AnimationState&GetIdleSAnimation()override;
	AnimationState&GetIdleWAnimation()override;
};

struct Witch:Player{
	static std::string name;
	static Class cl;
	static Ability rightClickAbility,ability1,ability2,ability3,ability4;
	static AnimationState walk_n,walk_e,walk_s,walk_w,idle_n,idle_e,idle_s,idle_w;
	Witch();
	Witch(Player*player);
	Class GetClass()override;
	bool AutoAttack()override;
	//Include only WITCHs-specific implementations!
	void OnUpdate(float fElapsedTime)override;
	static void InitializeClassAbilities();
	std::string GetClassName()override;
	Ability&GetRightClickAbility()override;
	Ability&GetAbility1()override;
	Ability&GetAbility2()override;
	Ability&GetAbility3()override;
	Ability&GetAbility4()override;
	AnimationState&GetWalkNAnimation()override;
	AnimationState&GetWalkEAnimation()override;
	AnimationState&GetWalkSAnimation()override;
	AnimationState&GetWalkWAnimation()override;
	AnimationState&GetIdleNAnimation()override;
	AnimationState&GetIdleEAnimation()override;
	AnimationState&GetIdleSAnimation()override;
	AnimationState&GetIdleWAnimation()override;
};