mirror of
https://github.com/EQEmu/Server.git
synced 2025-12-11 21:01:29 +00:00
- Fix ^discipline saying no bots were selected - Adds more information to certain commands to explain how they function. - Rewrote ^copysettings and ^defaultsettings to not be so bloated and accept arguments properly. - Added long names for setting categories - Add Spell ID output to ^spells
259 lines
6.6 KiB
C++
259 lines
6.6 KiB
C++
#include "../bot_command.h"
|
|
|
|
void bot_command_discipline(Client* c, const Seperator* sep)
|
|
{
|
|
if (helper_command_alias_fail(c, "bot_command_discipline", sep->arg[0], "discipline")) {
|
|
c->Message(Chat::White, "note: Tells applicable bots to use the specified disciplines.");
|
|
|
|
return;
|
|
}
|
|
|
|
if (helper_is_help_or_usage(sep->arg[1])) {
|
|
BotCommandHelpParams p;
|
|
|
|
p.description =
|
|
{
|
|
"Tells applicable bots to use the specified disciplines."
|
|
};
|
|
p.notes = { "Aside from Lay On Hands and Harm Touch, you will need to know the spell ID of the discipline to tell a bot to attempt to use it." };
|
|
p.example_format = { fmt::format("{} [aggressive | defensive | spell ID] [actionable, default: spawned]", sep->arg[0]) };
|
|
p.examples_one =
|
|
{
|
|
"To tell all bots to use an aggressive discipline:",
|
|
fmt::format(
|
|
"{} aggressive spawned",
|
|
sep->arg[0]
|
|
)
|
|
};
|
|
p.examples_two =
|
|
{
|
|
"To tell Warrior bots to use a defensive discipline:",
|
|
fmt::format(
|
|
"{} defensive byclass {}",
|
|
sep->arg[0],
|
|
Class::Warrior
|
|
)
|
|
};
|
|
p.examples_three =
|
|
{
|
|
"To tell all bots to use their Fearless discipline:",
|
|
fmt::format(
|
|
"{} 4587 spawned",
|
|
sep->arg[0]
|
|
)
|
|
};
|
|
p.actionables = { "target, byname, ownergroup, ownerraid, targetgroup, namesgroup, healrotationtargets, mmr, byclass, byrace, spawned" };
|
|
|
|
std::string popup_text = c->SendBotCommandHelpWindow(p);
|
|
popup_text = DialogueWindow::Table(popup_text);
|
|
|
|
c->SendPopupToClient(sep->arg[0], popup_text.c_str());
|
|
|
|
if (RuleB(Bots, SendClassRaceOnHelp)) {
|
|
c->Message(
|
|
Chat::Yellow,
|
|
fmt::format(
|
|
"Use {} for information about race/class IDs.",
|
|
Saylink::Silent("^classracelist")
|
|
).c_str()
|
|
);
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
std::string arg1 = sep->arg[1];
|
|
std::string arg2 = sep->arg[2];
|
|
int ab_arg = 2;
|
|
bool aggressive = false;
|
|
bool defensive = false;
|
|
Mob* tar = c->GetTarget();
|
|
uint16 spell_id = UINT16_MAX;
|
|
|
|
if (!arg1.compare("aggressive")) {
|
|
aggressive = true;
|
|
}
|
|
else if (!arg1.compare("defensive")) {
|
|
defensive = true;
|
|
}
|
|
else if (sep->IsNumber(1)) {
|
|
if (!IsValidSpell(atoi(sep->arg[1]))) {
|
|
c->Message(Chat::Yellow, "You must enter a valid spell ID.");
|
|
return;
|
|
}
|
|
|
|
spell_id = atoi(sep->arg[1]);
|
|
}
|
|
else {
|
|
c->Message(
|
|
Chat::Yellow,
|
|
fmt::format(
|
|
"Incorrect argument, use {} for information regarding this command.",
|
|
Saylink::Silent(
|
|
fmt::format("{} help", sep->arg[0])
|
|
)
|
|
).c_str()
|
|
);
|
|
|
|
return;
|
|
}
|
|
|
|
const int ab_mask = ActionableBots::ABM_Type1;
|
|
std::string actionable_arg = sep->arg[ab_arg];
|
|
|
|
if (actionable_arg.empty()) {
|
|
actionable_arg = "spawned";
|
|
}
|
|
|
|
std::string class_race_arg = sep->arg[ab_arg];
|
|
bool class_race_check = false;
|
|
|
|
if (!class_race_arg.compare("byclass") || !class_race_arg.compare("byrace")) {
|
|
class_race_check = true;
|
|
}
|
|
|
|
std::vector<Bot*> sbl;
|
|
|
|
if (ActionableBots::PopulateSBL(c, actionable_arg, sbl, ab_mask, !class_race_check ? sep->arg[ab_arg + 1] : nullptr, class_race_check ? atoi(sep->arg[ab_arg + 1]) : 0) == ActionableBots::ABT_None) {
|
|
return;
|
|
}
|
|
|
|
sbl.erase(std::remove(sbl.begin(), sbl.end(), nullptr), sbl.end());
|
|
|
|
bool is_success = false;
|
|
uint16 success_count = 0;
|
|
Bot* first_found = nullptr;
|
|
|
|
for (auto bot_iter : sbl) {
|
|
if (!bot_iter->ValidStateCheck(c)) {
|
|
continue;
|
|
}
|
|
|
|
if (spell_id == UINT16_MAX) { // Aggressive/Defensive type
|
|
const std::vector<BotSpells_wIndex>& bot_spell_list = bot_iter->BotGetSpellsByType(aggressive ? BotSpellTypes::DiscAggressive : BotSpellTypes::DiscDefensive);
|
|
|
|
for (int i = bot_spell_list.size() - 1; i >= 0; i--) {
|
|
if (!IsValidSpell(bot_spell_list[i].spellid)) {
|
|
continue;
|
|
}
|
|
|
|
if (!bot_iter->CheckDisciplineReuseTimer(bot_spell_list[i].spellid)) {
|
|
uint32 remaining_time = (bot_iter->GetDisciplineReuseRemainingTime(bot_spell_list[i].spellid) / 1000);
|
|
|
|
bot_iter->OwnerMessage(
|
|
fmt::format(
|
|
"I can use this discipline in {}.",
|
|
Strings::SecondsToTime(remaining_time)
|
|
)
|
|
);
|
|
|
|
continue;
|
|
}
|
|
|
|
if (bot_iter->GetEndurance() < spells[bot_spell_list[i].spellid].endurance_cost) {
|
|
continue;
|
|
}
|
|
|
|
if (bot_iter->DivineAura() && !IsCastNotStandingSpell(bot_spell_list[i].spellid)) {
|
|
continue;
|
|
}
|
|
|
|
if (spells[bot_spell_list[i].spellid].buff_duration_formula != 0 && spells[bot_spell_list[i].spellid].target_type == ST_Self && bot_iter->HasDiscBuff()) {
|
|
continue;
|
|
}
|
|
|
|
if (!tar || (spells[bot_spell_list[i].spellid].target_type == ST_Self && tar != bot_iter)) {
|
|
tar = bot_iter;
|
|
}
|
|
|
|
if (bot_iter->AttemptForcedCastSpell(tar, bot_spell_list[i].spellid, true)) {
|
|
if (!first_found) {
|
|
first_found = bot_iter;
|
|
}
|
|
|
|
is_success = true;
|
|
++success_count;
|
|
spell_id = bot_spell_list[i].spellid;
|
|
}
|
|
}
|
|
}
|
|
else { // Direct spell ID
|
|
if (!IsValidSpell(spell_id)) {
|
|
continue;
|
|
}
|
|
|
|
SPDat_Spell_Struct spell = spells[spell_id];
|
|
|
|
if (!bot_iter->CanUseBotSpell(spell_id)) {
|
|
continue;
|
|
}
|
|
|
|
if (!bot_iter->CheckDisciplineReuseTimer(spell_id)) {
|
|
uint32 remaining_time = (bot_iter->GetDisciplineReuseRemainingTime(spell_id) / 1000);
|
|
|
|
bot_iter->OwnerMessage(
|
|
fmt::format(
|
|
"I can use this item in {}.",
|
|
Strings::SecondsToTime(remaining_time)
|
|
)
|
|
);
|
|
|
|
continue;
|
|
}
|
|
|
|
if (bot_iter->GetEndurance() < spell.endurance_cost) {
|
|
continue;
|
|
}
|
|
|
|
if (bot_iter->DivineAura() && !IsCastNotStandingSpell(spell_id)) {
|
|
continue;
|
|
}
|
|
|
|
if (!tar || (spell.target_type == ST_Self && tar != bot_iter)) {
|
|
tar = bot_iter;
|
|
}
|
|
|
|
if (bot_iter->AttemptForcedCastSpell(tar, spell_id, true)) {
|
|
if (!first_found) {
|
|
first_found = bot_iter;
|
|
}
|
|
|
|
is_success = true;
|
|
++success_count;
|
|
}
|
|
}
|
|
|
|
continue;
|
|
}
|
|
|
|
if (!is_success) {
|
|
c->Message(Chat::Yellow, "No bots were capable of doing that. This could be because they don't have the ability or things like Line of Sight, range, endurance, etc.");
|
|
}
|
|
else {
|
|
if (aggressive || defensive) {
|
|
c->Message(
|
|
Chat::Yellow,
|
|
fmt::format(
|
|
"{} {} {} {} discipline.",
|
|
((success_count == 1 && first_found) ? first_found->GetCleanName() : (fmt::format("{}", success_count).c_str())),
|
|
((success_count == 1 && first_found) ? "used" : "of your bots used"),
|
|
(aggressive ? "an" : "a"),
|
|
(aggressive ? "aggressive" : "defensive")
|
|
).c_str()
|
|
);
|
|
}
|
|
else {
|
|
c->Message(
|
|
Chat::Yellow,
|
|
fmt::format(
|
|
"{} {} their {} [#{}] discipline.",
|
|
((success_count == 1 && first_found) ? first_found->GetCleanName() : (fmt::format("{}", success_count).c_str())),
|
|
((success_count == 1 && first_found) ? "used" : "of your bots used"),
|
|
spells[spell_id].name,
|
|
spell_id
|
|
).c_str()
|
|
);
|
|
}
|
|
}
|
|
}
|