/*
 * InspIRCd -- Internet Relay Chat Daemon
 *
 *   Copyright (C) 2019-2020 Sadie Powell <sadie@witchery.services>
 *
 * This file is part of InspIRCd.  InspIRCd is free software: you can
 * redistribute it and/or modify it under the terms of the GNU General Public
 * License as published by the Free Software Foundation, version 2.
 *
 * This program is distributed in the hope that it will be useful, but WITHOUT
 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
 * FOR A PARTICULAR PURPOSE.  See the GNU General Public License for more
 * details.
 *
 * You should have received a copy of the GNU General Public License
 * along with this program.  If not, see <http://www.gnu.org/licenses/>.
 */


#pragma once

#include "modules/cap.h"

namespace IRCv3
{
	namespace Replies
	{
		class CapReference;
		class Reply;
		class Fail;
		class Note;
		class Warn;
	}
}

/** Reference to the inspircd.org/standard-replies cap. */
class IRCv3::Replies::CapReference
	: public Cap::Reference
{
 public:
	CapReference(Module* mod)
		: Cap::Reference(mod, "inspircd.org/standard-replies")
	{
	}
};

/** Base class for standard replies. */
class IRCv3::Replies::Reply
{
 private:
	/** The name of the command for this reply. */
	const std::string cmd;

	/** The event provider for this reply. */
	ClientProtocol::EventProvider evprov;

	/** Wraps a message in an event and sends it to a user.
	 * @param user The user to send the message to.
	 * @param msg The message to send to the user.
	 */
	void SendInternal(LocalUser* user, ClientProtocol::Message& msg)
	{
		ClientProtocol::Event ev(evprov, msg);
		user->Send(ev);
	}

	void SendNoticeInternal(LocalUser* user, Command* command, const std::string& description)
	{
		user->WriteNotice(InspIRCd::Format("*** %s: %s", command->name.c_str(), description.c_str()));
	}

 protected:
	/** Initializes a new instance of the Reply class.
	 * @param Creator The module which created this instance.
	 * @param Cmd The name of the command to reply with.
	 */
	Reply(Module* Creator, const std::string& Cmd)
		: cmd(Cmd)
		, evprov(Creator, Cmd)
	{
	}

 public:
	/**
	 * Sends a standard reply to the specified user.
	 * @param user The user to send the reply to.
	 * @param command The command that the reply relates to.
	 * @param code A machine readable code for this reply.
	 * @param description A human readable description of this reply.
	 */
	void Send(LocalUser* user, Command* command, const std::string& code, const std::string& description)
	{
		ClientProtocol::Message msg(cmd.c_str(), ServerInstance->Config->ServerName);
		msg.PushParamRef(command->name);
		msg.PushParam(code);
		msg.PushParam(description);
		SendInternal(user, msg);
	}

	template<typename T1>
	void Send(LocalUser* user, Command* command, const std::string& code, const T1& p1, const std::string& description)
	{
		ClientProtocol::Message msg(cmd.c_str(), ServerInstance->Config->ServerName);
		msg.PushParamRef(command->name);
		msg.PushParam(code);
		msg.PushParam(ConvToStr(p1));
		msg.PushParam(description);
		SendInternal(user, msg);
	}

	template<typename T1, typename T2>
	void Send(LocalUser* user, Command* command, const std::string& code, const T1& p1, const T2& p2,
		const std::string& description)
	{
		ClientProtocol::Message msg(cmd.c_str(), ServerInstance->Config->ServerName);
		msg.PushParamRef(command->name);
		msg.PushParam(code);
		msg.PushParam(ConvToStr(p1));
		msg.PushParam(ConvToStr(p2));
		msg.PushParam(description);
		SendInternal(user, msg);
	}

	template<typename T1, typename T2, typename T3>
	void Send(LocalUser* user, Command* command, const std::string& code, const T1& p1, const T2& p2,
		const T3& p3, const std::string& description)
	{
		ClientProtocol::Message msg(cmd.c_str(), ServerInstance->Config->ServerName);
		msg.PushParamRef(command->name);
		msg.PushParam(code);
		msg.PushParam(ConvToStr(p1));
		msg.PushParam(ConvToStr(p2));
		msg.PushParam(ConvToStr(p3));
		msg.PushParam(description);
		SendInternal(user, msg);
	}

	template<typename T1, typename T2, typename T3, typename T4>
	void Send(LocalUser* user, Command* command, const std::string& code, const T1& p1, const T2& p2,
		const T3& p3, const T4& p4, const std::string& description)
	{
		ClientProtocol::Message msg(cmd.c_str(), ServerInstance->Config->ServerName);
		msg.PushParamRef(command->name);
		msg.PushParam(code);
		msg.PushParam(ConvToStr(p1));
		msg.PushParam(ConvToStr(p2));
		msg.PushParam(ConvToStr(p3));
		msg.PushParam(ConvToStr(p4));
		msg.PushParam(description);
		SendInternal(user, msg);
	}

	template<typename T1, typename T2, typename T3, typename T4, typename T5>
	void Send(LocalUser* user, Command* command, const std::string& code, const T1& p1, const T2& p2,
		const T3& p3, const T4& p4, const T5& p5, const std::string& description)
	{
		ClientProtocol::Message msg(cmd.c_str(), ServerInstance->Config->ServerName);
		if (command)
			msg.PushParamRef(command->name);
		else
			msg.PushParam("*");
		msg.PushParam(code);
		msg.PushParam(ConvToStr(p1));
		msg.PushParam(ConvToStr(p2));
		msg.PushParam(ConvToStr(p3));
		msg.PushParam(ConvToStr(p4));
		msg.PushParam(ConvToStr(p5));
		msg.PushParam(description);
		SendInternal(user, msg);
	}

	/**
	 * Sends a standard reply to the specified user if they have the specified cap
	 * or a notice if they do not.
	 * @param user The user to send the reply to.
	 * @param command The command that the reply relates to.
	 * @param code A machine readable code for this reply.
	 * @param description A human readable description of this reply.
	 */
	void SendIfCap(LocalUser* user, const Cap::Capability& cap, Command* command, const std::string& code,
		const std::string& description)
	{
		if (cap.get(user))
			Send(user, command, code, description);
		else
			SendNoticeInternal(user, command, description);
	}

	template<typename T1>
	void SendIfCap(LocalUser* user, const Cap::Capability& cap, Command* command, const std::string& code,
		const T1& p1, const std::string& description)
	{
		if (cap.get(user))
			Send(user, command, code, p1, description);
		else
			SendNoticeInternal(user, command, description);
	}

	template<typename T1, typename T2>
	void SendIfCap(LocalUser* user, const Cap::Capability& cap, Command* command, const std::string& code,
		const T1& p1, const T2& p2, const std::string& description)
	{
		if (cap.get(user))
			Send(user, command, code, p1, p2, description);
		else
			SendNoticeInternal(user, command, description);
	}

	template<typename T1, typename T2, typename T3>
	void SendIfCap(LocalUser* user, const Cap::Capability& cap, Command* command, const std::string& code,
		const T1& p1, const T2& p2, const T3& p3, const std::string& description)
	{
		if (cap.get(user))
			Send(user, command, code, p1, p2, p3, description);
		else
			SendNoticeInternal(user, command, description);
	}

	template<typename T1, typename T2, typename T3, typename T4>
	void SendIfCap(LocalUser* user, const Cap::Capability& cap, Command* command, const std::string& code,
		const T1& p1, const T2& p2, const T3& p3, const T4& p4, const std::string& description)
	{
		if (cap.get(user))
			Send(user, command, code, p1, p2, p3, p4, description);
		else
			SendNoticeInternal(user, command, description);
	}

	template<typename T1, typename T2, typename T3, typename T4, typename T5>
	void SendIfCap(LocalUser* user, const Cap::Capability& cap, Command* command, const std::string& code,
		const T1& p1, const T2& p2, const T3& p3, const T4& p4, const T5& p5, const std::string& description)
	{
		if (cap.get(user))
			Send(user, command, code, p1, p2, p3, p4, p5, description);
		else
			SendNoticeInternal(user, command, description);
	}
};

/** Sends a FAIL standard reply. */
class IRCv3::Replies::Fail
	: public IRCv3::Replies::Reply
{
public:
	/** Initializes a new instance of the Fail class.
	 * @param Creator The module which created this instance.
	 */
	Fail(Module* Creator)
		: Reply(Creator, "FAIL")
	{
	}
};

/** Sends a NOTE standard reply. */
class IRCv3::Replies::Note
	: public IRCv3::Replies::Reply
{
public:
	/** Initializes a new instance of the Note class.
	 * @param Creator The module which created this instance.
	 */
	Note(Module* Creator)
		: Reply(Creator, "NOTE")
	{
	}
};

/** Sends a WARN standard reply. */
class IRCv3::Replies::Warn
	: public IRCv3::Replies::Reply
{
public:
	/** Initializes a new instance of the Warn class.
	 * @param Creator The module which created this instance.
	 */
	Warn(Module* Creator)
		: Reply(Creator, "WARN")
	{
	}
};