/*
EQ2Emulator: Everquest II Server Emulator
Copyright (C) 2007 EQ2EMulator Development Team (http://www.eq2emulator.net)
This file is part of EQ2Emulator.
EQ2Emulator 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, either version 3 of the License, or
(at your option) any later version.
EQ2Emulator is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; 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 EQ2Emulator. If not, see .
*/
#include
#include "../../common/debug.h"
#include "../../common/Log.h"
#include "../../common/database.h"
#include "Recipe.h"
#include "../../common/ConfigReader.h"
#include "../Items/Items.h"
extern ConfigReader configReader;
extern MasterItemList master_item_list;
Recipe::Recipe() {
id = 0;
book_id = 0;
memset(name, 0, sizeof(name));
memset(book_name, 0, sizeof(book_name));
memset(book, 0, sizeof(book));
memset(device, 0, sizeof(device));
level = 0;
tier = 0;
icon = 0;
skill = 0;
technique = 0;
knowledge = 0;
classes = 0;
unknown2 = 0;
unknown3 = 0;
unknown4 = 0;
}
Recipe::~Recipe() {
map::iterator itr;
for (itr = products.begin(); itr != products.end(); itr++)
safe_delete(itr->second);
}
Recipe::Recipe(Recipe *in){
assert(in);
id = in->GetID();
book_id = in->GetBookID();
strncpy(name, in->GetName(), sizeof(name));
strncpy(book_name, in->GetBookName(), sizeof(book_name));
strncpy(book, in->GetBook(), sizeof(book));
strncpy(device, in->GetDevice(), sizeof(device));
level = in->GetLevel();
tier = in->GetTier();
icon = in->GetIcon();
skill = in->GetSkill();
technique = in->GetTechnique();
knowledge = in->GetKnowledge();
classes = in->GetClasses();
unknown2 = in->GetUnknown2();
unknown3 = in->GetUnknown3();
unknown4 = in->GetUnknown4();
}
MasterRecipeList::MasterRecipeList() {
m_recipes.SetName("MasterRecipeList::recipes");
}
MasterRecipeList::~MasterRecipeList() {
ClearRecipes();
}
bool MasterRecipeList::AddRecipe(Recipe *recipe) {
bool ret = false;
int32 id;
assert(recipe);
id = recipe->GetID();
m_recipes.writelock(__FUNCTION__, __LINE__);
if (recipes.count(id) == 0) {
recipes[id] = recipe;
ret = true;
}
m_recipes.releasewritelock(__FUNCTION__, __LINE__);
return ret;
}
Recipe * MasterRecipeList::GetRecipe(int32 recipe_id) {
Recipe *ret = 0;
m_recipes.readlock(__FUNCTION__, __LINE__);
if (recipes.count(recipe_id) > 0)
ret = recipes[recipe_id];
m_recipes.releasereadlock(__FUNCTION__, __LINE__);
return ret;
}
Recipe* MasterRecipeList::GetRecipeByName(const char* name) {
Recipe* ret = 0;
map::iterator itr;
m_recipes.readlock(__FUNCTION__, __LINE__);
for (itr = recipes.begin(); itr != recipes.end(); itr++) {
if (::ToLower(string(name)) == ::ToLower(string(itr->second->GetName()))) {
ret = itr->second;
break;
}
}
m_recipes.releasereadlock(__FUNCTION__, __LINE__);
return ret;
}
void MasterRecipeList::ClearRecipes() {
map::iterator itr;
m_recipes.writelock(__FUNCTION__, __LINE__);
for (itr = recipes.begin(); itr != recipes.end(); itr++)
safe_delete(itr->second);
recipes.clear();
m_recipes.releasewritelock(__FUNCTION__, __LINE__);
}
int32 MasterRecipeList::Size() {
int32 ret;
m_recipes.readlock(__FUNCTION__, __LINE__);
ret = (int32)recipes.size();
m_recipes.releasereadlock(__FUNCTION__, __LINE__);
return ret;
}
vector* MasterRecipeList::GetRecipes(const char* book_name) {
vector* ret = new vector;
map::iterator itr;
m_recipes.writelock(__FUNCTION__, __LINE__);
for (itr = recipes.begin(); itr != recipes.end(); itr++) {
if (::ToLower(string(book_name)) == ::ToLower(string(itr->second->GetBook())))
ret->push_back(itr->second);
}
m_recipes.releasewritelock(__FUNCTION__, __LINE__);
return ret;
}
PlayerRecipeList::PlayerRecipeList(){
}
PlayerRecipeList::~PlayerRecipeList(){
ClearRecipes();
}
bool PlayerRecipeList::AddRecipe(Recipe *recipe){
assert(recipe);
if(recipes.count(recipe->GetID()) == 0){
recipes[recipe->GetID()] = recipe;
return true;
}
return false;
}
Recipe * PlayerRecipeList::GetRecipe(int32 recipe_id){
if (recipes.count(recipe_id) > 0)
return recipes[recipe_id];
return 0;
}
void PlayerRecipeList::ClearRecipes(){
map::iterator itr;
for (itr = recipes.begin(); itr != recipes.end(); itr++)
safe_delete(itr->second);
recipes.clear();
}
MasterRecipeBookList::MasterRecipeBookList(){
m_recipeBooks.SetName("MasterRecipeBookList::recipeBooks");
}
MasterRecipeBookList::~MasterRecipeBookList(){
ClearRecipeBooks();
}
bool MasterRecipeBookList::AddRecipeBook(Recipe *recipe){
bool ret = false;
int32 id = 0;
assert(recipe);
id = recipe->GetBookID();
m_recipeBooks.writelock(__FUNCTION__, __LINE__);
if(recipeBooks.count(id) == 0){
recipeBooks[id] = recipe;
ret = true;
}
m_recipeBooks.releasewritelock(__FUNCTION__, __LINE__);
return ret;
}
Recipe * MasterRecipeBookList::GetRecipeBooks(int32 recipebook_id){
Recipe *ret = 0;
m_recipeBooks.readlock(__FUNCTION__, __LINE__);
if (recipeBooks.count(recipebook_id) > 0)
ret = recipeBooks[recipebook_id];
m_recipeBooks.releasereadlock(__FUNCTION__, __LINE__);
return ret;
}
void MasterRecipeBookList::ClearRecipeBooks(){
map::iterator itr;
m_recipeBooks.writelock(__FUNCTION__, __LINE__);
for (itr = recipeBooks.begin(); itr != recipeBooks.end(); itr++)
safe_delete(itr->second);
recipeBooks.clear();
m_recipeBooks.releasewritelock(__FUNCTION__, __LINE__);
}
int32 MasterRecipeBookList::Size(){
int32 ret = 0;
m_recipeBooks.readlock(__FUNCTION__, __LINE__);
ret = (int32)recipeBooks.size();
m_recipeBooks.releasereadlock(__FUNCTION__, __LINE__);
return ret;
}
EQ2Packet* MasterRecipeList::GetRecipePacket(int32 recipe_id, Client* client, bool display, int8 packet_type){
Recipe *recipe = GetRecipe(recipe_id);
if(recipe){
LogWrite(TRADESKILL__DEBUG, 5, "Recipes", "Recipe ID: %u Recipe Name: %s", recipe->GetID(), recipe->GetName());
return recipe->SerializeRecipe(client, recipe, display, packet_type);
}
return 0;
}
PlayerRecipeBookList::PlayerRecipeBookList(){
}
PlayerRecipeBookList::~PlayerRecipeBookList(){
ClearRecipeBooks();
}
bool PlayerRecipeBookList::AddRecipeBook(Recipe *recipe){
assert(recipe);
if(recipeBooks.count(recipe->GetBookID()) == 0){
recipeBooks[recipe->GetBookID()] = recipe;
return true;
}
return false;
}
Recipe * PlayerRecipeBookList::GetRecipeBook(int32 recipebook_id){
if(recipeBooks.count(recipebook_id) > 0)
return recipeBooks[recipebook_id];
return 0;
}
bool PlayerRecipeBookList::HasRecipeBook(int32 book_id) {
if (recipeBooks.count(book_id) > 0)
return true;
return false;
}
void PlayerRecipeBookList::ClearRecipeBooks(){
map::iterator itr;
for(itr = recipeBooks.begin(); itr != recipeBooks.end(); itr++)
safe_delete(itr->second);
recipeBooks.clear();
}
EQ2Packet * Recipe::SerializeRecipe(Client *client, Recipe *recipe, bool display, int8 packet_type, int8 subpacket_type, const char *struct_name){
int16 version = 1;
Item* item = 0;
RecipeProducts* rp = 0;
vector::iterator itr;
int8 i = 0;
int32 firstID = 0;
int32 primary_comp_id = 0;
if(client)
version = client->GetVersion();
if(!struct_name)
struct_name = "WS_ExamineRecipeInfo";
PacketStruct *packet = configReader.getStruct(struct_name, version);
if(display)
packet->setSubstructDataByName("info_header", "show_name", 1);
else
packet->setSubstructDataByName("info_header", "show_popup", 1);
if(packet_type > 0)
packet->setSubstructDataByName("info_header", "packettype", packet_type*256 + 0xFE);
else
if(version == 1096)
packet->setSubstructDataByName("info_header", "packettype",0x35FE);
if (version == 1208)
packet->setSubstructDataByName("info_header", "packettype", 0x45FE);
if(version >= 57048)
packet->setSubstructDataByName("info_header", "packettype",0x48FE);
if(subpacket_type == 0)
subpacket_type = 0x02;
packet->setSubstructDataByName("info_header", "packetsubtype", subpacket_type);
packet->setSubstructDataByName("recipe_info", "id", recipe->GetID());
packet->setSubstructDataByName("recipe_info", "unknown", 3);
packet->setSubstructDataByName("recipe_info", "level", recipe->GetLevel());
packet->setSubstructDataByName("recipe_info", "technique", recipe->GetTechnique());
packet->setSubstructDataByName("recipe_info", "skill_level", 50); //50
packet->setSubstructDataByName("recipe_info", "knowledge", recipe->GetKnowledge());
packet->setSubstructDataByName("recipe_info", "device", recipe->GetDevice());
packet->setSubstructDataByName("recipe_info", "icon", recipe->GetIcon());
packet->setSubstructDataByName("recipe_info", "unknown3", 1);
packet->setSubstructDataByName("recipe_info", "adventure_id", 0xFF);
packet->setSubstructDataByName("recipe_info", "tradeskill_id", client ? client->GetPlayer()->GetTradeskillClass() : 0);
packet->setSubstructDataByName("recipe_info", "unknown4", 100);
packet->setSubstructDataByName("recipe_info", "unknown4aa", 1);
packet->setSubstructDataByName("recipe_info", "unknown5a", 20);
packet->setSubstructDataByName("recipe_info", "product_classes", recipe->GetClasses());
packet->setSubstructDataByName("recipe_info", "show_previous", 15);
packet->setSubstructDataByName("recipe_info", "previous1_icon", 186);
packet->setSubstructDataByName("recipe_info", "previous1_name", "previous1_name");
packet->setSubstructDataByName("recipe_info", "previous1_qty", 1);
packet->setSubstructDataByName("recipe_info", "previous1_item_id", 4142);
packet->setSubstructDataByName("recipe_info", "previous1_item_crc", -853046774);
packet->setSubstructDataByName("recipe_info", "previous2_icon", 186);
packet->setSubstructDataByName("recipe_info", "previous2_name", "previous2_name");
packet->setSubstructDataByName("recipe_info", "previous2_qty", 1);
packet->setSubstructDataByName("recipe_info", "previous2_item_id", 4142);
packet->setSubstructDataByName("recipe_info", "previous2_item_crc", -853046774);
packet->setSubstructDataByName("recipe_info", "previous3_icon", 186);
packet->setSubstructDataByName("recipe_info", "previous3_name", "previous3_name");
packet->setSubstructDataByName("recipe_info", "previous3_qty", 1);
packet->setSubstructDataByName("recipe_info", "previous3_item_id", 4142);
packet->setSubstructDataByName("recipe_info", "previous3_item_crc", -853046774);
packet->setSubstructDataByName("recipe_info", "firstbar_icon", 186);
packet->setSubstructDataByName("recipe_info", "firstbar_name", "firstbar_name");
packet->setSubstructDataByName("recipe_info", "firstbar_qty", 1);
packet->setSubstructDataByName("recipe_info", "firstbar_item_id", 4142);
packet->setSubstructDataByName("recipe_info", "firstbar_item_crc", -853046774);
packet->setSubstructDataByName("recipe_info", "secondbar_icon", 186);
packet->setSubstructDataByName("recipe_info", "secondbar_name", "secondbar_name");
packet->setSubstructDataByName("recipe_info", "secondbar_qty", 1);
packet->setSubstructDataByName("recipe_info", "secondbar_item_id", 4142);
packet->setSubstructDataByName("recipe_info", "secondbar_item_crc", -853046774);
packet->setSubstructDataByName("recipe_info", "thirdbar_icon", 198);
packet->setSubstructDataByName("recipe_info", "thirdbar_name", "thirdbar_name");
packet->setSubstructDataByName("recipe_info", "thirdbar_qty", 1);
packet->setSubstructDataByName("recipe_info", "thirdbar_item_id", 12098);
packet->setSubstructDataByName("recipe_info", "thirdbar_item_crc", -2065846136);
item = master_item_list.GetItemByName(recipe->GetName());
if(item) {
packet->setSubstructDataByName("recipe_info", "product_icon", item->details.icon); //item->details.icon);
packet->setSubstructDataByName("recipe_info", "product_name", item->name.c_str()); //item->name);
packet->setSubstructDataByName("recipe_info", "product_qty", 1);
packet->setSubstructDataByName("recipe_info", "product_item_id", item->details.item_id); //item->details.item_id);
packet->setSubstructDataByName("recipe_info", "product_item_crc", 0); //item->details.item_crc);
}
rp = recipe->products[0];
if (rp->byproduct_id > 0) {
item = 0;
item = master_item_list.GetItem(rp->byproduct_id);
if (item) {
packet->setSubstructDataByName("recipe_info", "byproduct_icon", item->details.icon);//11
packet->setSubstructDataByName("recipe_info", "byproduct_id", item->details.item_id);
}
}
//string xxx = recipe->
//item = master_item_list.GetItemByName (recipe->GetBuild1ComponentTitle());
// Reset item to 0
item = 0;
// Check to see if we have a primary component (slot = 0)
if (recipe->components.count(0) > 0) {
vector rc = recipe->components[0];
for (itr = rc.begin(); itr != rc.end(); itr++, i++) {
if (firstID == 0)
firstID = *itr;
item = master_item_list.GetItem(*itr);
item = 0;
item = client->GetPlayer()->item_list.GetItemFromID((*itr));
if (item) {
packet->setSubstructDataByName("recipe_info", "primary_qty_avail", item->details.count);
packet->setSubstructDataByName("recipe_info", "primary_comp", recipe->primary_build_comp_title);
}
}
// store the id of the primary comp
primary_comp_id = firstID;
// Set the default item id to the first component id
item = 0;
item = client->GetPlayer()->item_list.GetItemFromID(firstID);
if (item) {
}
// Reset the variables we will reuse
i = 0;
firstID = 0;
item = 0;
}
else {
LogWrite(TRADESKILL__ERROR, 0, "Recipes", "Recipe has no primary component");
}
int8 total_build_components = 0;
if (recipe->components.count(1) > 0)
total_build_components++;
if (recipe->components.count(2))
total_build_components++;
if (recipe->components.count(3))
total_build_components++;
if (recipe->components.count(4))
total_build_components++;
if (total_build_components > 0) {
packet->setSubstructArrayLengthByName("recipe_info", "num_comps", total_build_components);
for (int8 index = 0; index < 4; index++) {
if (recipe->components.count(index + 1) == 0)
continue;
//packet->setArrayDataByName("build_slot", index, index);
vector rc = recipe->components[index + 1];
//packet->setSubArrayLengthByName("num_build_choices", rc.size(), index);
for (itr = rc.begin(); itr != rc.end(); itr++, i++) {
if (firstID == 0)
firstID = *itr;
string comp_title;
int8 comp_qty;
if (index == 0) {
comp_title = recipe->build1_comp_title;
comp_qty = recipe->build_comp_qty;
item = master_item_list.GetItem(*itr);
packet->setArrayDataByName("build_comp", comp_title.c_str(), index );
packet->setArrayDataByName("build_comp_qty", comp_qty, index );
item = master_item_list.GetItem(*itr);
item = 0;
item = client->GetPlayer()->item_list.GetItemFromID((*itr));
if (item) {
packet->setArrayDataByName("build_comp_qty_avail", item->details.count, index );
}
}
else if (index == 1) {
comp_title = recipe->build2_comp_title;
comp_qty = recipe->build2_comp_qty;
item = master_item_list.GetItem(*itr);
packet->setArrayDataByName("build_comp", comp_title.c_str(), index );
packet->setArrayDataByName("build_comp_qty", comp_qty, index );
item = master_item_list.GetItem(*itr);
item = 0;
item = client->GetPlayer()->item_list.GetItemFromID((*itr));
if (item) {
packet->setArrayDataByName("build_comp_qty_avail", item->details.count, index );
}
}
else if (index == 2) {
comp_title = recipe->build3_comp_title;
comp_qty = recipe->build3_comp_qty;
item = master_item_list.GetItem(*itr);
packet->setArrayDataByName("build_comp", comp_title.c_str(), index );
packet->setArrayDataByName("build_comp_qty", comp_qty, index );
item = master_item_list.GetItem(*itr);
item = 0;
item = client->GetPlayer()->item_list.GetItemFromID((*itr));
if (item) {
packet->setArrayDataByName("build_comp_qty_avail", item->details.count, index );
}
}
else if (index == 3) {
comp_title = recipe->build4_comp_title;
comp_qty = recipe->build4_comp_qty;
item = master_item_list.GetItem(*itr);
packet->setArrayDataByName("build_comp", comp_title.c_str(), index );
packet->setArrayDataByName("build_comp_qty", comp_qty, index );
item = master_item_list.GetItem(*itr);
item = 0;
item = client->GetPlayer()->item_list.GetItemFromID((*itr));
if (item) {
packet->setArrayDataByName("build_comp_qty_avail", item->details.count, index );
}
}
//item = master_item_list.GetItem(*itr);
//packet->setArrayDataByName("build_comp", comp_title.c_str(), index -1);
//packet->setArrayDataByName("build_comp_qty", comp_qty, index - 1);
//item = master_item_list.GetItem(*itr);
//item = 0;
//item = client->GetPlayer()->item_list.GetItemFromID((*itr));
//if (item) {
// packet->setArrayDataByName("build_comp_qty_avail", item->details.count, index - 1);
//}
item = 0;
item = client->GetPlayer()->item_list.GetItemFromID((*itr));
if (item) {
//packet->setSubArrayDataByName("build_total_quantity", item->details.count, index, i);
}
}
// Set the default item id to the first component id
//packet->setArrayDataByName("build_item_selected", 1, index);
//packet->setArrayDataByName("build_selected_item_id", firstID, index);
item = 0;
item = client->GetPlayer()->item_list.GetItemFromID(firstID);
int8 qty = 0;
if (item) {
qty = (int8)item->details.count;
if (qty > 0 && firstID == primary_comp_id)
qty -= 1;
}
if (index == 0) {
// packet->setArrayDataByName("build_title", recipe->GetBuild1ComponentTitle(), index);
//packet->setArrayDataByName("build_qty", recipe->GetBuild1ComponentQuantity(), index);
if (item) {
// packet->setArrayDataByName("build_selected_item_qty", min(qty, recipe->GetBuild1ComponentQuantity()), index);
}
}
else if (index == 1) {
// packet->setArrayDataByName("build_title", recipe->GetBuild2ComponentTitle(), index);
// packet->setArrayDataByName("build_qty", recipe->GetBuild2ComponentQuantity(), index);
if (item) {
// packet->setArrayDataByName("build_selected_item_qty", min(qty, recipe->GetBuild2ComponentQuantity()), index);
}
}
else if (index == 2) {
// packet->setArrayDataByName("build_title", recipe->GetBuild3ComponentTitle(), index);
// packet->setArrayDataByName("build_qty", recipe->GetBuild3ComponentQuantity(), index);
if (item) {
// packet->setArrayDataByName("build_selected_item_qty", min(qty, recipe->GetBuild3ComponentQuantity()), index);
}
}
else {
// packet->setArrayDataByName("build_title", recipe->GetBuild4ComponentTitle(), index);
// packet->setArrayDataByName("build_qty", recipe->GetBuild4ComponentQuantity(), index);
if (item) {
//packet->setArrayDataByName("build_selected_item_qty", min(qty, recipe->GetBuild4ComponentQuantity()), index);
}
}
// Reset the variables we will reuse
i = 0;
firstID = 0;
item = 0;
}
}
//packet->setArrayDataByName("build_comp", "build_comp1", 0);
//packet->setArrayDataByName("build_comp_qty", 1, 0);
//packet->setArrayDataByName("build_comp_qty_avail", 24, 0);
//packet->setArrayDataByName("build_comp", "build_comp2", 1);
//packet->setArrayDataByName("build_comp_qty", 1, 1);
//packet->setArrayDataByName("build_comp_qty_avail", 16, 1);
//packet->setArrayDataByName("build_comp", "build_comp3", 2);
//packet->setArrayDataByName("build_comp_qty", 1, 2);
//packet->setArrayDataByName("build_comp_qty_avail", 15, 2);
// Check to see if we have a fuel component (slot = 5)
if (recipe->components.count(5) > 0) {
vector rc = recipe->components[5];
//packet->setArrayLengthByName("num_fuel_choices", rc.size());
for (itr = rc.begin(); itr != rc.end(); itr++, i++) {
if (firstID == 0)
firstID = *itr;
item = master_item_list.GetItem(*itr);
//packet->setSubstructDataByName("recipe_info", "fuel_comp_qty_avail", item->details.count);
item = 0;
item = client->GetPlayer()->item_list.GetItemFromID((*itr));
if (item) {
packet->setSubstructDataByName("recipe_info", "fuel_comp_qty_avail", item->details.count);
}
}
// Set the default item id to the first component id
//packet->setDataByName("fuel_selected_item_id", firstID);
//packet->setDataByName("fuel_item_selected", 1);
item = 0;
item = client->GetPlayer()->item_list.GetItemFromID(firstID);
if (item) {
// packet->setDataByName("fuel_selected_item_qty", min(recipe->GetFuelComponentQuantity(), (int8)item->details.count));
}
//packet->setDataByName("fuel_title", recipe->GetFuelComponentTitle());
//packet->setDataByName("fuel_qty", recipe->GetFuelComponentQuantity());
// Reset the variables we will reuse
i = 0;
firstID = 0;
item = 0;
}
else {
LogWrite(TRADESKILL__ERROR, 0, "Recipes", "Recipe has no fuel component");
}
packet->setSubstructDataByName("recipe_info", "fuel_comp", recipe->fuel_comp_title);
packet->setSubstructDataByName("recipe_info", "fuel_comp_qty", recipe->fuel_comp_qty);
//packet->setSubstructDataByName("recipe_info", "fuel_comp_qty_avail", 10);
packet->setSubstructDataByName("recipe_info", "build_comp_qty_avail_flag", 1);
packet->setSubstructDataByName("recipe_info", "unknown6", 4, 0);
packet->setSubstructDataByName("recipe_info", "min_product", 1);
packet->setSubstructDataByName("recipe_info", "max_product", 1);
packet->setSubstructDataByName("recipe_info", "available_flag", 4);
packet->setSubstructDataByName("recipe_info", "not_commissionable", 1);
packet->setSubstructDataByName("recipe_info", "recipe_name", recipe->GetName());
packet->setSubstructDataByName("recipe_info", "recipe_description", "The art of sculpting metal into a candelabra.");
//packet->PrintPacket();
EQ2Packet* data = packet->serialize();
EQ2Packet* app = new EQ2Packet(OP_ClientCmdMsg, data->pBuffer, data->size);
safe_delete(packet);
safe_delete(data);
//DumpPacket(app);
return app;
}
void Recipe::AddBuildComp(int32 itemID, int8 slot) {
components[slot].push_back(itemID);
}