mirror of
https://github.com/EQEmu/Server.git
synced 2025-12-11 21:01:29 +00:00
607 lines
18 KiB
C++
607 lines
18 KiB
C++
/* EQEMu: Everquest Server Emulator
|
|
Copyright (C) 2001-2002 EQEMu Development Team (http://eqemu.org)
|
|
|
|
This program is free software; you can redistribute it and/or modify
|
|
it under the terms of the GNU General Public License as published by
|
|
the Free Software Foundation; version 2 of the License.
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY except by those people which sell it, which
|
|
are required to give you total support for your newly bought product;
|
|
without even the implied warranty of MERCHANTABILITY or FITNESS FOR
|
|
A PARTICULAR PURPOSE. See the GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with this program; if not, write to the Free Software
|
|
Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
*/
|
|
#include "../common/debug.h"
|
|
#include <stdio.h>
|
|
#include <iostream>
|
|
using namespace std;
|
|
#include <stdlib.h>
|
|
#include "npc.h"
|
|
#include "masterentity.h"
|
|
#include "zonedb.h"
|
|
#include "../common/MiscFunctions.h"
|
|
#ifdef _WINDOWS
|
|
#define snprintf _snprintf
|
|
#endif
|
|
|
|
#include "../common/EMuShareMem.h"
|
|
extern LoadEMuShareMemDLL EMuShareMemDLL;
|
|
bool SharedDatabase::extDBLoadLoot() {
|
|
return s_usedb->DBLoadLoot();
|
|
}
|
|
|
|
bool SharedDatabase::LoadLoot() {
|
|
char errbuf[MYSQL_ERRMSG_SIZE];
|
|
char *query = 0;
|
|
MYSQL_RES *result;
|
|
MYSQL_ROW row;
|
|
uint32 tmpLootTableCount = 0;
|
|
uint32 tmpLootTableEntriesCount = 0;
|
|
uint32 tmpLootDropCount = 0;
|
|
uint32 tmpLootDropEntriesCount = 0;
|
|
if (RunQuery(query, MakeAnyLenString(&query, "SELECT max(id), count(*) FROM loottable"), errbuf, &result)) {
|
|
safe_delete_array(query);
|
|
if (mysql_num_rows(result) == 1) {
|
|
row = mysql_fetch_row(result);
|
|
if (row[0])
|
|
loottable_max = atoi(row[0]);
|
|
else
|
|
loottable_max = 0;
|
|
tmpLootTableCount = atoi(row[1]);
|
|
}
|
|
else {
|
|
mysql_free_result(result);
|
|
return false;
|
|
}
|
|
mysql_free_result(result);
|
|
}
|
|
else {
|
|
cerr << "Error in LoadLoot query, loottable part: '" << query << "' " << errbuf << endl;
|
|
safe_delete_array(query);
|
|
return false;
|
|
}
|
|
if (RunQuery(query, MakeAnyLenString(&query, "SELECT count(*) FROM loottable_entries"), errbuf, &result)) {
|
|
safe_delete_array(query);
|
|
if (mysql_num_rows(result) == 1) {
|
|
row = mysql_fetch_row(result);
|
|
tmpLootTableEntriesCount = atoi(row[0]);
|
|
}
|
|
else {
|
|
mysql_free_result(result);
|
|
return false;
|
|
}
|
|
mysql_free_result(result);
|
|
}
|
|
else {
|
|
cerr << "Error in LoadLoot query, loottable2 part: '" << query << "' " << errbuf << endl;
|
|
safe_delete_array(query);
|
|
return false;
|
|
}
|
|
|
|
if (RunQuery(query, MakeAnyLenString(&query, "SELECT max(id), count(*) FROM lootdrop"), errbuf, &result)) {
|
|
safe_delete_array(query);
|
|
if (mysql_num_rows(result) == 1) {
|
|
row = mysql_fetch_row(result);
|
|
if (row[0])
|
|
lootdrop_max = atoi(row[0]);
|
|
else
|
|
lootdrop_max = 0;
|
|
tmpLootDropCount = atoi(row[1]);
|
|
}
|
|
else {
|
|
mysql_free_result(result);
|
|
return false;
|
|
}
|
|
mysql_free_result(result);
|
|
}
|
|
else {
|
|
cerr << "Error in LoadLoot query, lootdrop1 part: '" << query << "' " << errbuf << endl;
|
|
safe_delete_array(query);
|
|
return false;
|
|
}
|
|
if (RunQuery(query, MakeAnyLenString(&query, "SELECT max(lootdrop_id), count(*) FROM lootdrop_entries"), errbuf, &result)) {
|
|
safe_delete_array(query);
|
|
if (mysql_num_rows(result) == 1) {
|
|
row = mysql_fetch_row(result);
|
|
tmpLootDropEntriesCount = atoi(row[1]);
|
|
}
|
|
else {
|
|
mysql_free_result(result);
|
|
return false;
|
|
}
|
|
mysql_free_result(result);
|
|
}
|
|
else {
|
|
cerr << "Error in LoadLoot query, lootdrop part: '" << query << "' " << errbuf << endl;
|
|
safe_delete_array(query);
|
|
return false;
|
|
}
|
|
return EMuShareMemDLL.Loot.DLLLoadLoot(&extDBLoadLoot,
|
|
sizeof(LootTable_Struct), tmpLootTableCount, loottable_max,
|
|
sizeof(LootTableEntries_Struct), tmpLootTableEntriesCount,
|
|
sizeof(LootDrop_Struct), tmpLootDropCount, lootdrop_max,
|
|
sizeof(LootDropEntries_Struct), tmpLootDropEntriesCount);
|
|
}
|
|
|
|
bool SharedDatabase::DBLoadLoot() {
|
|
LogFile->write(EQEMuLog::Status, "Loading Loot tables from database...");
|
|
char errbuf[MYSQL_ERRMSG_SIZE];
|
|
char *query = 0;
|
|
MYSQL_RES *result;
|
|
MYSQL_ROW row;
|
|
MYSQL_RES *result2;
|
|
uint32 i, tmpid = 0, tmpmincash = 0, tmpmaxcash = 0, tmpavgcoin = 0;
|
|
if (RunQuery(query, MakeAnyLenString(&query, "SELECT id, mincash, maxcash, avgcoin FROM loottable"), errbuf, &result)) {
|
|
safe_delete_array(query);
|
|
LootTable_Struct* tmpLT = 0;
|
|
while ((row = mysql_fetch_row(result))) {
|
|
tmpid = atoi(row[0]);
|
|
tmpmincash = atoi(row[1]);
|
|
tmpmaxcash = atoi(row[2]);
|
|
tmpavgcoin = atoi(row[3]);
|
|
if (RunQuery(query, MakeAnyLenString(&query, "SELECT loottable_id, lootdrop_id, droplimit, mindrop, multiplier, probability FROM loottable_entries WHERE loottable_id=%i", tmpid), errbuf, &result2)) {
|
|
safe_delete_array(query);
|
|
tmpLT = (LootTable_Struct*) new uchar[sizeof(LootTable_Struct) + (sizeof(LootTableEntries_Struct) * mysql_num_rows(result2))];
|
|
memset(tmpLT, 0, sizeof(LootTable_Struct) + (sizeof(LootTableEntries_Struct) * mysql_num_rows(result2)));
|
|
tmpLT->NumEntries = mysql_num_rows(result2);
|
|
tmpLT->mincash = tmpmincash;
|
|
tmpLT->maxcash = tmpmaxcash;
|
|
tmpLT->avgcoin = tmpavgcoin;
|
|
i=0;
|
|
while ((row = mysql_fetch_row(result2))) {
|
|
if (i >= tmpLT->NumEntries) {
|
|
mysql_free_result(result);
|
|
mysql_free_result(result2);
|
|
safe_delete_array(tmpLT);
|
|
cerr << "Error in ZoneDatabase::DBLoadLoot, i >= NumEntries" << endl;
|
|
return false;
|
|
}
|
|
tmpLT->Entries[i].lootdrop_id = atoi(row[1]);
|
|
tmpLT->Entries[i].droplimit = atoi(row[2]);
|
|
tmpLT->Entries[i].mindrop = atoi(row[3]);
|
|
tmpLT->Entries[i].multiplier = atoi(row[4]);
|
|
tmpLT->Entries[i].probability = atof(row[5]);
|
|
i++;
|
|
}
|
|
if (!EMuShareMemDLL.Loot.cbAddLootTable(tmpid, tmpLT)) {
|
|
mysql_free_result(result);
|
|
mysql_free_result(result2);
|
|
safe_delete_array(tmpLT);
|
|
cout << "Error in ZoneDatabase::DBLoadLoot: !cbAddLootTable(" << tmpid << ")" << endl;
|
|
return false;
|
|
}
|
|
safe_delete_array(tmpLT);
|
|
mysql_free_result(result2);
|
|
}
|
|
else {
|
|
mysql_free_result(result);
|
|
cerr << "Error in LoadLoot (memshare) #1 query '" << query << "' " << errbuf << endl;
|
|
safe_delete_array(query);
|
|
return false;
|
|
}
|
|
}
|
|
mysql_free_result(result);
|
|
}
|
|
else {
|
|
cerr << "Error in LoadLoot (memshare) #2 query '" << query << "' " << errbuf << endl;
|
|
safe_delete_array(query);
|
|
return false;
|
|
}
|
|
|
|
if (RunQuery(query, MakeAnyLenString(&query, "SELECT id FROM lootdrop", tmpid), errbuf, &result)) {
|
|
safe_delete_array(query);
|
|
LootDrop_Struct* tmpLD = 0;
|
|
while ((row = mysql_fetch_row(result))) {
|
|
tmpid = atoi(row[0]);
|
|
if (RunQuery(query, MakeAnyLenString(&query, "SELECT lootdrop_id, item_id, item_charges, equip_item, chance, minlevel, maxlevel, multiplier FROM lootdrop_entries WHERE lootdrop_id=%i order by chance desc", tmpid), errbuf, &result2)) {
|
|
safe_delete_array(query);
|
|
tmpLD = (LootDrop_Struct*) new uchar[sizeof(LootDrop_Struct) + (sizeof(LootDropEntries_Struct) * mysql_num_rows(result2))];
|
|
memset(tmpLD, 0, sizeof(LootDrop_Struct) + (sizeof(LootDropEntries_Struct) * mysql_num_rows(result2)));
|
|
tmpLD->NumEntries = mysql_num_rows(result2);
|
|
i=0;
|
|
while ((row = mysql_fetch_row(result2))) {
|
|
if (i >= tmpLD->NumEntries) {
|
|
mysql_free_result(result);
|
|
mysql_free_result(result2);
|
|
safe_delete_array(tmpLD);
|
|
cerr << "Error in ZoneDatabase::DBLoadLoot, i >= NumEntries" << endl;
|
|
return false;
|
|
}
|
|
tmpLD->Entries[i].item_id = atoi(row[1]);
|
|
tmpLD->Entries[i].item_charges = atoi(row[2]);
|
|
tmpLD->Entries[i].equip_item = atoi(row[3]);
|
|
tmpLD->Entries[i].chance = atof(row[4]);
|
|
tmpLD->Entries[i].minlevel = atoi(row[5]);
|
|
tmpLD->Entries[i].maxlevel = atoi(row[6]);
|
|
tmpLD->Entries[i].multiplier = atoi(row[7]);
|
|
i++;
|
|
}
|
|
if (!EMuShareMemDLL.Loot.cbAddLootDrop(tmpid, tmpLD)) {
|
|
mysql_free_result(result);
|
|
mysql_free_result(result2);
|
|
safe_delete_array(tmpLD);
|
|
cout << "Error in ZoneDatabase::DBLoadLoot: !cbAddLootDrop(" << tmpid << ")" << endl;
|
|
return false;
|
|
}
|
|
safe_delete_array(tmpLD);
|
|
mysql_free_result(result2);
|
|
}
|
|
else {
|
|
cerr << "Error in LoadLoot (memshare) #3 query '" << query << "' " << errbuf << endl;
|
|
mysql_free_result(result);
|
|
safe_delete_array(query);
|
|
return false;
|
|
}
|
|
}
|
|
mysql_free_result(result);
|
|
}
|
|
else {
|
|
cerr << "Error in LoadLoot (memshare) #4 query '" << query << "' " << errbuf << endl;
|
|
safe_delete_array(query);
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
const LootTable_Struct* SharedDatabase::GetLootTable(uint32 loottable_id) {
|
|
return EMuShareMemDLL.Loot.GetLootTable(loottable_id);
|
|
}
|
|
|
|
const LootDrop_Struct* SharedDatabase::GetLootDrop(uint32 lootdrop_id) {
|
|
return EMuShareMemDLL.Loot.GetLootDrop(lootdrop_id);
|
|
}
|
|
|
|
// Queries the loottable: adds item & coin to the npc
|
|
void ZoneDatabase::AddLootTableToNPC(NPC* npc,uint32 loottable_id, ItemList* itemlist, uint32* copper, uint32* silver, uint32* gold, uint32* plat) {
|
|
_ZP(Database_AddLootTableToNPC);
|
|
//if (loottable_id == 178190)
|
|
//DebugBreak();
|
|
const LootTable_Struct* lts = 0;
|
|
*copper = 0;
|
|
*silver = 0;
|
|
*gold = 0;
|
|
*plat = 0;
|
|
|
|
lts = database.GetLootTable(loottable_id);
|
|
if (!lts)
|
|
return;
|
|
|
|
// do coin
|
|
if (lts->mincash > lts->maxcash) {
|
|
cerr << "Error in loottable #" << loottable_id << ": mincash > maxcash" << endl;
|
|
}
|
|
else if (lts->maxcash != 0) {
|
|
uint32 cash = 0;
|
|
if (lts->mincash == lts->maxcash)
|
|
cash = lts->mincash;
|
|
else
|
|
cash = MakeRandomInt(lts->mincash, lts->maxcash);
|
|
if (cash != 0) {
|
|
if (lts->avgcoin != 0) {
|
|
//this is some crazy ass stuff... and makes very little sense... dont use it, k?
|
|
uint32 mincoin = (uint32) (lts->avgcoin * 0.75 + 1);
|
|
uint32 maxcoin = (uint32) (lts->avgcoin * 1.25 + 1);
|
|
*copper = MakeRandomInt(mincoin, maxcoin);
|
|
*silver = MakeRandomInt(mincoin, maxcoin);
|
|
*gold = MakeRandomInt(mincoin, maxcoin);
|
|
if(*copper > cash) { *copper = cash; }
|
|
cash -= *copper;
|
|
if(*silver>(cash/10)) { *silver = (cash/10); }
|
|
cash -= *silver*10;
|
|
if(*gold > (cash/100)) { *gold = (cash/100); }
|
|
cash -= *gold*100;
|
|
}
|
|
if (cash < 0) {
|
|
cash = 0;
|
|
}
|
|
*plat = cash / 1000;
|
|
cash -= *plat * 1000;
|
|
uint32 gold2 = cash / 100;
|
|
cash -= gold2 * 100;
|
|
uint32 silver2 = cash / 10;
|
|
cash -= silver2 * 10;
|
|
*gold += gold2;
|
|
*silver += silver2;
|
|
*copper += cash;
|
|
}
|
|
}
|
|
|
|
// Do items
|
|
for (uint32 i=0; i<lts->NumEntries; i++) {
|
|
for (uint32 k = 1; k <= lts->Entries[i].multiplier; k++) {
|
|
uint8 droplimit = lts->Entries[i].droplimit;
|
|
uint8 mindrop = lts->Entries[i].mindrop;
|
|
|
|
//LootTable Entry probability
|
|
float ltchance = 0.0f;
|
|
ltchance = lts->Entries[i].probability;
|
|
|
|
float drop_chance = 0.0f;
|
|
if(ltchance > 0.0 && ltchance < 100.0) {
|
|
drop_chance = MakeRandomFloat(0.0, 100.0);
|
|
}
|
|
|
|
if (ltchance != 0.0 && (ltchance == 100.0 || drop_chance < ltchance)) {
|
|
AddLootDropToNPC(npc,lts->Entries[i].lootdrop_id, itemlist, droplimit, mindrop);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// Called by AddLootTableToNPC
|
|
// maxdrops = size of the array npcd
|
|
void ZoneDatabase::AddLootDropToNPC(NPC* npc,uint32 lootdrop_id, ItemList* itemlist, uint8 droplimit, uint8 mindrop) {
|
|
const LootDrop_Struct* lds = GetLootDrop(lootdrop_id);
|
|
if (!lds) {
|
|
return;
|
|
}
|
|
if(lds->NumEntries == 0) //nothing possible to add
|
|
return;
|
|
|
|
// Too long a list needs to be limited.
|
|
if(lds->NumEntries > 99 && droplimit < 1)
|
|
droplimit = lds->NumEntries/100;
|
|
|
|
uint8 limit = 0;
|
|
// Start at a random point in itemlist.
|
|
uint32 item = MakeRandomInt(0, lds->NumEntries-1);
|
|
// Main loop.
|
|
for (uint32 i=0; i<lds->NumEntries;)
|
|
{
|
|
//Force the itemlist back to beginning.
|
|
if (item > (lds->NumEntries-1))
|
|
item = 0;
|
|
|
|
uint8 charges = lds->Entries[item].multiplier;
|
|
uint8 pickedcharges = 0;
|
|
// Loop to check multipliers.
|
|
for (uint32 x=1; x<=charges; x++)
|
|
{
|
|
// Actual roll.
|
|
float thischance = 0.0;
|
|
thischance = lds->Entries[item].chance;
|
|
|
|
float drop_chance = 0.0;
|
|
if(thischance != 100.0)
|
|
drop_chance = MakeRandomFloat(0.0, 100.0);
|
|
|
|
#if EQDEBUG>=11
|
|
LogFile->write(EQEMuLog::Debug, "Drop chance for npc: %s, this chance:%f, drop roll:%f", npc->GetName(), thischance, drop_chance);
|
|
#endif
|
|
if (thischance == 100.0 || drop_chance < thischance)
|
|
{
|
|
uint32 itemid = lds->Entries[item].item_id;
|
|
|
|
const Item_Struct* dbitem = GetItem(itemid);
|
|
npc->AddLootDrop(dbitem, itemlist, lds->Entries[item].item_charges, lds->Entries[item].minlevel, lds->Entries[item].maxlevel, lds->Entries[item].equip_item, false);
|
|
pickedcharges++;
|
|
}
|
|
}
|
|
// Items with multipliers only count as 1 towards the limit.
|
|
if(pickedcharges > 0)
|
|
limit++;
|
|
|
|
// If true, limit reached.
|
|
if(limit >= droplimit && droplimit > 0)
|
|
break;
|
|
|
|
item++;
|
|
i++;
|
|
|
|
// We didn't reach our minimium, run loop again.
|
|
if(i == lds->NumEntries){
|
|
if(limit < mindrop){
|
|
i = 0;
|
|
}
|
|
}
|
|
} // We either ran out of items or reached our limit.
|
|
}
|
|
|
|
//if itemlist is null, just send wear changes
|
|
void NPC::AddLootDrop(const Item_Struct *item2, ItemList* itemlist, int16 charges, uint8 minlevel, uint8 maxlevel, bool equipit, bool wearchange) {
|
|
if(item2 == NULL)
|
|
return;
|
|
|
|
//make sure we are doing something...
|
|
if(!itemlist && !wearchange)
|
|
return;
|
|
|
|
ServerLootItem_Struct* item = new ServerLootItem_Struct;
|
|
#if EQDEBUG>=11
|
|
LogFile->write(EQEMuLog::Debug, "Adding drop to npc: %s, Item: %i", GetName(), item2->ID);
|
|
#endif
|
|
|
|
EQApplicationPacket* outapp = NULL;
|
|
WearChange_Struct* wc = NULL;
|
|
if(wearchange) {
|
|
outapp = new EQApplicationPacket(OP_WearChange, sizeof(WearChange_Struct));
|
|
wc = (WearChange_Struct*)outapp->pBuffer;
|
|
wc->spawn_id = GetID();
|
|
wc->material=0;
|
|
}
|
|
|
|
item->item_id = item2->ID;
|
|
item->charges = charges;
|
|
item->aug1 = 0;
|
|
item->aug2 = 0;
|
|
item->aug3 = 0;
|
|
item->aug4 = 0;
|
|
item->aug5 = 0;
|
|
item->minlevel = minlevel;
|
|
item->maxlevel = maxlevel;
|
|
if (equipit) {
|
|
uint8 eslot = 0xFF;
|
|
char newid[20];
|
|
const Item_Struct* compitem = NULL;
|
|
bool found = false; // track if we found an empty slot we fit into
|
|
int32 foundslot = -1; // for multi-slot items
|
|
|
|
// Equip rules are as follows:
|
|
// If the item has the NoPet flag set it will not be equipped.
|
|
// An empty slot takes priority. The first empty one that an item can
|
|
// fit into will be the one picked for the item.
|
|
// AC is the primary choice for which item gets picked for a slot.
|
|
// If AC is identical HP is considered next.
|
|
// If an item can fit into multiple slots we'll pick the last one where
|
|
// it is an improvement.
|
|
|
|
if (!item2->NoPet) {
|
|
for (int i=0; !found && i<MAX_WORN_INVENTORY; i++) {
|
|
uint32 slots = (1 << i);
|
|
if (item2->Slots & slots) {
|
|
if(equipment[i])
|
|
{
|
|
compitem = database.GetItem(equipment[i]);
|
|
if (item2->AC > compitem->AC ||
|
|
(item2->AC == compitem->AC && item2->HP > compitem->HP))
|
|
{
|
|
// item would be an upgrade
|
|
// check if we're multi-slot, if yes then we have to keep
|
|
// looking in case any of the other slots we can fit into are empty.
|
|
if (item2->Slots != slots) {
|
|
foundslot = i;
|
|
}
|
|
else {
|
|
equipment[i] = item2->ID;
|
|
foundslot = i;
|
|
found = true;
|
|
}
|
|
} // end if ac
|
|
}
|
|
else
|
|
{
|
|
equipment[i] = item2->ID;
|
|
foundslot = i;
|
|
found = true;
|
|
}
|
|
} // end if (slots)
|
|
} // end for
|
|
} // end if NoPet
|
|
|
|
// Possible slot was found but not selected. Pick it now.
|
|
if (!found && foundslot >= 0) {
|
|
equipment[foundslot] = item2->ID;
|
|
found = true;
|
|
}
|
|
|
|
// @merth: IDFile size has been increased, this needs to change
|
|
uint16 emat;
|
|
if(item2->Material <= 0
|
|
|| item2->Slots & (1 << SLOT_PRIMARY | 1 << SLOT_SECONDARY)) {
|
|
memset(newid, 0, sizeof(newid));
|
|
for(int i=0;i<7;i++){
|
|
if (!isalpha(item2->IDFile[i])){
|
|
strn0cpy(newid, &item2->IDFile[i],6);
|
|
i=8;
|
|
}
|
|
}
|
|
|
|
emat = atoi(newid);
|
|
} else {
|
|
emat = item2->Material;
|
|
}
|
|
|
|
if (foundslot == SLOT_PRIMARY) {
|
|
if (item2->Proc.Effect != 0)
|
|
CastToMob()->AddProcToWeapon(item2->Proc.Effect, true);
|
|
|
|
eslot = MATERIAL_PRIMARY;
|
|
}
|
|
else if (foundslot == SLOT_SECONDARY
|
|
&& (GetOwner() != NULL || (GetLevel() >= 13 && MakeRandomInt(0,99) < NPC_DW_CHANCE) || (item2->Damage==0)) &&
|
|
(item2->ItemType == ItemType1HS || item2->ItemType == ItemType1HB || item2->ItemType == ItemTypeShield ||
|
|
item2->ItemType == ItemTypePierce))
|
|
{
|
|
if (item2->Proc.Effect!=0)
|
|
CastToMob()->AddProcToWeapon(item2->Proc.Effect, true);
|
|
|
|
eslot = MATERIAL_SECONDARY;
|
|
}
|
|
else if (foundslot == SLOT_HEAD) {
|
|
eslot = MATERIAL_HEAD;
|
|
}
|
|
else if (foundslot == SLOT_CHEST) {
|
|
eslot = MATERIAL_CHEST;
|
|
}
|
|
else if (foundslot == SLOT_ARMS) {
|
|
eslot = MATERIAL_ARMS;
|
|
}
|
|
else if (foundslot == SLOT_BRACER01 || foundslot == SLOT_BRACER02) {
|
|
eslot = MATERIAL_BRACER;
|
|
}
|
|
else if (foundslot == SLOT_HANDS) {
|
|
eslot = MATERIAL_HANDS;
|
|
}
|
|
else if (foundslot == SLOT_LEGS) {
|
|
eslot = MATERIAL_LEGS;
|
|
}
|
|
else if (foundslot == SLOT_FEET) {
|
|
eslot = MATERIAL_FEET;
|
|
}
|
|
|
|
/*
|
|
what was this about???
|
|
|
|
if (((npc->GetRace()==127) && (npc->CastToMob()->GetOwnerID()!=0)) && (item2->Slots==24576) || (item2->Slots==8192) || (item2->Slots==16384)){
|
|
npc->d_meele_texture2=atoi(newid);
|
|
wc->wear_slot_id=8;
|
|
if (item2->Material >0)
|
|
wc->material=item2->Material;
|
|
else
|
|
wc->material=atoi(newid);
|
|
npc->AC+=item2->AC;
|
|
npc->STR+=item2->STR;
|
|
npc->INT+=item2->INT;
|
|
}
|
|
*/
|
|
|
|
//if we found an open slot it goes in...
|
|
if(eslot != 0xFF) {
|
|
if(wearchange) {
|
|
wc->wear_slot_id = eslot;
|
|
wc->material = emat;
|
|
}
|
|
|
|
}
|
|
if (found) {
|
|
CalcBonuses(); // This is less than ideal for bulk adding of items
|
|
}
|
|
item->equipSlot = item2->Slots;
|
|
}
|
|
|
|
if(itemlist != NULL)
|
|
itemlist->push_back(item);
|
|
else
|
|
safe_delete(item);
|
|
|
|
if(wearchange && outapp) {
|
|
entity_list.QueueClients(this, outapp);
|
|
safe_delete(outapp);
|
|
}
|
|
}
|
|
|
|
void NPC::AddItem(const Item_Struct* item, uint16 charges, bool equipitem) {
|
|
//slot isnt needed, its determined from the item.
|
|
AddLootDrop(item, &itemlist, charges, 1, 127, equipitem, equipitem);
|
|
}
|
|
|
|
void NPC::AddItem(uint32 itemid, uint16 charges, bool equipitem) {
|
|
//slot isnt needed, its determined from the item.
|
|
const Item_Struct * i = database.GetItem(itemid);
|
|
if(i == NULL)
|
|
return;
|
|
AddLootDrop(i, &itemlist, charges, 1, 127, equipitem, equipitem);
|
|
}
|
|
|
|
void NPC::AddLootTable() {
|
|
if (npctype_id != 0) { // check if it's a GM spawn
|
|
database.AddLootTableToNPC(this,loottable_id, &itemlist, &copper, &silver, &gold, &platinum);
|
|
}
|
|
}
|
|
|