/* Minetest Copyright (C) 2013 celeron55, Perttu Ahola This program is free software; you can redistribute it and/or modify it under the terms of the GNU Lesser General Public License as published by the Free Software Foundation; either version 2.1 of the License, or (at your option) any later version. This program 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 Lesser General Public License for more details. You should have received a copy of the GNU Lesser General Public License along with this program; if not, write to the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. */ #include "craftdef.h" #include "irrlichttypes.h" #include "log.h" #include #include #include #include "gamedef.h" #include "inventory.h" #include "util/serialize.h" #include "strfnd.h" #include "exceptions.h" // Check if input matches recipe // Takes recipe groups into account static bool inputItemMatchesRecipe(const std::string &inp_name, const std::string &rec_name, IItemDefManager *idef) { // Exact name if(inp_name == rec_name) return true; // Group if(rec_name.substr(0,6) == "group:" && idef->isKnown(inp_name)){ const struct ItemDefinition &def = idef->get(inp_name); Strfnd f(rec_name.substr(6)); bool all_groups_match = true; do{ std::string check_group = f.next(","); if(itemgroup_get(def.groups, check_group) == 0){ all_groups_match = false; break; } }while(!f.atend()); if(all_groups_match) return true; } // Didn't match return false; } // Deserialize an itemstring then return the name of the item static std::string craftGetItemName(const std::string &itemstring, IGameDef *gamedef) { ItemStack item; item.deSerialize(itemstring, gamedef->idef()); return item.name; } // (mapcar craftGetItemName itemstrings) static std::vector craftGetItemNames( const std::vector &itemstrings, IGameDef *gamedef) { std::vector result; for(std::vector::const_iterator i = itemstrings.begin(); i != itemstrings.end(); i++) { result.push_back(craftGetItemName(*i, gamedef)); } return result; } // Get name of each item, and return them as a new list. static std::vector craftGetItemNames( const std::vector &items, IGameDef *gamedef) { std::vector result; for(std::vector::const_iterator i = items.begin(); i != items.end(); i++) { result.push_back(i->name); } return result; } // convert a list of item names, to ItemStacks. static std::vector craftGetItems( const std::vector &items, IGameDef *gamedef) { std::vector result; for(std::vector::const_iterator i = items.begin(); i != items.end(); i++) { result.push_back(ItemStack(std::string(*i),(u16)1,(u16)0,"",gamedef->getItemDefManager())); } return result; } // Compute bounding rectangle given a matrix of items // Returns false if every item is "" static bool craftGetBounds(const std::vector &items, unsigned int width, unsigned int &min_x, unsigned int &max_x, unsigned int &min_y, unsigned int &max_y) { bool success = false; unsigned int x = 0; unsigned int y = 0; for(std::vector::const_iterator i = items.begin(); i != items.end(); i++) { if(*i != "") // Is this an actual item? { if(!success) { // This is the first nonempty item min_x = max_x = x; min_y = max_y = y; success = true; } else { if(x < min_x) min_x = x; if(x > max_x) max_x = x; if(y < min_y) min_y = y; if(y > max_y) max_y = y; } } // Step coordinate x++; if(x == width) { x = 0; y++; } } return success; } // Removes 1 from each item stack static void craftDecrementInput(CraftInput &input, IGameDef *gamedef) { for(std::vector::iterator i = input.items.begin(); i != input.items.end(); i++) { if(i->count != 0) i->remove(1); } } // Removes 1 from each item stack with replacement support // Example: if replacements contains the pair ("bucket:bucket_water", "bucket:bucket_empty"), // a water bucket will not be removed but replaced by an empty bucket. static void craftDecrementOrReplaceInput(CraftInput &input, const CraftReplacements &replacements, IGameDef *gamedef) { if(replacements.pairs.empty()) { craftDecrementInput(input, gamedef); return; } // Make a copy of the replacements pair list std::vector > pairs = replacements.pairs; for(std::vector::iterator i = input.items.begin(); i != input.items.end(); i++) { if(i->count == 1) { // Find an appropriate replacement bool found_replacement = false; for(std::vector >::iterator j = pairs.begin(); j != pairs.end(); j++) { ItemStack from_item; from_item.deSerialize(j->first, gamedef->idef()); if(i->name == from_item.name) { i->deSerialize(j->second, gamedef->idef()); found_replacement = true; pairs.erase(j); break; } } // No replacement was found, simply decrement count to zero if(!found_replacement) i->remove(1); } else if(i->count >= 2) { // Ignore replacements for items with count >= 2 i->remove(1); } } } // Dump an itemstring matrix static std::string craftDumpMatrix(const std::vector &items, unsigned int width) { std::ostringstream os(std::ios::binary); os<<"{ "; unsigned int x = 0; for(std::vector::const_iterator i = items.begin(); i != items.end(); i++, x++) { if(x == width) { os<<"; "; x = 0; } else if(x != 0) { os<<","; } os<<"\""<<(*i)<<"\""; } os<<" }"; return os.str(); } // Dump an item matrix std::string craftDumpMatrix(const std::vector &items, unsigned int width) { std::ostringstream os(std::ios::binary); os<<"{ "; unsigned int x = 0; for(std::vector::const_iterator i = items.begin(); i != items.end(); i++, x++) { if(x == width) { os<<"; "; x = 0; } else if(x != 0) { os<<","; } os<<"\""<<(i->getItemString())<<"\""; } os<<" }"; return os.str(); } /* CraftInput */ std::string CraftInput::dump() const { std::ostringstream os(std::ios::binary); os<<"(method="<<((int)method)<<", items="< >::const_iterator i = pairs.begin(); i != pairs.end(); i++) { os<first)<<"\"=>\""<<(i->second)<<"\""; sep = ","; } os<<"}"; return os.str(); } void CraftReplacements::serialize(std::ostream &os) const { writeU16(os, pairs.size()); for(u32 i=0; i std::string getMessage() { std::string s = m_message; m_message = ""; if(s != "") return std::string("[quicktune] ") + s; return ""; } std::string getSelectedName() { if(m_selected_i < m_names.size()) return m_names[m_selected_i]; return "(nothing)"; } void next() { m_names = getQuicktuneNames(); if(m_selected_i < m_names.size()-1) m_selected_i++; else m_selected_i = 0; m_message = std::string("Selected \"")+getSelectedName()+"\""; } void prev() { m_names = getQuicktuneNames(); if(m_selected_i > 0) m_selected_i--; else m_selected_i = m_names.size()-1; m_message = std::string("Selected \"")+getSelectedName()+"\""; } void inc() { QuicktuneValue val = getQuicktuneValue(getSelectedName()); val.relativeAdd(0.05); m_message = std::string("\"")+getSelectedName() +"\" = "+val.getString(); setQuicktuneValue(getSelectedName(), val); } void dec() { QuicktuneValue val = getQuicktuneValue(getSelectedName()); val.relativeAdd(-0.05); m_message = std::string("\"")+getSelectedName() +"\" = "+val.getString(); setQuicktuneValue(getSelectedName(), val); } }; #endif const ItemStack &item2, float additional_wear, IGameDef *gamedef) { IItemDefManager *idef = gamedef->idef(); if(item1.count != 1 || item2.count != 1 || item1.name != item2.name || idef->get(item1.name).type != ITEM_TOOL || idef->get(item2.name).type != ITEM_TOOL) { // Failure return ItemStack(); } s32 item1_uses = 65536 - (u32) item1.wear; s32 item2_uses = 65536 - (u32) item2.wear; s32 new_uses = item1_uses + item2_uses; s32 new_wear = 65536 - new_uses + floor(additional_wear * 65536 + 0.5); if(new_wear >= 65536) return ItemStack(); if(new_wear < 0) new_wear = 0; ItemStack repaired = item1; repaired.wear = new_wear; return repaired; } std::string CraftDefinitionToolRepair::getName() const { return "toolrepair"; } bool CraftDefinitionToolRepair::check(const CraftInput &input, IGameDef *gamedef) const { if(input.method != CRAFT_METHOD_NORMAL) return false; ItemStack item1; ItemStack item2; for(std::vector::const_iterator i = input.items.begin(); i != input.items.end(); i++) { if(!i->empty()) { if(item1.empty()) item1 = *i; else if(item2.empty()) item2 = *i; else return false; } } ItemStack repaired = craftToolRepair(item1, item2, additional_wear, gamedef); return !repaired.empty(); } CraftOutput CraftDefinitionToolRepair::getOutput(const CraftInput &input, IGameDef *gamedef) const { ItemStack item1; ItemStack item2; for(std::vector::const_iterator i = input.items.begin(); i != input.items.end(); i++) { if(!i->empty()) { if(item1.empty()) item1 = *i; else if(item2.empty()) item2 = *i; } } ItemStack repaired = craftToolRepair(item1, item2, additional_wear, gamedef); return CraftOutput(repaired.getItemString(), 0); } CraftInput CraftDefinitionToolRepair::getInput(const CraftOutput &output, IGameDef *gamedef) const { std::vector stack; stack.push_back(ItemStack()); return CraftInput(CRAFT_METHOD_COOKING,additional_wear,stack); } void CraftDefinitionToolRepair::decrementInput(CraftInput &input, IGameDef *gamedef) const { craftDecrementInput(input, gamedef); } std::string CraftDefinitionToolRepair::dump() const { std::ostringstream os(std::ios::binary); os<<"(toolrepair, additional_wear="< input_filtered; for(std::vector::const_iterator i = input.items.begin(); i != input.items.end(); i++) { if(i->name != "") input_filtered.push_back(i->name); } // If there is a wrong number of items in input, no match if(input_filtered.size() != 1){ /*dstream<<"Number of input items ("< input_filtered; for(std::vector::const_iterator i = input.items.begin(); i != input.items.end(); i++) { if(i->name != "") input_filtered.push_back(i->name); } // If there is a wrong number of items in input, no match if(input_filtered.size() != 1){ /*dstream<<"Number of input items ("< rec; rec.push_back(recipe); return CraftInput(CRAFT_METHOD_COOKING,(int)burntime,craftGetItems(rec,gamedef)); } void CraftDefinitionFuel::decrementInput(CraftInput &input, IGameDef *gamedef) const { craftDecrementOrReplaceInput(input, replacements, gamedef); } std::string CraftDefinitionFuel::dump() const { std::ostringstream os(std::ios::binary); os<<"(fuel, recipe=\""<::const_iterator i = input.items.begin(); i != input.items.end(); i++) { if(!i->empty()) { all_empty = false; break; } } if(all_empty) return false; // Walk crafting definitions from back to front, so that later // definitions can override earlier ones. for(std::vector::const_reverse_iterator i = m_craft_definitions.rbegin(); i != m_craft_definitions.rend(); i++) { CraftDefinition *def = *i; /*infostream<<"Checking "<dump()<check(input, gamedef)) { // Get output, then decrement input (if requested) output = def->getOutput(input, gamedef); if(decrementInput) def->decrementInput(input, gamedef); return true; } } catch(SerializationError &e) { errorstream<<"getCraftResult: ERROR: " <<"Serialization error in recipe " <dump()<::const_reverse_iterator i = m_craft_definitions.rbegin(); i != m_craft_definitions.rend(); i++) { CraftDefinition *def = *i; /*infostream<<"Checking "<dump()<getOutput(input, gamedef); if((tmpout.item.substr(0,output.item.length()) == output.item) && ((tmpout.item[output.item.length()] == 0) || (tmpout.item[output.item.length()] == ' '))) { // Get output, then decrement input (if requested) input = def->getInput(output, gamedef); return true; } } catch(SerializationError &e) { errorstream<<"getCraftResult: ERROR: " <<"Serialization error in recipe " <dump()< getCraftRecipes(CraftOutput &output, IGameDef *gamedef) const { std::vector recipes_list; CraftInput input; CraftOutput tmpout; tmpout.item = ""; tmpout.time = 0; for(std::vector::const_reverse_iterator i = m_craft_definitions.rbegin(); i != m_craft_definitions.rend(); i++) { CraftDefinition *def = *i; /*infostream<<"Checking "<dump()<getOutput(input, gamedef); if(tmpout.item.substr(0,output.item.length()) == output.item) { // Get output, then decrement input (if requested) input = def->getInput(output, gamedef); recipes_list.push_back(*i); } } catch(SerializationError &e) { errorstream<<"getCraftResult: ERROR: " <<"Serialization error in recipe " <dump()<::const_iterator i = m_craft_definitions.begin(); i != m_craft_definitions.end(); i++) { os<<(*i)->dump()<<"\n"; } return os.str(); } virtual void registerCraft(CraftDefinition *def) { verbosestream<<"registerCraft: registering craft definition: " <dump()<::iterator i = m_craft_definitions.begin(); i != m_craft_definitions.end(); i++){ delete *i; } m_craft_definitions.clear(); } virtual void serialize(std::ostream &os) const { writeU8(os, 0); // version u16 count = m_craft_definitions.size(); writeU16(os, count); for(std::vector::const_iterator i = m_craft_definitions.begin(); i != m_craft_definitions.end(); i++){ CraftDefinition *def = *i; // Serialize wrapped in a string std::ostringstream tmp_os(std::ios::binary); def->serialize(tmp_os); os< m_craft_definitions; }; IWritableCraftDefManager* createCraftDefManager() { return new CCraftDefManager(); }