18#ifndef TRINITY_CHATCOMMANDTAGS_H
19#define TRINITY_CHATCOMMANDTAGS_H
26#include <boost/preprocessor/repetition/repeat.hpp>
27#include <boost/preprocessor/punctuation/comma_if.hpp>
28#include <fmt/ostream.h>
51 struct tag_base<T,
std::enable_if_t<std::is_base_of_v<ContainerTag, T>>>
53 using type =
typename T::value_type;
57 inline constexpr char GetChar(
char const (&s)[N],
size_t i)
59 static_assert(N <= 25,
"The EXACT_SEQUENCE macro can only be used with up to 25 character long literals. Specify them char-by-char (null terminated) as parameters to ExactSequence<> instead.");
60 return i >= N ?
'\0' : s[i];
63#define CHATCOMMANDS_IMPL_SPLIT_LITERAL_EXTRACT_CHAR(z, i, strliteral) \
64 BOOST_PP_COMMA_IF(i) Trinity::Impl::ChatCommands::GetChar(strliteral, i)
66#define CHATCOMMANDS_IMPL_SPLIT_LITERAL_CONSTRAINED(maxlen, strliteral) \
67 BOOST_PP_REPEAT(maxlen, CHATCOMMANDS_IMPL_SPLIT_LITERAL_EXTRACT_CHAR, strliteral)
70#define CHATCOMMANDS_IMPL_SPLIT_LITERAL(strliteral) CHATCOMMANDS_IMPL_SPLIT_LITERAL_CONSTRAINED(25, strliteral)
89 template <
char... chars>
98 std::string_view start = args.substr(0,
_string.length());
102 if (remainingToken.empty())
104 start = args.substr(0,
_string.length() + remainingToken.length());
110 static constexpr std::array<char,
sizeof...(chars)>
_storage = { chars... };
111 static_assert(!
_storage.empty() && (
_storage.back() ==
'\0'),
"ExactSequence parameters must be null terminated! Use the EXACT_SEQUENCE macro to make this easier!");
115#define EXACT_SEQUENCE(str) Trinity::ChatCommands::ExactSequence<CHATCOMMANDS_IMPL_SPLIT_LITERAL(str)>
121 using std::string_view::operator=;
125 std::string_view::operator=(args);
126 return std::string_view();
134 using std::wstring::operator=;
139 return std::string_view();
160 operator std::string
const& ()
const {
return _name; }
161 operator std::string_view()
const {
return { _name }; }
164 std::string
const&
GetName()
const {
return _name; }
186 operator std::string
const&()
const {
return _name; }
187 operator std::string_view()
const {
return _name; }
189 std::string
const&
GetName()
const {
return _name; }
203 return FromSelf(handler);
212 template <
typename linktag>
230 if (info.
tag != linktag::tag())
234 if (!linktag::StoreTo(
val, info.
data))
255 template <
typename T>
258 template <
typename U>
265 template <
typename T1,
typename... Ts>
266 struct Variant :
public std::variant<T1, Ts...>
268 using base = std::variant<T1, Ts...>;
273 template <
bool C = have_operators>
279 template <
bool C = have_operators>
280 operator std::enable_if_t<C, first_type>()
const
285 template <
bool C = have_operators>
286 std::enable_if_t<C, bool>
operator!()
const {
return !**
this; }
288 template <
typename T>
291 template <
size_t index>
292 constexpr decltype(
auto)
get() {
return std::get<index>(
static_cast<base&
>(*
this)); }
293 template <
size_t index>
294 constexpr decltype(
auto)
get()
const {
return std::get<index>(
static_cast<base const&
>(*
this)); }
295 template <
typename type>
296 constexpr decltype(
auto)
get() {
return std::get<type>(
static_cast<base&
>(*
this)); }
297 template <
typename type>
298 constexpr decltype(
auto)
get()
const {
return std::get<type>(
static_cast<base const&
>(*
this)); }
300 template <
typename T>
301 constexpr decltype(
auto)
visit(T&& arg) {
return std::visit(std::forward<T>(arg),
static_cast<base&
>(*
this)); }
302 template <
typename T>
303 constexpr decltype(
auto)
visit(T&& arg)
const {
return std::visit(std::forward<T>(arg),
static_cast<base const&
>(*
this)); }
305 template <
typename T>
308 template <
bool C = have_operators>
316template <
typename T1,
typename... Ts>
#define STRING_VIEW_FMT_ARG(str)
@ LANG_CMDPARSER_EXACT_SEQ_MISMATCH
@ LANG_CMDPARSER_LINKDATA_INVALID
@ LANG_CMDPARSER_INVALID_UTF8
std::optional< T > Optional
Optional helper class to wrap optional values within.
bool StringEqualI(std::string_view a, std::string_view b)
bool Utf8toWStr(char const *utf8str, size_t csize, wchar_t *wstr, size_t &wsize)
Player session in the World.
HyperlinkInfo TC_GAME_API ParseSingleHyperlink(std::string_view str)
TokenizeResult tokenize(std::string_view args)
typename tag_base< T >::type tag_base_t
TC_GAME_API char const * GetTrinityString(ChatHandler const *handler, TrinityStrings which)
constexpr char GetChar(char const (&s)[N], size_t i)
TC_GAME_API std::string FormatTrinityString(std::string_view messageFormat, fmt::printf_args messageFormatArgs)
std::string const & GetName() const
WorldSession * GetConnectedSession()
static constexpr std::string_view _string
static constexpr std::array< char, sizeof...(chars)> _storage
ChatCommandResult TryConsume(ChatHandler const *handler, std::string_view args) const
std::remove_cvref_t< value_type > storage_type
ChatCommandResult TryConsume(ChatHandler const *handler, std::string_view args)
storage_type const * operator->() const
typename linktag::value_type value_type
value_type operator*() const
Player * GetConnectedPlayer() const
ObjectGuid GetGUID() const
std::string const & GetName() const
static Optional< PlayerIdentifier > FromTargetOrSelf(ChatHandler *handler)
TC_GAME_API ChatCommandResult TryConsume(ChatHandler const *handler, std::string_view args)
ChatCommandResult TryConsume(ChatHandler const *, std::string_view args)
std::string_view value_type
constexpr bool holds_alternative() const
constexpr decltype(auto) visit(T &&arg)
Trinity::Impl::ChatCommands::tag_base_t< T1 > first_type
std::enable_if_t< C, first_type > operator*() const
constexpr decltype(auto) visit(T &&arg) const
constexpr decltype(auto) get()
Variant & operator=(T &&arg)
std::enable_if_t< C, bool > operator!() const
std::variant< T1, Ts... > base
static constexpr bool have_operators
friend std::enable_if_t< C, std::ostream & > operator<<(std::ostream &os, Trinity::ChatCommands::Variant< T1, Ts... > const &v)
constexpr decltype(auto) get() const
ChatCommandResult TryConsume(ChatHandler const *handler, std::string_view args)
std::string_view const data
std::string_view const tag
std::string_view const tail
T operator()(U const &v) const
typename T::value_type type