#ifndef RENDERCOMPONENT_H
#define RENDERCOMPONENT_H

#include "BitStream.h"
#include <vector>
#include <string>
#include <unordered_map>

#include "Amf3.h"
#include "Component.h"
#include "eReplicaComponentType.h"

class Entity;

/**
 * An effect that plays for an entity. This might seem a bit abstract so digging through the CDClient is recommended
 * here.
 */
struct Effect {
	explicit Effect(const int32_t effectID, const std::string& name, const std::u16string& type, const float priority = 1.0f) noexcept
		: effectID{ effectID }
		, name{ name }
		, type{ type }
		, priority{ priority } {
	}

	/**
	 * The ID of the effect
	 */
	int32_t effectID = 0;

	/**
	 * The name of the effect
	 */
	std::string name = "";

	/**
	 * The type of the effect
	 */
	std::u16string type = u"";

	/**
	 * The importantness of the effect
	 */
	float priority = 1.0f;

	/**
	 * Some related entity that casted the effect
	 */
	uint64_t secondary = 0;

	/**
	 * The time that this effect plays for
	 */
	float time = 0;
};

/**
 * Determines that a component should be visibly rendered into the world, most entities have this. This component
 * also handles effects that play for entities.
 */
class RenderComponent final : public Component {
public:
	static constexpr eReplicaComponentType ComponentType = eReplicaComponentType::RENDER;

	RenderComponent(Entity* const parentEntity, const int32_t componentId = -1);

	void Serialize(RakNet::BitStream& outBitStream, bool bIsInitialUpdate) override;
	void Update(float deltaTime) override;

	/**
	 * Adds an effect to this entity, if successful the effect is returned
	 * @param effectId the ID of the effect
	 * @param name the name of the effect
	 * @param type the type of the effect
	 * @param priority the priority of the effect
	 * @return if successful, the effect that was created
	 */
	[[maybe_unused]] Effect& AddEffect(const int32_t effectId, const std::string& name, const std::u16string& type, const float priority);

	/**
	 * Removes an effect for this entity
	 * @param name the name of the effect to remove
	 */
	void RemoveEffect(const std::string& name);

	/**
	 * Plays an effect, removes any effects under this name and plays the one according to these params
	 * @param effectId the ID of the effect
	 * @param effectType the type of the effect
	 * @param name the name of the effect
	 * @param secondary some entity that cast the effect
	 * @param priority effect priority (determines if the client will play it over other effects)
	 * @param scale effect scale
	 * @param serialize whether to serialize the change or not
	 */
	void PlayEffect(int32_t effectId, const std::u16string& effectType, const std::string& name, LWOOBJID secondary = LWOOBJID_EMPTY, float priority = 1, float scale = 1, bool serialize = true);

	/**
	 * Removes and stops the effect for a certain name
	 * @param name name of the effect to stop
	 * @param killImmediate whether ot not to immediately stop playing the effect or phase it out
	 */
	void StopEffect(const std::string& name, bool killImmediate = true);

	/**
	 * Verifies that an animation can be played on this entity by checking
	 * if it has the animation assigned to its group.  If it does, the animation is echo'd
	 * down to all clients to be played and the duration of the played animation is returned.
	 * If the animation did not exist or the function was called in an invalid state, 0 is returned.
	 *
	 * The logic here matches the exact client logic.
	 *
	 * @param self The entity that wants to play an animation
	 * @param animation The animation_type (animationID in the client) to be played.
	 * @param sendAnimation Whether or not to echo the animation down to all clients.
	 * @param priority The priority of the animation.  Only used if sendAnimation is true.
	 * @param scale	The scale of the animation.  Only used if sendAnimation is true.
	 *
	 * @return The duration of the animation that was played.
	 */
	static float DoAnimation(Entity* self, const std::string& animation, bool sendAnimation, float priority = 0.0f, float scale = 1.0f);

	static float PlayAnimation(Entity* self, const std::u16string& animation, float priority = 0.0f, float scale = 1.0f);
	static float PlayAnimation(Entity* self, const std::string& animation, float priority = 0.0f, float scale = 1.0f);
	[[nodiscard]] static float GetAnimationTime(Entity* self, const std::string& animation);
	[[nodiscard]] static float GetAnimationTime(Entity* self, const std::u16string& animation);

	[[nodiscard]] const std::string& GetLastAnimationName() const { return m_LastAnimationName; };
	void SetLastAnimationName(const std::string& name) { m_LastAnimationName = name; };

private:

	/**
	 * List of currently active effects
	 */
	std::vector<Effect> m_Effects;

	std::vector<int32_t> m_animationGroupIds;

	// The last animationName that was played
	std::string m_LastAnimationName;

	/**
	 * Cache of queries that look for the length of each effect, indexed by effect ID
	 */
	static std::unordered_map<int32_t, float> m_DurationCache;
};

#endif // RENDERCOMPONENT_H