/* 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); }