diff options
Diffstat (limited to 'src/server')
| -rw-r--r-- | src/server/game/Chat/ChatCommands/ChatCommandArgs.h | 79 | ||||
| -rw-r--r-- | src/server/scripts/Commands/cs_npc.cpp | 15 |
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); |
