#include "testevents.h"
#include "sourcehook_test.h"

// Tests support for functions which return references

namespace
{
	StateList g_States;
	SourceHook::ISourceHook *g_SHPtr;
	SourceHook::Plugin g_PLID;

	MAKE_STATE_1(State_Func1_Pre1, int*);			// p1: the ref Func1_Pre1 is going to return
	MAKE_STATE_3(State_Func1_Pre2, META_RES, int*, int*);	// p1: current status
															// p2: override ret
															// p3: what this handler is going to supercede with
	MAKE_STATE_1(State_Func1, int*);				// p1: the ref Func1 is going to return
	MAKE_STATE_2(State_Func1_Post1, int*, int*);	// p1: orig_ret; p2: override_ret
	MAKE_STATE_1(State_Func1_Post2, int*);			// p1: what it's going to return
	MAKE_STATE_1(State_Func1_Ret, int*);			// p1: the ref it returned


	MAKE_STATE_2(State_Func2_Pre1, int, const int*);				// p1: func's p1; p2: what it's going to ret
	MAKE_STATE_2(State_Func2, int, const int*);						// p1: func's p1; p2: what it's going to ret
	MAKE_STATE_3(State_Func2_Post1, int, const int*, const int*);	// p1: func's p1; p2: orig ret; p3: override ret
	MAKE_STATE_1(State_Func2_Ret, const int*);						// p1: ret

	int g_Var;

	class Test
	{
	public:
		int m_Var1;
		int m_Var2;

		Test() : m_Var1(87)
		{
		}

		virtual int& Func1()
		{
			ADD_STATE(State_Func1(&m_Var1));
			return m_Var1;
		}
		virtual const int& Func2(int p1)
		{
			ADD_STATE(State_Func2(p1, &m_Var2));
			m_Var2 = p1;
			return m_Var2;
		}
	};

	SH_DECL_HOOK0(Test, Func1, SH_NOATTRIB, 0, int&);
	SH_DECL_MANUALHOOK0(Manual_Test_Func1, 0, 0, 0, int&);
	SH_DECL_HOOK1(Test, Func2, SH_NOATTRIB, 0, const int&, int);
	SH_DECL_MANUALHOOK1(Manual_Test_Func2, 1, 0, 0, const int&, int);

	class CHook
	{
	public:
		int m_Var;
		virtual int& Func1_Pre1()
		{
			ADD_STATE(State_Func1_Pre1(&m_Var));
			RETURN_META_VALUE(MRES_OVERRIDE, m_Var);
		}
		virtual int &Func1_Pre2()
		{
			int &overrideret = META_RESULT_OVERRIDE_RET(int&);
			overrideret = 1337;
			ADD_STATE(State_Func1_Pre2(META_RESULT_STATUS, &overrideret, &g_Var));
			RETURN_META_VALUE(MRES_SUPERCEDE, g_Var);
		}
		virtual int& Func1_Post1()
		{
			ADD_STATE(State_Func1_Post1(&META_RESULT_ORIG_RET(int&), &META_RESULT_OVERRIDE_RET(int&)));
			RETURN_META_NOREF(MRES_IGNORED, int&);
		}
		virtual int& Func1_Post2()
		{
			ADD_STATE(State_Func1_Post2(&m_Var));
			RETURN_META_VALUE(MRES_OVERRIDE, m_Var);
		}

		virtual const int& Func2_PreRecall_OverrideRet_Normal(int p1)
		{
			ADD_STATE(State_Func2_Pre1(p1, &m_Var));
			RETURN_META_VALUE_NEWPARAMS(MRES_OVERRIDE, m_Var, static_cast<const int& (Test::*)(int)>(&Test::Func2), (1337));
		}

		virtual const int& Func2_PreRecall_OverrideRet_Manual(int p1)
		{
			ADD_STATE(State_Func2_Pre1(p1, &m_Var));
			RETURN_META_VALUE_MNEWPARAMS(MRES_OVERRIDE, m_Var, Manual_Test_Func2, (1337));
		}

		virtual const int& Func2_Post1(int p1)
		{
			ADD_STATE(State_Func2_Post1(p1, &META_RESULT_ORIG_RET(int&), &META_RESULT_OVERRIDE_RET(int&)));
			RETURN_META_NOREF(MRES_IGNORED, const int&);
		}
	};

	Test *MyTestFactory()
	{
		return new Test;
	}
}

class TesterRefRet
{
public:
	Test *pTest;
	CHook hook;

	TesterRefRet()
	{
		pTest = NULL;
	}

	bool doTests(std::string &error)
	{
		GET_SHPTR(g_SHPtr);
		g_PLID = 1;

		pTest = MyTestFactory();
		CAutoPtrDestruction<Test> apd(pTest);

		int &ret1 = pTest->Func1();
		ADD_STATE(State_Func1_Ret(&ret1));

		CHECK_STATES((&g_States,
			new State_Func1(&pTest->m_Var1),
			new State_Func1_Ret(&pTest->m_Var1),
			NULL), "Part 1");

		// Now add Func1_Pre1, which supercedes and returns hook.m_Var
		AddHook__Test_Func1__Func1_Pre();

		int &ret2 = pTest->Func1();
		ADD_STATE(State_Func1_Ret(&ret2));
		
		CHECK_STATES((&g_States,
			new State_Func1_Pre1(&hook.m_Var),		// Pre1 says that it's going to override with hook.m_Var
			new State_Func1(&pTest->m_Var1),			// Function says that it's going to return pTest->m_Var1
			new State_Func1_Ret(&hook.m_Var),		// hook.m_Var is returned
			NULL), "Part 2");

		// Now add Func1_Post1, which only reports orig ret and override ret
		AddHook__Test_Func1__Func1_Post();

		int &ret3 = pTest->Func1();
		ADD_STATE(State_Func1_Ret(&ret3));
		
		CHECK_STATES((&g_States,
			new State_Func1_Pre1(&hook.m_Var),		// Pre1 says that it's going to override with hook.m_Var
			new State_Func1(&pTest->m_Var1),			// Function says that it's going to return pTest->m_Var1
			new State_Func1_Post1(&pTest->m_Var1, &hook.m_Var),	// origret(=p1) is what it wanted to
																// return, overrideret(=p2) is pre1's var
			new State_Func1_Ret(&hook.m_Var),		// hook.m_Var is returned
			NULL), "Part 3");

		// Now add Func1_Pre2, which supercedes and returns g_Var (it also sets the override ret from pre1 to 1337)
		// and add Func1_Post2, which overrides and returns hook.m_Var again.

		AddHook__Test_Func1__Func1_Pre2();
		AddHook__Test_Func1__Func1_Post2();

		int &ret4 = pTest->Func1();
		ADD_STATE(State_Func1_Ret(&ret4));

		CHECK_STATES((&g_States,
			new State_Func1_Pre1(&hook.m_Var),		// Pre1 says that it's going to override with hook.m_Var
			new State_Func1_Pre2(MRES_OVERRIDE,			// current status
								&hook.m_Var,			// override ret (which it set to 1337)
								&g_Var),				// what it's going to ret, AND supercede with

			new State_Func1_Post1(&g_Var, &g_Var),	// origret(=p1) is what pre2 superceded with,
													//  so  overrideret(=p2) has to be the same
			new State_Func1_Post2(&hook.m_Var),			// post2 is going to override with hook.m_Var again
			new State_Func1_Ret(&hook.m_Var),		// ==>>> hook.m_Var is returned
			NULL), "Part 4");

		CHECK_COND(hook.m_Var == 1337, "Part 4.1");
		
		// Through a callclass
		SourceHook::CallClass<Test> *cc1 = SH_GET_CALLCLASS(pTest);
		int &ret5 = SH_CALL(cc1, &Test::Func1)();
		ADD_STATE(State_Func1_Ret(&ret5));

		CHECK_STATES((&g_States,
			new State_Func1(&pTest->m_Var1),
			new State_Func1_Ret(&pTest->m_Var1),
			NULL), "Part 5");

		SH_RELEASE_CALLCLASS(cc1);

		////////////////////////////////////////////////////////////////////////////////////////
		// Func2 tests
		const int &ret21 = pTest->Func2(500);
		ADD_STATE(State_Func2_Ret(&ret21));

		CHECK_STATES((&g_States,
			new State_Func2(500, &pTest->m_Var2),
			new State_Func2_Ret(&pTest->m_Var2),
			NULL), "Part 6");

		AddHook__Test_Func2__Func2_Pre1();
		AddHook__Test_Func2__Func2_Post1();

		const int &ret22 = pTest->Func2(500);
		ADD_STATE(State_Func2_Ret(&ret22));

		CHECK_STATES((&g_States,
			new State_Func2_Pre1(500, &hook.m_Var),		// p1 was 500; it's going to override with hook.m_Var; and also change p1 to 1337
			new State_Func2(1337, &pTest->m_Var2),		// p1 was 1337; it's going to ret pTest->m_Var2
			new State_Func2_Post1(1337,					// p1 was 1337
				&pTest->m_Var2,							// orig ret was pTest->m_Var2
				&hook.m_Var),							// override ret was hook.m_Var
			new State_Func2_Ret(&hook.m_Var),			// really returned hook.m_Var
			NULL), "Part 7");
		
   		return true;
	}

	virtual void AddHook__Test_Func1__Func1_Pre() = 0;
	virtual void AddHook__Test_Func1__Func1_Post() = 0;

	virtual void AddHook__Test_Func1__Func1_Pre2() = 0;
	virtual void AddHook__Test_Func1__Func1_Post2() = 0;

	virtual void AddHook__Test_Func2__Func2_Pre1() = 0;
	virtual void AddHook__Test_Func2__Func2_Post1() = 0;
};

class TesterRefRetNonManual : public TesterRefRet
{
public:
	void AddHook__Test_Func1__Func1_Pre()
	{
		SH_ADD_HOOK(Test, Func1, pTest, SH_MEMBER(&hook, &CHook::Func1_Pre1), false);
	}

	void AddHook__Test_Func1__Func1_Post()
	{
		SH_ADD_HOOK(Test, Func1, pTest, SH_MEMBER(&hook, &CHook::Func1_Post1), true);
	}

	void AddHook__Test_Func1__Func1_Pre2()
	{
		SH_ADD_HOOK(Test, Func1, pTest, SH_MEMBER(&hook, &CHook::Func1_Pre2), false);
	}

	void AddHook__Test_Func1__Func1_Post2()
	{
		SH_ADD_HOOK(Test, Func1, pTest, SH_MEMBER(&hook, &CHook::Func1_Post2), true);
	}

	void AddHook__Test_Func2__Func2_Pre1()
	{
		SH_ADD_HOOK(Test, Func2, pTest, SH_MEMBER(&hook, &CHook::Func2_PreRecall_OverrideRet_Normal), false);
	}

	void AddHook__Test_Func2__Func2_Post1()
	{
		SH_ADD_HOOK(Test, Func2, pTest, SH_MEMBER(&hook, &CHook::Func2_Post1), true);
	}
};

class TesterRefRetManual : public TesterRefRet
{
public:
	void AddHook__Test_Func1__Func1_Pre()
	{
		SH_ADD_MANUALHOOK(Manual_Test_Func1, pTest, SH_MEMBER(&hook, &CHook::Func1_Pre1), false);
	}

	void AddHook__Test_Func1__Func1_Post()
	{
		SH_ADD_MANUALHOOK(Manual_Test_Func1, pTest, SH_MEMBER(&hook, &CHook::Func1_Post1), true);
	}

	void AddHook__Test_Func1__Func1_Pre2()
	{
		SH_ADD_MANUALHOOK(Manual_Test_Func1, pTest, SH_MEMBER(&hook, &CHook::Func1_Pre2), false);
	}

	void AddHook__Test_Func1__Func1_Post2()
	{
		SH_ADD_MANUALHOOK(Manual_Test_Func1, pTest, SH_MEMBER(&hook, &CHook::Func1_Post2), true);
	}

	void AddHook__Test_Func2__Func2_Pre1()
	{
		SH_ADD_MANUALHOOK(Manual_Test_Func2, pTest, SH_MEMBER(&hook, &CHook::Func2_PreRecall_OverrideRet_Manual), false);
	}

	void AddHook__Test_Func2__Func2_Post1()
	{
		SH_ADD_MANUALHOOK(Manual_Test_Func2, pTest, SH_MEMBER(&hook, &CHook::Func2_Post1), true);
	}
};

bool TestRefRet(std::string &error)
{
	TesterRefRet *testerNonManual = new TesterRefRetNonManual;
	bool resNonManual = testerNonManual->doTests(error);
	if (!resNonManual)
	{
		return false;
	}

	TesterRefRet *testerManual = new TesterRefRetManual;
	bool resManual = testerManual->doTests(error);
	if (!resManual)
	{
		return false;
	}
	return true;
}