aboutsummaryrefslogtreecommitdiff
path: root/src/server
diff options
context:
space:
mode:
Diffstat (limited to 'src/server')
-rw-r--r--src/server/game/Chat/ChatCommands/ChatCommandArgs.h79
-rw-r--r--src/server/scripts/Commands/cs_npc.cpp15
2 files changed, 83 insertions, 11 deletions
diff --git a/src/server/game/Chat/ChatCommands/ChatCommandArgs.h b/src/server/game/Chat/ChatCommands/ChatCommandArgs.h
index a2ebccf03bb..6b5121c5044 100644
--- a/src/server/game/Chat/ChatCommands/ChatCommandArgs.h
+++ b/src/server/game/Chat/ChatCommands/ChatCommandArgs.h
@@ -20,6 +20,9 @@
#include "ChatCommandHelpers.h"
#include "ChatCommandTags.h"
+#include "SmartEnum.h"
+#include "Util.h"
+#include <map>
struct GameTele;
@@ -99,6 +102,82 @@ struct ArgInfo<std::string, void>
}
};
+// enum
+template <typename T>
+struct ArgInfo<T, std::enable_if_t<std::is_enum_v<T>>>
+{
+ static std::map<std::string, Optional<T>> MakeSearchMap()
+ {
+ std::map<std::string, Optional<T>> map;
+ for (T val : EnumUtils::Iterate<T>())
+ {
+ EnumText text = EnumUtils::ToString(val);
+
+ std::string title(text.Title);
+ strToLower(title);
+ std::string constant(text.Constant);
+ strToLower(constant);
+
+ auto [constantIt, constantNew] = map.try_emplace(constant, val);
+ if (!constantNew)
+ constantIt->second = std::nullopt;
+
+ if (title != constant)
+ {
+ auto [titleIt, titleNew] = map.try_emplace(title, val);
+ if (!titleNew)
+ titleIt->second = std::nullopt;
+ }
+ }
+ return map;
+ }
+
+ static inline std::map<std::string, Optional<T>> const SearchMap = MakeSearchMap();
+
+ static T const* Match(std::string s)
+ {
+ strToLower(s);
+
+ auto it = SearchMap.lower_bound(s);
+ if (it == SearchMap.end() || !StringStartsWith(it->first, s)) // not a match
+ return nullptr;
+
+ auto it2 = it;
+ ++it2;
+ if (it2 != SearchMap.end() && StringStartsWith(it2->first, s)) // not unique
+ return nullptr;
+
+ if (it->second)
+ return &*it->second;
+ else
+ return nullptr;
+ }
+
+ static char const* TryConsume(T& val, char const* args)
+ {
+ std::string strVal;
+ char const* ret = ArgInfo<std::string>::TryConsume(strVal, args);
+
+ if (!ret)
+ return nullptr;
+
+ if (T const* tmpVal = Match(strVal))
+ {
+ val = *tmpVal;
+ return ret;
+ }
+
+ // Value not found. Try to parse arg as underlying type and cast it to enum type
+ using U = std::underlying_type_t<T>;
+ U uVal = 0;
+ ret = ArgInfo<U>::TryConsume(uVal, args);
+ if (ret)
+ val = static_cast<T>(uVal);
+
+ return ret;
+ }
+};
+
// a container tag
template <typename T>
struct ArgInfo<T, std::enable_if_t<std::is_base_of_v<ContainerTag, T>>>
diff --git a/src/server/scripts/Commands/cs_npc.cpp b/src/server/scripts/Commands/cs_npc.cpp
index fd77e4dbea9..d30d77f4f39 100644
--- a/src/server/scripts/Commands/cs_npc.cpp
+++ b/src/server/scripts/Commands/cs_npc.cpp
@@ -549,13 +549,8 @@ public:
}
//set npcflag of creature
- static bool HandleNpcSetFlagCommand(ChatHandler* handler, char const* args)
+ static bool HandleNpcSetFlagCommand(ChatHandler* handler, NPCFlags npcFlags, NPCFlags2 npcFlags2)
{
- if (!*args)
- return false;
-
- uint64 npcFlags = atoull(args);
-
Creature* creature = handler->getSelectedCreature();
if (!creature)
@@ -565,14 +560,12 @@ public:
return false;
}
- uint32 raw[2];
- memcpy(raw, &npcFlags, sizeof(raw));
- creature->SetNpcFlags(NPCFlags(raw[0]));
- creature->SetNpcFlags2(NPCFlags2(raw[1]));
+ creature->SetNpcFlags(npcFlags);
+ creature->SetNpcFlags2(npcFlags2);
WorldDatabasePreparedStatement* stmt = WorldDatabase.GetPreparedStatement(WORLD_UPD_CREATURE_NPCFLAG);
- stmt->setUInt64(0, npcFlags);
+ stmt->setUInt64(0, uint64(npcFlags) | (uint64(npcFlags2) << 32));
stmt->setUInt32(1, creature->GetEntry());
WorldDatabase.Execute(stmt);