mirror of
https://github.com/EQEmu/Server.git
synced 2026-06-13 02:38:45 +00:00
Compare commits
5 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| 2a91f08845 | |||
| adc64005f1 | |||
| 605480f1c4 | |||
| 3b95601c62 | |||
| a4f2ed28f1 |
@@ -1,3 +1,9 @@
|
|||||||
|
## [22.28.1] - 10/20/2023
|
||||||
|
|
||||||
|
### Build
|
||||||
|
|
||||||
|
* Perl Linux build fix
|
||||||
|
|
||||||
## [22.28.0] - 10/15/2023
|
## [22.28.0] - 10/15/2023
|
||||||
|
|
||||||
### Bots
|
### Bots
|
||||||
|
|||||||
@@ -4956,6 +4956,17 @@ ALTER TABLE `items`
|
|||||||
MODIFY COLUMN `augdistiller` INT(11) UNSIGNED NOT NULL DEFAULT 0,
|
MODIFY COLUMN `augdistiller` INT(11) UNSIGNED NOT NULL DEFAULT 0,
|
||||||
MODIFY COLUMN `scrollunk1` INT(11) UNSIGNED NOT NULL DEFAULT 0,
|
MODIFY COLUMN `scrollunk1` INT(11) UNSIGNED NOT NULL DEFAULT 0,
|
||||||
MODIFY COLUMN `bardeffect` MEDIUMINT(6) NOT NULL DEFAULT 0;
|
MODIFY COLUMN `bardeffect` MEDIUMINT(6) NOT NULL DEFAULT 0;
|
||||||
|
)"
|
||||||
|
},
|
||||||
|
ManifestEntry{
|
||||||
|
.version = 9238,
|
||||||
|
.description = "2023_10_18_tradeskill_add_learned_by_item_id.sql",
|
||||||
|
.check = "SHOW COLUMNS FROM `tradeskill_recipe` LIKE 'learned_by_item_id'",
|
||||||
|
.condition = "empty",
|
||||||
|
.match = "",
|
||||||
|
.sql = R"(
|
||||||
|
ALTER TABLE `tradeskill_recipe`
|
||||||
|
ADD COLUMN `learned_by_item_id` int(11) NOT NULL DEFAULT 0 AFTER `must_learn`;
|
||||||
)"
|
)"
|
||||||
},
|
},
|
||||||
|
|
||||||
|
|||||||
@@ -61,6 +61,29 @@ DROP TABLE IF EXISTS `bot_group_members`;
|
|||||||
SET FOREIGN_KEY_CHECKS = 1;
|
SET FOREIGN_KEY_CHECKS = 1;
|
||||||
)",
|
)",
|
||||||
},
|
},
|
||||||
|
ManifestEntry{
|
||||||
|
.version = 9040,
|
||||||
|
.description = "2023_11_16_bot_starting_items.sql",
|
||||||
|
.check = "SHOW TABLES LIKE 'bot_starting_items'",
|
||||||
|
.condition = "",
|
||||||
|
.match = "empty",
|
||||||
|
.sql = R"(
|
||||||
|
CREATE TABLE `bot_starting_items` (
|
||||||
|
`id` int(11) UNSIGNED NOT NULL AUTO_INCREMENT,
|
||||||
|
`races` int(11) UNSIGNED NOT NULL DEFAULT 0,
|
||||||
|
`classes` int(11) UNSIGNED NOT NULL DEFAULT 0,
|
||||||
|
`item_id` int(11) UNSIGNED NOT NULL DEFAULT 0,
|
||||||
|
`item_charges` tinyint(3) UNSIGNED NOT NULL DEFAULT 1,
|
||||||
|
`min_status` tinyint(3) UNSIGNED NOT NULL DEFAULT 0,
|
||||||
|
`slot_id` mediumint(9) NOT NULL DEFAULT -1,
|
||||||
|
`min_expansion` tinyint(4) NOT NULL DEFAULT -1,
|
||||||
|
`max_expansion` tinyint(4) NOT NULL DEFAULT -1,
|
||||||
|
`content_flags` varchar(100) CHARACTER SET latin1 COLLATE latin1_swedish_ci NULL DEFAULT NULL,
|
||||||
|
`content_flags_disabled` varchar(100) CHARACTER SET latin1 COLLATE latin1_swedish_ci NULL DEFAULT NULL,
|
||||||
|
PRIMARY KEY (`id`)
|
||||||
|
) ENGINE = InnoDB CHARACTER SET = latin1 COLLATE = latin1_swedish_ci;
|
||||||
|
)"
|
||||||
|
}
|
||||||
// -- template; copy/paste this when you need to create a new entry
|
// -- template; copy/paste this when you need to create a new entry
|
||||||
// ManifestEntry{
|
// ManifestEntry{
|
||||||
// .version = 9228,
|
// .version = 9228,
|
||||||
|
|||||||
@@ -0,0 +1,414 @@
|
|||||||
|
/**
|
||||||
|
* DO NOT MODIFY THIS FILE
|
||||||
|
*
|
||||||
|
* This repository was automatically generated and is NOT to be modified directly.
|
||||||
|
* Any repository modifications are meant to be made to the repository extending the base.
|
||||||
|
* Any modifications to base repositories are to be made by the generator only
|
||||||
|
*
|
||||||
|
* @generator ./utils/scripts/generators/repository-generator.pl
|
||||||
|
* @docs https://eqemu.gitbook.io/server/in-development/developer-area/repositories
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef EQEMU_BASE_BOT_STARTING_ITEMS_REPOSITORY_H
|
||||||
|
#define EQEMU_BASE_BOT_STARTING_ITEMS_REPOSITORY_H
|
||||||
|
|
||||||
|
#include "../../database.h"
|
||||||
|
#include "../../strings.h"
|
||||||
|
#include <ctime>
|
||||||
|
|
||||||
|
|
||||||
|
class BaseBotStartingItemsRepository {
|
||||||
|
public:
|
||||||
|
struct BotStartingItems {
|
||||||
|
uint32_t id;
|
||||||
|
uint32_t races;
|
||||||
|
uint32_t classes;
|
||||||
|
uint32_t item_id;
|
||||||
|
uint8_t item_charges;
|
||||||
|
int32_t slot_id;
|
||||||
|
int8_t min_expansion;
|
||||||
|
int8_t max_expansion;
|
||||||
|
std::string content_flags;
|
||||||
|
std::string content_flags_disabled;
|
||||||
|
};
|
||||||
|
|
||||||
|
static std::string PrimaryKey()
|
||||||
|
{
|
||||||
|
return std::string("id");
|
||||||
|
}
|
||||||
|
|
||||||
|
static std::vector<std::string> Columns()
|
||||||
|
{
|
||||||
|
return {
|
||||||
|
"id",
|
||||||
|
"races",
|
||||||
|
"classes",
|
||||||
|
"item_id",
|
||||||
|
"item_charges",
|
||||||
|
"slot_id",
|
||||||
|
"min_expansion",
|
||||||
|
"max_expansion",
|
||||||
|
"content_flags",
|
||||||
|
"content_flags_disabled",
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
static std::vector<std::string> SelectColumns()
|
||||||
|
{
|
||||||
|
return {
|
||||||
|
"id",
|
||||||
|
"races",
|
||||||
|
"classes",
|
||||||
|
"item_id",
|
||||||
|
"item_charges",
|
||||||
|
"slot_id",
|
||||||
|
"min_expansion",
|
||||||
|
"max_expansion",
|
||||||
|
"content_flags",
|
||||||
|
"content_flags_disabled",
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
static std::string ColumnsRaw()
|
||||||
|
{
|
||||||
|
return std::string(Strings::Implode(", ", Columns()));
|
||||||
|
}
|
||||||
|
|
||||||
|
static std::string SelectColumnsRaw()
|
||||||
|
{
|
||||||
|
return std::string(Strings::Implode(", ", SelectColumns()));
|
||||||
|
}
|
||||||
|
|
||||||
|
static std::string TableName()
|
||||||
|
{
|
||||||
|
return std::string("bot_starting_items");
|
||||||
|
}
|
||||||
|
|
||||||
|
static std::string BaseSelect()
|
||||||
|
{
|
||||||
|
return fmt::format(
|
||||||
|
"SELECT {} FROM {}",
|
||||||
|
SelectColumnsRaw(),
|
||||||
|
TableName()
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
static std::string BaseInsert()
|
||||||
|
{
|
||||||
|
return fmt::format(
|
||||||
|
"INSERT INTO {} ({}) ",
|
||||||
|
TableName(),
|
||||||
|
ColumnsRaw()
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
static BotStartingItems NewEntity()
|
||||||
|
{
|
||||||
|
BotStartingItems e{};
|
||||||
|
|
||||||
|
e.id = 0;
|
||||||
|
e.races = 0;
|
||||||
|
e.classes = 0;
|
||||||
|
e.item_id = 0;
|
||||||
|
e.item_charges = 1;
|
||||||
|
e.slot_id = -1;
|
||||||
|
e.min_expansion = -1;
|
||||||
|
e.max_expansion = -1;
|
||||||
|
e.content_flags = "";
|
||||||
|
e.content_flags_disabled = "";
|
||||||
|
|
||||||
|
return e;
|
||||||
|
}
|
||||||
|
|
||||||
|
static BotStartingItems GetBotStartingItems(
|
||||||
|
const std::vector<BotStartingItems> &bot_starting_itemss,
|
||||||
|
int bot_starting_items_id
|
||||||
|
)
|
||||||
|
{
|
||||||
|
for (auto &bot_starting_items : bot_starting_itemss) {
|
||||||
|
if (bot_starting_items.id == bot_starting_items_id) {
|
||||||
|
return bot_starting_items;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return NewEntity();
|
||||||
|
}
|
||||||
|
|
||||||
|
static BotStartingItems FindOne(
|
||||||
|
Database& db,
|
||||||
|
int bot_starting_items_id
|
||||||
|
)
|
||||||
|
{
|
||||||
|
auto results = db.QueryDatabase(
|
||||||
|
fmt::format(
|
||||||
|
"{} WHERE {} = {} LIMIT 1",
|
||||||
|
BaseSelect(),
|
||||||
|
PrimaryKey(),
|
||||||
|
bot_starting_items_id
|
||||||
|
)
|
||||||
|
);
|
||||||
|
|
||||||
|
auto row = results.begin();
|
||||||
|
if (results.RowCount() == 1) {
|
||||||
|
BotStartingItems e{};
|
||||||
|
|
||||||
|
e.id = static_cast<uint32_t>(strtoul(row[0], nullptr, 10));
|
||||||
|
e.races = static_cast<uint32_t>(strtoul(row[1], nullptr, 10));
|
||||||
|
e.classes = static_cast<uint32_t>(strtoul(row[2], nullptr, 10));
|
||||||
|
e.item_id = static_cast<uint32_t>(strtoul(row[3], nullptr, 10));
|
||||||
|
e.item_charges = static_cast<uint8_t>(strtoul(row[4], nullptr, 10));
|
||||||
|
e.slot_id = static_cast<int32_t>(atoi(row[5]));
|
||||||
|
e.min_expansion = static_cast<int8_t>(atoi(row[6]));
|
||||||
|
e.max_expansion = static_cast<int8_t>(atoi(row[7]));
|
||||||
|
e.content_flags = row[8] ? row[8] : "";
|
||||||
|
e.content_flags_disabled = row[9] ? row[9] : "";
|
||||||
|
|
||||||
|
return e;
|
||||||
|
}
|
||||||
|
|
||||||
|
return NewEntity();
|
||||||
|
}
|
||||||
|
|
||||||
|
static int DeleteOne(
|
||||||
|
Database& db,
|
||||||
|
int bot_starting_items_id
|
||||||
|
)
|
||||||
|
{
|
||||||
|
auto results = db.QueryDatabase(
|
||||||
|
fmt::format(
|
||||||
|
"DELETE FROM {} WHERE {} = {}",
|
||||||
|
TableName(),
|
||||||
|
PrimaryKey(),
|
||||||
|
bot_starting_items_id
|
||||||
|
)
|
||||||
|
);
|
||||||
|
|
||||||
|
return (results.Success() ? results.RowsAffected() : 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int UpdateOne(
|
||||||
|
Database& db,
|
||||||
|
const BotStartingItems &e
|
||||||
|
)
|
||||||
|
{
|
||||||
|
std::vector<std::string> v;
|
||||||
|
|
||||||
|
auto columns = Columns();
|
||||||
|
|
||||||
|
v.push_back(columns[1] + " = " + std::to_string(e.races));
|
||||||
|
v.push_back(columns[2] + " = " + std::to_string(e.classes));
|
||||||
|
v.push_back(columns[3] + " = " + std::to_string(e.item_id));
|
||||||
|
v.push_back(columns[4] + " = " + std::to_string(e.item_charges));
|
||||||
|
v.push_back(columns[5] + " = " + std::to_string(e.slot_id));
|
||||||
|
v.push_back(columns[6] + " = " + std::to_string(e.min_expansion));
|
||||||
|
v.push_back(columns[7] + " = " + std::to_string(e.max_expansion));
|
||||||
|
v.push_back(columns[8] + " = '" + Strings::Escape(e.content_flags) + "'");
|
||||||
|
v.push_back(columns[9] + " = '" + Strings::Escape(e.content_flags_disabled) + "'");
|
||||||
|
|
||||||
|
auto results = db.QueryDatabase(
|
||||||
|
fmt::format(
|
||||||
|
"UPDATE {} SET {} WHERE {} = {}",
|
||||||
|
TableName(),
|
||||||
|
Strings::Implode(", ", v),
|
||||||
|
PrimaryKey(),
|
||||||
|
e.id
|
||||||
|
)
|
||||||
|
);
|
||||||
|
|
||||||
|
return (results.Success() ? results.RowsAffected() : 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
static BotStartingItems InsertOne(
|
||||||
|
Database& db,
|
||||||
|
BotStartingItems e
|
||||||
|
)
|
||||||
|
{
|
||||||
|
std::vector<std::string> v;
|
||||||
|
|
||||||
|
v.push_back(std::to_string(e.id));
|
||||||
|
v.push_back(std::to_string(e.races));
|
||||||
|
v.push_back(std::to_string(e.classes));
|
||||||
|
v.push_back(std::to_string(e.item_id));
|
||||||
|
v.push_back(std::to_string(e.item_charges));
|
||||||
|
v.push_back(std::to_string(e.slot_id));
|
||||||
|
v.push_back(std::to_string(e.min_expansion));
|
||||||
|
v.push_back(std::to_string(e.max_expansion));
|
||||||
|
v.push_back("'" + Strings::Escape(e.content_flags) + "'");
|
||||||
|
v.push_back("'" + Strings::Escape(e.content_flags_disabled) + "'");
|
||||||
|
|
||||||
|
auto results = db.QueryDatabase(
|
||||||
|
fmt::format(
|
||||||
|
"{} VALUES ({})",
|
||||||
|
BaseInsert(),
|
||||||
|
Strings::Implode(",", v)
|
||||||
|
)
|
||||||
|
);
|
||||||
|
|
||||||
|
if (results.Success()) {
|
||||||
|
e.id = results.LastInsertedID();
|
||||||
|
return e;
|
||||||
|
}
|
||||||
|
|
||||||
|
e = NewEntity();
|
||||||
|
|
||||||
|
return e;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int InsertMany(
|
||||||
|
Database& db,
|
||||||
|
const std::vector<BotStartingItems> &entries
|
||||||
|
)
|
||||||
|
{
|
||||||
|
std::vector<std::string> insert_chunks;
|
||||||
|
|
||||||
|
for (auto &e: entries) {
|
||||||
|
std::vector<std::string> v;
|
||||||
|
|
||||||
|
v.push_back(std::to_string(e.id));
|
||||||
|
v.push_back(std::to_string(e.races));
|
||||||
|
v.push_back(std::to_string(e.classes));
|
||||||
|
v.push_back(std::to_string(e.item_id));
|
||||||
|
v.push_back(std::to_string(e.item_charges));
|
||||||
|
v.push_back(std::to_string(e.slot_id));
|
||||||
|
v.push_back(std::to_string(e.min_expansion));
|
||||||
|
v.push_back(std::to_string(e.max_expansion));
|
||||||
|
v.push_back("'" + Strings::Escape(e.content_flags) + "'");
|
||||||
|
v.push_back("'" + Strings::Escape(e.content_flags_disabled) + "'");
|
||||||
|
|
||||||
|
insert_chunks.push_back("(" + Strings::Implode(",", v) + ")");
|
||||||
|
}
|
||||||
|
|
||||||
|
std::vector<std::string> v;
|
||||||
|
|
||||||
|
auto results = db.QueryDatabase(
|
||||||
|
fmt::format(
|
||||||
|
"{} VALUES {}",
|
||||||
|
BaseInsert(),
|
||||||
|
Strings::Implode(",", insert_chunks)
|
||||||
|
)
|
||||||
|
);
|
||||||
|
|
||||||
|
return (results.Success() ? results.RowsAffected() : 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
static std::vector<BotStartingItems> All(Database& db)
|
||||||
|
{
|
||||||
|
std::vector<BotStartingItems> all_entries;
|
||||||
|
|
||||||
|
auto results = db.QueryDatabase(
|
||||||
|
fmt::format(
|
||||||
|
"{}",
|
||||||
|
BaseSelect()
|
||||||
|
)
|
||||||
|
);
|
||||||
|
|
||||||
|
all_entries.reserve(results.RowCount());
|
||||||
|
|
||||||
|
for (auto row = results.begin(); row != results.end(); ++row) {
|
||||||
|
BotStartingItems e{};
|
||||||
|
|
||||||
|
e.id = static_cast<uint32_t>(strtoul(row[0], nullptr, 10));
|
||||||
|
e.races = static_cast<uint32_t>(strtoul(row[1], nullptr, 10));
|
||||||
|
e.classes = static_cast<uint32_t>(strtoul(row[2], nullptr, 10));
|
||||||
|
e.item_id = static_cast<uint32_t>(strtoul(row[3], nullptr, 10));
|
||||||
|
e.item_charges = static_cast<uint8_t>(strtoul(row[4], nullptr, 10));
|
||||||
|
e.slot_id = static_cast<int32_t>(atoi(row[5]));
|
||||||
|
e.min_expansion = static_cast<int8_t>(atoi(row[6]));
|
||||||
|
e.max_expansion = static_cast<int8_t>(atoi(row[7]));
|
||||||
|
e.content_flags = row[8] ? row[8] : "";
|
||||||
|
e.content_flags_disabled = row[9] ? row[9] : "";
|
||||||
|
|
||||||
|
all_entries.push_back(e);
|
||||||
|
}
|
||||||
|
|
||||||
|
return all_entries;
|
||||||
|
}
|
||||||
|
|
||||||
|
static std::vector<BotStartingItems> GetWhere(Database& db, const std::string &where_filter)
|
||||||
|
{
|
||||||
|
std::vector<BotStartingItems> all_entries;
|
||||||
|
|
||||||
|
auto results = db.QueryDatabase(
|
||||||
|
fmt::format(
|
||||||
|
"{} WHERE {}",
|
||||||
|
BaseSelect(),
|
||||||
|
where_filter
|
||||||
|
)
|
||||||
|
);
|
||||||
|
|
||||||
|
all_entries.reserve(results.RowCount());
|
||||||
|
|
||||||
|
for (auto row = results.begin(); row != results.end(); ++row) {
|
||||||
|
BotStartingItems e{};
|
||||||
|
|
||||||
|
e.id = static_cast<uint32_t>(strtoul(row[0], nullptr, 10));
|
||||||
|
e.races = static_cast<uint32_t>(strtoul(row[1], nullptr, 10));
|
||||||
|
e.classes = static_cast<uint32_t>(strtoul(row[2], nullptr, 10));
|
||||||
|
e.item_id = static_cast<uint32_t>(strtoul(row[3], nullptr, 10));
|
||||||
|
e.item_charges = static_cast<uint8_t>(strtoul(row[4], nullptr, 10));
|
||||||
|
e.slot_id = static_cast<int32_t>(atoi(row[5]));
|
||||||
|
e.min_expansion = static_cast<int8_t>(atoi(row[6]));
|
||||||
|
e.max_expansion = static_cast<int8_t>(atoi(row[7]));
|
||||||
|
e.content_flags = row[8] ? row[8] : "";
|
||||||
|
e.content_flags_disabled = row[9] ? row[9] : "";
|
||||||
|
|
||||||
|
all_entries.push_back(e);
|
||||||
|
}
|
||||||
|
|
||||||
|
return all_entries;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int DeleteWhere(Database& db, const std::string &where_filter)
|
||||||
|
{
|
||||||
|
auto results = db.QueryDatabase(
|
||||||
|
fmt::format(
|
||||||
|
"DELETE FROM {} WHERE {}",
|
||||||
|
TableName(),
|
||||||
|
where_filter
|
||||||
|
)
|
||||||
|
);
|
||||||
|
|
||||||
|
return (results.Success() ? results.RowsAffected() : 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int Truncate(Database& db)
|
||||||
|
{
|
||||||
|
auto results = db.QueryDatabase(
|
||||||
|
fmt::format(
|
||||||
|
"TRUNCATE TABLE {}",
|
||||||
|
TableName()
|
||||||
|
)
|
||||||
|
);
|
||||||
|
|
||||||
|
return (results.Success() ? results.RowsAffected() : 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int64 GetMaxId(Database& db)
|
||||||
|
{
|
||||||
|
auto results = db.QueryDatabase(
|
||||||
|
fmt::format(
|
||||||
|
"SELECT COALESCE(MAX({}), 0) FROM {}",
|
||||||
|
PrimaryKey(),
|
||||||
|
TableName()
|
||||||
|
)
|
||||||
|
);
|
||||||
|
|
||||||
|
return (results.Success() && results.begin()[0] ? strtoll(results.begin()[0], nullptr, 10) : 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int64 Count(Database& db, const std::string &where_filter = "")
|
||||||
|
{
|
||||||
|
auto results = db.QueryDatabase(
|
||||||
|
fmt::format(
|
||||||
|
"SELECT COUNT(*) FROM {} {}",
|
||||||
|
TableName(),
|
||||||
|
(where_filter.empty() ? "" : "WHERE " + where_filter)
|
||||||
|
)
|
||||||
|
);
|
||||||
|
|
||||||
|
return (results.Success() && results.begin()[0] ? strtoll(results.begin()[0], nullptr, 10) : 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
|
#endif //EQEMU_BASE_BOT_STARTING_ITEMS_REPOSITORY_H
|
||||||
@@ -0,0 +1,50 @@
|
|||||||
|
#ifndef EQEMU_BOT_STARTING_ITEMS_REPOSITORY_H
|
||||||
|
#define EQEMU_BOT_STARTING_ITEMS_REPOSITORY_H
|
||||||
|
|
||||||
|
#include "../database.h"
|
||||||
|
#include "../strings.h"
|
||||||
|
#include "base/base_bot_starting_items_repository.h"
|
||||||
|
|
||||||
|
class BotStartingItemsRepository: public BaseBotStartingItemsRepository {
|
||||||
|
public:
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This file was auto generated and can be modified and extended upon
|
||||||
|
*
|
||||||
|
* Base repository methods are automatically
|
||||||
|
* generated in the "base" version of this repository. The base repository
|
||||||
|
* is immutable and to be left untouched, while methods in this class
|
||||||
|
* are used as extension methods for more specific persistence-layer
|
||||||
|
* accessors or mutators.
|
||||||
|
*
|
||||||
|
* Base Methods (Subject to be expanded upon in time)
|
||||||
|
*
|
||||||
|
* Note: Not all tables are designed appropriately to fit functionality with all base methods
|
||||||
|
*
|
||||||
|
* InsertOne
|
||||||
|
* UpdateOne
|
||||||
|
* DeleteOne
|
||||||
|
* FindOne
|
||||||
|
* GetWhere(std::string where_filter)
|
||||||
|
* DeleteWhere(std::string where_filter)
|
||||||
|
* InsertMany
|
||||||
|
* All
|
||||||
|
*
|
||||||
|
* Example custom methods in a repository
|
||||||
|
*
|
||||||
|
* BotStartingItemsRepository::GetByZoneAndVersion(int zone_id, int zone_version)
|
||||||
|
* BotStartingItemsRepository::GetWhereNeverExpires()
|
||||||
|
* BotStartingItemsRepository::GetWhereXAndY()
|
||||||
|
* BotStartingItemsRepository::DeleteWhereXAndY()
|
||||||
|
*
|
||||||
|
* Most of the above could be covered by base methods, but if you as a developer
|
||||||
|
* find yourself re-using logic for other parts of the code, its best to just make a
|
||||||
|
* method that can be re-used easily elsewhere especially if it can use a base repository
|
||||||
|
* method and encapsulate filters there
|
||||||
|
*/
|
||||||
|
|
||||||
|
// Custom extended repository methods here
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
|
#endif //EQEMU_BOT_STARTING_ITEMS_REPOSITORY_H
|
||||||
@@ -531,6 +531,7 @@ RULE_BOOL(Combat, BackstabIgnoresBane, false, "Enable or disable Bane weapon dam
|
|||||||
RULE_BOOL(Combat, SummonMeleeRange, true, "Enable or disable summoning of a player when already in melee range of the summoner.")
|
RULE_BOOL(Combat, SummonMeleeRange, true, "Enable or disable summoning of a player when already in melee range of the summoner.")
|
||||||
RULE_BOOL(Combat, WaterMatchRequiredForAutoFireLoS, true, "Enable/Disable the requirement of both the attacker/victim being both in or out of water for AutoFire LoS to pass.")
|
RULE_BOOL(Combat, WaterMatchRequiredForAutoFireLoS, true, "Enable/Disable the requirement of both the attacker/victim being both in or out of water for AutoFire LoS to pass.")
|
||||||
RULE_INT(Combat, ExtraAllowedKickClassesBitmask, 0, "Bitmask for allowing extra classes beyond Warrior, Ranger, Beastlord, and Berserker to kick, No Extra Classes (0) by default")
|
RULE_INT(Combat, ExtraAllowedKickClassesBitmask, 0, "Bitmask for allowing extra classes beyond Warrior, Ranger, Beastlord, and Berserker to kick, No Extra Classes (0) by default")
|
||||||
|
RULE_INT(Combat, MaxProcs, 4, "Adjustable maximum number of procs per round, the hard cap is MAX_PROCS (11). Requires mob repop or client zone when changed")
|
||||||
RULE_CATEGORY_END()
|
RULE_CATEGORY_END()
|
||||||
|
|
||||||
RULE_CATEGORY(NPC)
|
RULE_CATEGORY(NPC)
|
||||||
|
|||||||
+2
-2
@@ -42,9 +42,9 @@
|
|||||||
* Manifest: https://github.com/EQEmu/Server/blob/master/utils/sql/db_update_manifest.txt
|
* Manifest: https://github.com/EQEmu/Server/blob/master/utils/sql/db_update_manifest.txt
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#define CURRENT_BINARY_DATABASE_VERSION 9237
|
#define CURRENT_BINARY_DATABASE_VERSION 9238
|
||||||
|
|
||||||
#define CURRENT_BINARY_BOTS_DATABASE_VERSION 9039
|
#define CURRENT_BINARY_BOTS_DATABASE_VERSION 9040
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
|||||||
+1
-1
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "eqemu-server",
|
"name": "eqemu-server",
|
||||||
"version": "22.28.0",
|
"version": "22.28.1",
|
||||||
"repository": {
|
"repository": {
|
||||||
"type": "git",
|
"type": "git",
|
||||||
"url": "https://github.com/EQEmu/Server.git"
|
"url": "https://github.com/EQEmu/Server.git"
|
||||||
|
|||||||
@@ -9,7 +9,7 @@ git submodule init && git submodule update
|
|||||||
|
|
||||||
perl utils/scripts/build/tag-version.pl
|
perl utils/scripts/build/tag-version.pl
|
||||||
|
|
||||||
mkdir -p build && cd build && cmake -DEQEMU_BUILD_TESTS=ON -DEQEMU_BUILD_STATIC=ON -DEQEMU_BUILD_LOGIN=ON -DEQEMU_BUILD_LUA=ON -DCMAKE_CXX_FLAGS_RELWITHDEBINFO:STRING="-Os" -DCMAKE_CXX_COMPILER_LAUNCHER=ccache -G 'Unix Makefiles' .. && make -j$((`nproc`-4))
|
mkdir -p build && cd build && cmake -DEQEMU_BUILD_TESTS=ON -DEQEMU_BUILD_STATIC=ON -DEQEMU_BUILD_LOGIN=ON -DEQEMU_BUILD_LUA=ON -DEQEMU_BUILD_PERL=ON -DCMAKE_CXX_FLAGS_RELWITHDEBINFO:STRING="-Os" -DCMAKE_CXX_COMPILER_LAUNCHER=ccache -G 'Unix Makefiles' .. && make -j$((`nproc`-4))
|
||||||
|
|
||||||
curl https://raw.githubusercontent.com/Akkadius/eqemu-install-v2/master/eqemu_config.json --output eqemu_config.json
|
curl https://raw.githubusercontent.com/Akkadius/eqemu-install-v2/master/eqemu_config.json --output eqemu_config.json
|
||||||
./bin/tests
|
./bin/tests
|
||||||
|
|||||||
+10
-7
@@ -3628,7 +3628,7 @@ int64 Mob::ReduceAllDamage(int64 damage)
|
|||||||
|
|
||||||
bool Mob::HasProcs() const
|
bool Mob::HasProcs() const
|
||||||
{
|
{
|
||||||
for (int i = 0; i < MAX_PROCS; i++) {
|
for (int i = 0; i < m_max_procs; i++) {
|
||||||
if (IsValidSpell(PermaProcs[i].spellID) || IsValidSpell(SpellProcs[i].spellID)) {
|
if (IsValidSpell(PermaProcs[i].spellID) || IsValidSpell(SpellProcs[i].spellID)) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
@@ -3646,7 +3646,7 @@ bool Mob::HasProcs() const
|
|||||||
|
|
||||||
bool Mob::HasDefensiveProcs() const
|
bool Mob::HasDefensiveProcs() const
|
||||||
{
|
{
|
||||||
for (int i = 0; i < MAX_PROCS; i++) {
|
for (int i = 0; i < m_max_procs; i++) {
|
||||||
if (IsValidSpell(DefensiveProcs[i].spellID)) {
|
if (IsValidSpell(DefensiveProcs[i].spellID)) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
@@ -3682,7 +3682,7 @@ bool Mob::HasSkillProcSuccess() const
|
|||||||
|
|
||||||
bool Mob::HasRangedProcs() const
|
bool Mob::HasRangedProcs() const
|
||||||
{
|
{
|
||||||
for (int i = 0; i < MAX_PROCS; i++){
|
for (int i = 0; i < m_max_procs; i++){
|
||||||
if (IsValidSpell(RangedProcs[i].spellID)) {
|
if (IsValidSpell(RangedProcs[i].spellID)) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
@@ -4580,7 +4580,7 @@ void Mob::TryDefensiveProc(Mob *on, uint16 hand)
|
|||||||
}
|
}
|
||||||
|
|
||||||
//Spell Procs and Quest added procs
|
//Spell Procs and Quest added procs
|
||||||
for (int i = 0; i < MAX_PROCS; i++) {
|
for (int i = 0; i < m_max_procs; i++) {
|
||||||
if (IsValidSpell(DefensiveProcs[i].spellID)) {
|
if (IsValidSpell(DefensiveProcs[i].spellID)) {
|
||||||
if (!IsProcLimitTimerActive(DefensiveProcs[i].base_spellID, DefensiveProcs[i].proc_reuse_time, ProcType::DEFENSIVE_PROC)) {
|
if (!IsProcLimitTimerActive(DefensiveProcs[i].base_spellID, DefensiveProcs[i].proc_reuse_time, ProcType::DEFENSIVE_PROC)) {
|
||||||
float chance = proc_chance * (static_cast<float>(DefensiveProcs[i].chance) / 100.0f);
|
float chance = proc_chance * (static_cast<float>(DefensiveProcs[i].chance) / 100.0f);
|
||||||
@@ -4783,7 +4783,7 @@ void Mob::TrySpellProc(const EQ::ItemInstance *inst, const EQ::ItemData *weapon,
|
|||||||
|
|
||||||
int16 poison_slot=-1;
|
int16 poison_slot=-1;
|
||||||
|
|
||||||
for (uint32 i = 0; i < MAX_PROCS; i++) {
|
for (uint32 i = 0; i < m_max_procs; i++) {
|
||||||
if (IsPet() && hand != EQ::invslot::slotPrimary) //Pets can only proc spell procs from their primay hand (ie; beastlord pets)
|
if (IsPet() && hand != EQ::invslot::slotPrimary) //Pets can only proc spell procs from their primay hand (ie; beastlord pets)
|
||||||
continue; // If pets ever can proc from off hand, this will need to change
|
continue; // If pets ever can proc from off hand, this will need to change
|
||||||
|
|
||||||
@@ -5212,8 +5212,11 @@ bool Mob::TryFinishingBlow(Mob *defender, int64 &damage)
|
|||||||
FB_Level = itembonuses.FinishingBlowLvl[SBIndex::FINISHING_EFFECT_LEVEL_MAX];
|
FB_Level = itembonuses.FinishingBlowLvl[SBIndex::FINISHING_EFFECT_LEVEL_MAX];
|
||||||
|
|
||||||
// modern AA description says rank 1 (500) is 50% chance
|
// modern AA description says rank 1 (500) is 50% chance
|
||||||
int ProcChance =
|
int ProcChance = (
|
||||||
aabonuses.FinishingBlow[SBIndex::FINISHING_EFFECT_PROC_CHANCE] + spellbonuses.FinishingBlow[SBIndex::FINISHING_EFFECT_PROC_CHANCE] + spellbonuses.FinishingBlow[SBIndex::FINISHING_EFFECT_PROC_CHANCE];
|
aabonuses.FinishingBlow[SBIndex::FINISHING_EFFECT_PROC_CHANCE] +
|
||||||
|
itembonuses.FinishingBlow[SBIndex::FINISHING_EFFECT_PROC_CHANCE] +
|
||||||
|
spellbonuses.FinishingBlow[SBIndex::FINISHING_EFFECT_PROC_CHANCE]
|
||||||
|
);
|
||||||
|
|
||||||
if (FB_Level && FB_Dmg && (defender->GetLevel() <= FB_Level) &&
|
if (FB_Level && FB_Dmg && (defender->GetLevel() <= FB_Level) &&
|
||||||
(ProcChance >= zone->random.Int(1, 1000))) {
|
(ProcChance >= zone->random.Int(1, 1000))) {
|
||||||
|
|||||||
@@ -22,8 +22,11 @@
|
|||||||
#include "doors.h"
|
#include "doors.h"
|
||||||
#include "quest_parser_collection.h"
|
#include "quest_parser_collection.h"
|
||||||
#include "lua_parser.h"
|
#include "lua_parser.h"
|
||||||
|
#include "../common/repositories/bot_inventories_repository.h"
|
||||||
#include "../common/repositories/bot_spell_settings_repository.h"
|
#include "../common/repositories/bot_spell_settings_repository.h"
|
||||||
|
#include "../common/repositories/bot_starting_items_repository.h"
|
||||||
#include "../common/data_verification.h"
|
#include "../common/data_verification.h"
|
||||||
|
#include "../common/repositories/criteria/content_filter_criteria.h"
|
||||||
|
|
||||||
// This constructor is used during the bot create command
|
// This constructor is used during the bot create command
|
||||||
Bot::Bot(NPCType *npcTypeData, Client* botOwner) : NPC(npcTypeData, nullptr, glm::vec4(), Ground, false), rest_timer(1), ping_timer(1) {
|
Bot::Bot(NPCType *npcTypeData, Client* botOwner) : NPC(npcTypeData, nullptr, glm::vec4(), Ground, false), rest_timer(1), ping_timer(1) {
|
||||||
@@ -8727,4 +8730,48 @@ bool Bot::CheckSpawnConditions(Client* c) {
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void Bot::AddBotStartingItems(uint16 race_id, uint8 class_id)
|
||||||
|
{
|
||||||
|
if (!IsPlayerRace(race_id) || !IsPlayerClass(class_id)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const uint16 race_bitmask = GetPlayerRaceBit(race_id);
|
||||||
|
const uint16 class_bitmask = GetPlayerClassBit(class_id);
|
||||||
|
|
||||||
|
const auto& l = BotStartingItemsRepository::GetWhere(
|
||||||
|
content_db,
|
||||||
|
fmt::format(
|
||||||
|
"(races & {} OR races = 0) AND "
|
||||||
|
"(classes & {} OR classes = 0) {}",
|
||||||
|
race_bitmask,
|
||||||
|
class_bitmask,
|
||||||
|
ContentFilterCriteria::apply()
|
||||||
|
)
|
||||||
|
);
|
||||||
|
|
||||||
|
if (l.empty()) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::vector<BotInventoriesRepository::BotInventories> v;
|
||||||
|
|
||||||
|
for (const auto& e : l) {
|
||||||
|
if (
|
||||||
|
CanClassEquipItem(e.item_id) &&
|
||||||
|
(CanRaceEquipItem(e.item_id) || RuleB(Bots, AllowBotEquipAnyRaceGear))
|
||||||
|
) {
|
||||||
|
auto i = BotInventoriesRepository::NewEntity();
|
||||||
|
i.bot_id = GetBotID();
|
||||||
|
i.slot_id = e.slot_id;
|
||||||
|
i.item_id = e.item_id;
|
||||||
|
i.inst_charges = e.item_charges;
|
||||||
|
|
||||||
|
v.emplace_back(i);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
BotInventoriesRepository::InsertMany(content_db, v);
|
||||||
|
}
|
||||||
|
|
||||||
uint8 Bot::spell_casting_chances[SPELL_TYPE_COUNT][PLAYER_CLASS_COUNT][EQ::constants::STANCE_TYPE_COUNT][cntHSND] = { 0 };
|
uint8 Bot::spell_casting_chances[SPELL_TYPE_COUNT][PLAYER_CLASS_COUNT][EQ::constants::STANCE_TYPE_COUNT][cntHSND] = { 0 };
|
||||||
|
|||||||
@@ -460,6 +460,8 @@ public:
|
|||||||
uint8 gender
|
uint8 gender
|
||||||
);
|
);
|
||||||
|
|
||||||
|
void AddBotStartingItems(uint16 race_id, uint8 class_id);
|
||||||
|
|
||||||
// Static Bot Group Methods
|
// Static Bot Group Methods
|
||||||
static bool AddBotToGroup(Bot* bot, Group* group);
|
static bool AddBotToGroup(Bot* bot, Group* group);
|
||||||
static bool RemoveBotFromGroup(Bot* bot, Group* group);
|
static bool RemoveBotFromGroup(Bot* bot, Group* group);
|
||||||
|
|||||||
@@ -9041,6 +9041,8 @@ uint32 helper_bot_create(Client *bot_owner, std::string bot_name, uint8 bot_clas
|
|||||||
parse->EventPlayer(EVENT_BOT_CREATE, bot_owner, export_string, 0);
|
parse->EventPlayer(EVENT_BOT_CREATE, bot_owner, export_string, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
my_bot->AddBotStartingItems(bot_race, bot_class);
|
||||||
|
|
||||||
safe_delete(my_bot);
|
safe_delete(my_bot);
|
||||||
|
|
||||||
return bot_id;
|
return bot_id;
|
||||||
|
|||||||
+5
-1
@@ -288,8 +288,11 @@ Mob::Mob(
|
|||||||
|
|
||||||
feigned = false;
|
feigned = false;
|
||||||
|
|
||||||
|
int max_procs = MAX_PROCS;
|
||||||
|
m_max_procs = std::min(RuleI(Combat, MaxProcs), max_procs);
|
||||||
|
|
||||||
// clear the proc arrays
|
// clear the proc arrays
|
||||||
for (int j = 0; j < MAX_PROCS; j++) {
|
for (int j = 0; j < m_max_procs; j++) {
|
||||||
PermaProcs[j].spellID = SPELL_UNKNOWN;
|
PermaProcs[j].spellID = SPELL_UNKNOWN;
|
||||||
PermaProcs[j].chance = 0;
|
PermaProcs[j].chance = 0;
|
||||||
PermaProcs[j].base_spellID = SPELL_UNKNOWN;
|
PermaProcs[j].base_spellID = SPELL_UNKNOWN;
|
||||||
@@ -511,6 +514,7 @@ Mob::Mob(
|
|||||||
SetCanOpenDoors(true);
|
SetCanOpenDoors(true);
|
||||||
|
|
||||||
is_boat = IsBoat();
|
is_boat = IsBoat();
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
Mob::~Mob()
|
Mob::~Mob()
|
||||||
|
|||||||
+4
-1
@@ -1459,6 +1459,9 @@ protected:
|
|||||||
|
|
||||||
Timer m_z_clip_check_timer;
|
Timer m_z_clip_check_timer;
|
||||||
|
|
||||||
|
// dynamically set via memory on constructor
|
||||||
|
int8 m_max_procs = 0;
|
||||||
|
|
||||||
virtual bool AI_EngagedCastCheck() { return(false); }
|
virtual bool AI_EngagedCastCheck() { return(false); }
|
||||||
virtual bool AI_PursueCastCheck() { return(false); }
|
virtual bool AI_PursueCastCheck() { return(false); }
|
||||||
virtual bool AI_IdleCastCheck() { return(false); }
|
virtual bool AI_IdleCastCheck() { return(false); }
|
||||||
@@ -1600,7 +1603,7 @@ protected:
|
|||||||
int16 GetSympatheticSpellProcRate(uint16 spell_id);
|
int16 GetSympatheticSpellProcRate(uint16 spell_id);
|
||||||
uint16 GetSympatheticSpellProcID(uint16 spell_id);
|
uint16 GetSympatheticSpellProcID(uint16 spell_id);
|
||||||
|
|
||||||
enum {MAX_PROCS = 4};
|
enum {MAX_PROCS = 10};
|
||||||
tProc PermaProcs[MAX_PROCS];
|
tProc PermaProcs[MAX_PROCS];
|
||||||
tProc SpellProcs[MAX_PROCS];
|
tProc SpellProcs[MAX_PROCS];
|
||||||
tProc DefensiveProcs[MAX_PROCS];
|
tProc DefensiveProcs[MAX_PROCS];
|
||||||
|
|||||||
@@ -2650,6 +2650,8 @@ bool QuestManager::createBot(const char *name, const char *lastname, uint8 level
|
|||||||
).c_str()
|
).c_str()
|
||||||
);
|
);
|
||||||
} else {
|
} else {
|
||||||
|
new_bot->AddBotStartingItems(race, botclass);
|
||||||
|
|
||||||
initiator->Message(
|
initiator->Message(
|
||||||
Chat::White,
|
Chat::White,
|
||||||
fmt::format(
|
fmt::format(
|
||||||
|
|||||||
+9
-9
@@ -6017,7 +6017,7 @@ bool Mob::IsCombatProc(uint16 spell_id) {
|
|||||||
/*
|
/*
|
||||||
Procs that originate from casted spells are still limited by SPA 311 (~Kayen confirmed on live 2/4/22)
|
Procs that originate from casted spells are still limited by SPA 311 (~Kayen confirmed on live 2/4/22)
|
||||||
*/
|
*/
|
||||||
for (int i = 0; i < MAX_PROCS; i++) {
|
for (int i = 0; i < m_max_procs; i++) {
|
||||||
if (PermaProcs[i].spellID == spell_id ||
|
if (PermaProcs[i].spellID == spell_id ||
|
||||||
SpellProcs[i].spellID == spell_id ||
|
SpellProcs[i].spellID == spell_id ||
|
||||||
RangedProcs[i].spellID == spell_id ||
|
RangedProcs[i].spellID == spell_id ||
|
||||||
@@ -6044,7 +6044,7 @@ bool Mob::AddProcToWeapon(uint16 spell_id, bool bPerma, uint16 iChance, uint16 b
|
|||||||
|
|
||||||
int i;
|
int i;
|
||||||
if (bPerma) {
|
if (bPerma) {
|
||||||
for (i = 0; i < MAX_PROCS; i++) {
|
for (i = 0; i < m_max_procs; i++) {
|
||||||
if (!IsValidSpell(PermaProcs[i].spellID)) {
|
if (!IsValidSpell(PermaProcs[i].spellID)) {
|
||||||
PermaProcs[i].spellID = spell_id;
|
PermaProcs[i].spellID = spell_id;
|
||||||
PermaProcs[i].chance = iChance;
|
PermaProcs[i].chance = iChance;
|
||||||
@@ -6059,7 +6059,7 @@ bool Mob::AddProcToWeapon(uint16 spell_id, bool bPerma, uint16 iChance, uint16 b
|
|||||||
} else {
|
} else {
|
||||||
// If its a poison proc, replace any existing one if present.
|
// If its a poison proc, replace any existing one if present.
|
||||||
if (base_spell_id == POISON_PROC) {
|
if (base_spell_id == POISON_PROC) {
|
||||||
for (i = 0; i < MAX_PROCS; i++) {
|
for (i = 0; i < m_max_procs; i++) {
|
||||||
// If we already have a poison proc active replace it and return
|
// If we already have a poison proc active replace it and return
|
||||||
if (SpellProcs[i].base_spellID == POISON_PROC) {
|
if (SpellProcs[i].base_spellID == POISON_PROC) {
|
||||||
SpellProcs[i].spellID = spell_id;
|
SpellProcs[i].spellID = spell_id;
|
||||||
@@ -6076,7 +6076,7 @@ bool Mob::AddProcToWeapon(uint16 spell_id, bool bPerma, uint16 iChance, uint16 b
|
|||||||
// or it is poison and no poison procs are currently present.
|
// or it is poison and no poison procs are currently present.
|
||||||
// Find a slot and use it as normal.
|
// Find a slot and use it as normal.
|
||||||
|
|
||||||
for (i = 0; i < MAX_PROCS; i++) {
|
for (i = 0; i < m_max_procs; i++) {
|
||||||
if (!IsValidSpell(SpellProcs[i].spellID)) {
|
if (!IsValidSpell(SpellProcs[i].spellID)) {
|
||||||
SpellProcs[i].spellID = spell_id;
|
SpellProcs[i].spellID = spell_id;
|
||||||
SpellProcs[i].chance = iChance;
|
SpellProcs[i].chance = iChance;
|
||||||
@@ -6093,7 +6093,7 @@ bool Mob::AddProcToWeapon(uint16 spell_id, bool bPerma, uint16 iChance, uint16 b
|
|||||||
}
|
}
|
||||||
|
|
||||||
bool Mob::RemoveProcFromWeapon(uint16 spell_id, bool bAll) {
|
bool Mob::RemoveProcFromWeapon(uint16 spell_id, bool bAll) {
|
||||||
for (int i = 0; i < MAX_PROCS; i++) {
|
for (int i = 0; i < m_max_procs; i++) {
|
||||||
if (bAll || SpellProcs[i].spellID == spell_id) {
|
if (bAll || SpellProcs[i].spellID == spell_id) {
|
||||||
SpellProcs[i].spellID = SPELL_UNKNOWN;
|
SpellProcs[i].spellID = SPELL_UNKNOWN;
|
||||||
SpellProcs[i].chance = 0;
|
SpellProcs[i].chance = 0;
|
||||||
@@ -6112,7 +6112,7 @@ bool Mob::AddDefensiveProc(uint16 spell_id, uint16 iChance, uint16 base_spell_id
|
|||||||
return(false);
|
return(false);
|
||||||
|
|
||||||
int i;
|
int i;
|
||||||
for (i = 0; i < MAX_PROCS; i++) {
|
for (i = 0; i < m_max_procs; i++) {
|
||||||
if (!IsValidSpell(DefensiveProcs[i].spellID)) {
|
if (!IsValidSpell(DefensiveProcs[i].spellID)) {
|
||||||
DefensiveProcs[i].spellID = spell_id;
|
DefensiveProcs[i].spellID = spell_id;
|
||||||
DefensiveProcs[i].chance = iChance;
|
DefensiveProcs[i].chance = iChance;
|
||||||
@@ -6128,7 +6128,7 @@ bool Mob::AddDefensiveProc(uint16 spell_id, uint16 iChance, uint16 base_spell_id
|
|||||||
|
|
||||||
bool Mob::RemoveDefensiveProc(uint16 spell_id, bool bAll)
|
bool Mob::RemoveDefensiveProc(uint16 spell_id, bool bAll)
|
||||||
{
|
{
|
||||||
for (int i = 0; i < MAX_PROCS; i++) {
|
for (int i = 0; i < m_max_procs; i++) {
|
||||||
if (bAll || DefensiveProcs[i].spellID == spell_id) {
|
if (bAll || DefensiveProcs[i].spellID == spell_id) {
|
||||||
DefensiveProcs[i].spellID = SPELL_UNKNOWN;
|
DefensiveProcs[i].spellID = SPELL_UNKNOWN;
|
||||||
DefensiveProcs[i].chance = 0;
|
DefensiveProcs[i].chance = 0;
|
||||||
@@ -6146,7 +6146,7 @@ bool Mob::AddRangedProc(uint16 spell_id, uint16 iChance, uint16 base_spell_id, u
|
|||||||
return(false);
|
return(false);
|
||||||
|
|
||||||
int i;
|
int i;
|
||||||
for (i = 0; i < MAX_PROCS; i++) {
|
for (i = 0; i < m_max_procs; i++) {
|
||||||
if (!IsValidSpell(RangedProcs[i].spellID)) {
|
if (!IsValidSpell(RangedProcs[i].spellID)) {
|
||||||
RangedProcs[i].spellID = spell_id;
|
RangedProcs[i].spellID = spell_id;
|
||||||
RangedProcs[i].chance = iChance;
|
RangedProcs[i].chance = iChance;
|
||||||
@@ -6162,7 +6162,7 @@ bool Mob::AddRangedProc(uint16 spell_id, uint16 iChance, uint16 base_spell_id, u
|
|||||||
|
|
||||||
bool Mob::RemoveRangedProc(uint16 spell_id, bool bAll)
|
bool Mob::RemoveRangedProc(uint16 spell_id, bool bAll)
|
||||||
{
|
{
|
||||||
for (int i = 0; i < MAX_PROCS; i++) {
|
for (int i = 0; i < m_max_procs; i++) {
|
||||||
if (bAll || RangedProcs[i].spellID == spell_id) {
|
if (bAll || RangedProcs[i].spellID == spell_id) {
|
||||||
RangedProcs[i].spellID = SPELL_UNKNOWN;
|
RangedProcs[i].spellID = SPELL_UNKNOWN;
|
||||||
RangedProcs[i].chance = 0;
|
RangedProcs[i].chance = 0;
|
||||||
|
|||||||
Reference in New Issue
Block a user