1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078 |
- /*
- Minetest
- Copyright (C) 2013 celeron55, Perttu Ahola <celeron55@gmail.com>
- 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 <sstream>
- #include <set>
- #include <algorithm>
- #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<std::string> craftGetItemNames(
- const std::vector<std::string> &itemstrings, IGameDef *gamedef)
- {
- std::vector<std::string> result;
- for(std::vector<std::string>::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<std::string> craftGetItemNames(
- const std::vector<ItemStack> &items, IGameDef *gamedef)
- {
- std::vector<std::string> result;
- for(std::vector<ItemStack>::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<ItemStack> craftGetItems(
- const std::vector<std::string> &items, IGameDef *gamedef)
- {
- std::vector<ItemStack> result;
- for(std::vector<std::string>::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<std::string> &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<std::string>::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<ItemStack>::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<std::pair<std::string, std::string> > pairs = replacements.pairs;
- for(std::vector<ItemStack>::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<std::pair<std::string, std::string> >::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<std::string> &items,
- unsigned int width)
- {
- std::ostringstream os(std::ios::binary);
- os<<"{ ";
- unsigned int x = 0;
- for(std::vector<std::string>::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<ItemStack> &items,
- unsigned int width)
- {
- std::ostringstream os(std::ios::binary);
- os<<"{ ";
- unsigned int x = 0;
- for(std::vector<ItemStack>::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="<<craftDumpMatrix(items, width)<<")";
- return os.str();
- }
- /*
- CraftOutput
- */
- std::string CraftOutput::dump() const
- {
- std::ostringstream os(std::ios::binary);
- os<<"(item=\""<<item<<"\", time="<<time<<")";
- return os.str();
- }
- /*
- CraftReplacements
- */
- std::string CraftReplacements::dump() const
- {
- std::ostringstream os(std::ios::binary);
- os<<"{";
- const char *sep = "";
- for(std::vector<std::pair<std::string, std::string> >::const_iterator
- i = pairs.begin();
- i != pairs.end(); i++)
- {
- os<<sep<<"\""<<(i->first)<<"\"=>\""<<(i->second)<<"\"";
- sep = ",";
- }
- os<<"}";
- return os.str();
- }
- void CraftReplacements::serialize(std::ostream &os) const
- {
- writeU16(os, pairs.size());
- for(u32 i=0; i<pairs.size(); i++)
- {
- os<<serializeString(pairs[i].first);
- os<<serializeString(pairs[i].second);
- }
- }
- void CraftReplacements::deSerialize(std::istream &is)
- {
- pairs.clear();
- u32 count = readU16(is);
- for(u32 i=0; i<count; i++)
- {
- std::string first = deSerializeString(is);
- std::string second = deSerializeString(is);
- pairs.push_back(std::make_pair(first, second));
- }
- }
- /*
- CraftDefinition
- */
- void CraftDefinition::serialize(std::ostream &os) const
- {
- writeU8(os, 1); // version
- os<<serializeString(getName());
- serializeBody(os);
- }
- CraftDefinition* CraftDefinition::deSerialize(std::istream &is)
- {
- int version = readU8(is);
- if(version != 1) throw SerializationError(
- "unsupported CraftDefinition version");
- std::string name = deSerializeString(is);
- CraftDefinition *def = NULL;
- if(name == "shaped")
- {
- def = new CraftDefinitionShaped;
- }
- else if(name == "shapeless")
- {
- def = new CraftDefinitionShapeless;
- }
- else if(name == "toolrepair")
- {
- def = new CraftDefinitionToolRepair;
- }
- else if(name == "cooking")
- {
- def = new CraftDefinitionCooking;
- }
- else if(name == "fuel")
- {
- def = new CraftDefinitionFuel;
- }
- else
- {
- infostream<<"Unknown CraftDefinition name=\""<<name<<"\""<<std::endl;
- throw SerializationError("Unknown CraftDefinition name");
- }
- def->deSerializeBody(is, version);
- return def;
- }
- /*
- CraftDefinitionShaped
- */
- std::string CraftDefinitionShaped::getName() const
- {
- return "shaped";
- }
- bool CraftDefinitionShaped::check(const CraftInput &input, IGameDef *gamedef) const
- {
- if(input.method != CRAFT_METHOD_NORMAL)
- return false;
- // Get input item matrix
- std::vector<std::string> inp_names = craftGetItemNames(input.items, gamedef);
- unsigned int inp_width = input.width;
- if(inp_width == 0)
- return false;
- while(inp_names.size() % inp_width != 0)
- inp_names.push_back("");
- // Get input bounds
- unsigned int inp_min_x=0, inp_max_x=0, inp_min_y=0, inp_max_y=0;
- if(!craftGetBounds(inp_names, inp_width, inp_min_x, inp_max_x, inp_min_y, inp_max_y))
- return false; // it was empty
- // Get recipe item matrix
- std::vector<std::string> rec_names = craftGetItemNames(recipe, gamedef);
- unsigned int rec_width = width;
- if(rec_width == 0)
- return false;
- while(rec_names.size() % rec_width != 0)
- rec_names.push_back("");
- // Get recipe bounds
- unsigned int rec_min_x=0, rec_max_x=0, rec_min_y=0, rec_max_y=0;
- if(!craftGetBounds(rec_names, rec_width, rec_min_x, rec_max_x, rec_min_y, rec_max_y))
- return false; // it was empty
- // Different sizes?
- if(inp_max_x - inp_min_x != rec_max_x - rec_min_x)
- return false;
- if(inp_max_y - inp_min_y != rec_max_y - rec_min_y)
- return false;
- // Verify that all item names in the bounding box are equal
- unsigned int w = inp_max_x - inp_min_x + 1;
- unsigned int h = inp_max_y - inp_min_y + 1;
- for(unsigned int y=0; y<h; y++)
- for(unsigned int x=0; x<w; x++)
- {
- unsigned int inp_x = inp_min_x + x;
- unsigned int inp_y = inp_min_y + y;
- unsigned int rec_x = rec_min_x + x;
- unsigned int rec_y = rec_min_y + y;
- if(!inputItemMatchesRecipe(
- inp_names[inp_y * inp_width + inp_x],
- rec_names[rec_y * rec_width + rec_x], gamedef->idef())
- ){
- return false;
- }
- }
- return true;
- }
- CraftOutput CraftDefinitionShaped::getOutput(const CraftInput &input, IGameDef *gamedef) const
- {
- return CraftOutput(output, 0);
- }
- CraftInput CraftDefinitionShaped::getInput(const CraftOutput &output, IGameDef *gamedef) const
- {
- return CraftInput(CRAFT_METHOD_NORMAL,width,craftGetItems(recipe,gamedef));
- }
- void CraftDefinitionShaped::decrementInput(CraftInput &input, IGameDef *gamedef) const
- {
- craftDecrementOrReplaceInput(input, replacements, gamedef);
- }
- std::string CraftDefinitionShaped::dump() const
- {
- std::ostringstream os(std::ios::binary);
- os<<"(shaped, output=\""<<output
- <<"\", recipe="<<craftDumpMatrix(recipe, width)
- <<", replacements="<<replacements.dump()<<")";
- return os.str();
- }
- void CraftDefinitionShaped::serializeBody(std::ostream &os) const
- {
- os<<serializeString(output);
- writeU16(os, width);
- writeU16(os, recipe.size());
- for(u32 i=0; i<recipe.size(); i++)
- os<<serializeString(recipe[i]);
- replacements.serialize(os);
- }
- void CraftDefinitionShaped::deSerializeBody(std::istream &is, int version)
- {
- if(version != 1) throw SerializationError(
- "unsupported CraftDefinitionShaped version");
- output = deSerializeString(is);
- width = readU16(is);
- recipe.clear();
- u32 count = readU16(is);
- for(u32 i=0; i<count; i++)
- recipe.push_back(deSerializeString(is));
- replacements.deSerialize(is);
- }
- /*
- CraftDefinitionShapeless
- */
- std::string CraftDefinitionShapeless::getName() const
- {
- return "shapeless";
- }
- bool CraftDefinitionShapeless::check(const CraftInput &input, IGameDef *gamedef) const
- {
- if(input.method != CRAFT_METHOD_NORMAL)
- return false;
-
- // Filter empty items out of input
- std::vector<std::string> input_filtered;
- for(std::vector<ItemStack>::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() != recipe.size()){
- /*dstream<<"Number of input items ("<<input_filtered.size()
- <<") does not match recipe size ("<<recipe.size()<<") "
- <<"of recipe with output="<<output<<std::endl;*/
- return false;
- }
- // Try with all permutations of the recipe
- std::vector<std::string> recipe_copy = craftGetItemNames(recipe, gamedef);
- // Start from the lexicographically first permutation (=sorted)
- std::sort(recipe_copy.begin(), recipe_copy.end());
- //while(std::prev_permutation(recipe_copy.begin(), recipe_copy.end())){}
- do{
- // If all items match, the recipe matches
- bool all_match = true;
- //dstream<<"Testing recipe (output="<<output<<"):";
- for(size_t i=0; i<recipe.size(); i++){
- //dstream<<" ("<<input_filtered[i]<<" == "<<recipe_copy[i]<<")";
- if(!inputItemMatchesRecipe(input_filtered[i], recipe_copy[i],
- gamedef->idef())){
- all_match = false;
- break;
- }
- }
- //dstream<<" -> match="<<all_match<<std::endl;
- if(all_match)
- return true;
- }while(std::next_permutation(recipe_copy.begin(), recipe_copy.end()));
- return false;
- }
- CraftOutput CraftDefinitionShapeless::getOutput(const CraftInput &input, IGameDef *gamedef) const
- {
- return CraftOutput(output, 0);
- }
- CraftInput CraftDefinitionShapeless::getInput(const CraftOutput &output, IGameDef *gamedef) const
- {
- return CraftInput(CRAFT_METHOD_NORMAL,0,craftGetItems(recipe,gamedef));
- }
- void CraftDefinitionShapeless::decrementInput(CraftInput &input, IGameDef *gamedef) const
- {
- craftDecrementOrReplaceInput(input, replacements, gamedef);
- }
- std::string CraftDefinitionShapeless::dump() const
- {
- std::ostringstream os(std::ios::binary);
- os<<"(shapeless, output=\""<<output
- <<"\", recipe="<<craftDumpMatrix(recipe, recipe.size())
- <<", replacements="<<replacements.dump()<<")";
- return os.str();
- }
- void CraftDefinitionShapeless::serializeBody(std::ostream &os) const
- {
- os<<serializeString(output);
- writeU16(os, recipe.size());
- for(u32 i=0; i<recipe.size(); i++)
- os<<serializeString(recipe[i]);
- replacements.serialize(os);
- }
- void CraftDefinitionShapeless::deSerializeBody(std::istream &is, int version)
- {
- if(version != 1) throw SerializationError(
- "unsupported CraftDefinitionShapeless version");
- output = deSerializeString(is);
- recipe.clear();
- u32 count = readU16(is);
- for(u32 i=0; i<count; i++)
- recipe.push_back(deSerializeString(is));
- replacements.deSerialize(is);
- }
- /*
- CraftDefinitionToolRepair
- */
- static ItemStack craftToolRepair(
- const ItemStack &item1,
- 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<ItemStack>::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<ItemStack>::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<ItemStack> 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="<<additional_wear<<")";
- return os.str();
- }
- void CraftDefinitionToolRepair::serializeBody(std::ostream &os) const
- {
- writeF1000(os, additional_wear);
- }
- void CraftDefinitionToolRepair::deSerializeBody(std::istream &is, int version)
- {
- if(version != 1) throw SerializationError(
- "unsupported CraftDefinitionToolRepair version");
- additional_wear = readF1000(is);
- }
- /*
- CraftDefinitionCooking
- */
- std::string CraftDefinitionCooking::getName() const
- {
- return "cooking";
- }
- bool CraftDefinitionCooking::check(const CraftInput &input, IGameDef *gamedef) const
- {
- if(input.method != CRAFT_METHOD_COOKING)
- return false;
- // Filter empty items out of input
- std::vector<std::string> input_filtered;
- for(std::vector<ItemStack>::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.size()
- <<") does not match recipe size (1) "
- <<"of cooking recipe with output="<<output<<std::endl;*/
- return false;
- }
-
- // Check the single input item
- return inputItemMatchesRecipe(input_filtered[0], recipe, gamedef->idef());
- }
- CraftOutput CraftDefinitionCooking::getOutput(const CraftInput &input, IGameDef *gamedef) const
- {
- return CraftOutput(output, cooktime);
- }
- CraftInput CraftDefinitionCooking::getInput(const CraftOutput &output, IGameDef *gamedef) const
- {
- std::vector<std::string> rec;
- rec.push_back(recipe);
- return CraftInput(CRAFT_METHOD_COOKING,cooktime,craftGetItems(rec,gamedef));
- }
- void CraftDefinitionCooking::decrementInput(CraftInput &input, IGameDef *gamedef) const
- {
- craftDecrementOrReplaceInput(input, replacements, gamedef);
- }
- std::string CraftDefinitionCooking::dump() const
- {
- std::ostringstream os(std::ios::binary);
- os<<"(cooking, output=\""<<output
- <<"\", recipe=\""<<recipe
- <<"\", cooktime="<<cooktime<<")"
- <<", replacements="<<replacements.dump()<<")";
- return os.str();
- }
- void CraftDefinitionCooking::serializeBody(std::ostream &os) const
- {
- os<<serializeString(output);
- os<<serializeString(recipe);
- writeF1000(os, cooktime);
- replacements.serialize(os);
- }
- void CraftDefinitionCooking::deSerializeBody(std::istream &is, int version)
- {
- if(version != 1) throw SerializationError(
- "unsupported CraftDefinitionCooking version");
- output = deSerializeString(is);
- recipe = deSerializeString(is);
- cooktime = readF1000(is);
- replacements.deSerialize(is);
- }
- /*
- CraftDefinitionFuel
- */
- std::string CraftDefinitionFuel::getName() const
- {
- return "fuel";
- }
- bool CraftDefinitionFuel::check(const CraftInput &input, IGameDef *gamedef) const
- {
- if(input.method != CRAFT_METHOD_FUEL)
- return false;
- // Filter empty items out of input
- std::vector<std::string> input_filtered;
- for(std::vector<ItemStack>::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.size()
- <<") does not match recipe size (1) "
- <<"of fuel recipe with burntime="<<burntime<<std::endl;*/
- return false;
- }
-
- // Check the single input item
- return inputItemMatchesRecipe(input_filtered[0], recipe, gamedef->idef());
- }
- CraftOutput CraftDefinitionFuel::getOutput(const CraftInput &input, IGameDef *gamedef) const
- {
- return CraftOutput("", burntime);
- }
- CraftInput CraftDefinitionFuel::getInput(const CraftOutput &output, IGameDef *gamedef) const
- {
- std::vector<std::string> 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=\""<<recipe
- <<"\", burntime="<<burntime<<")"
- <<", replacements="<<replacements.dump()<<")";
- return os.str();
- }
- void CraftDefinitionFuel::serializeBody(std::ostream &os) const
- {
- os<<serializeString(recipe);
- writeF1000(os, burntime);
- replacements.serialize(os);
- }
- void CraftDefinitionFuel::deSerializeBody(std::istream &is, int version)
- {
- if(version != 1) throw SerializationError(
- "unsupported CraftDefinitionFuel version");
- recipe = deSerializeString(is);
- burntime = readF1000(is);
- replacements.deSerialize(is);
- }
- /*
- Craft definition manager
- */
- class CCraftDefManager: public IWritableCraftDefManager
- {
- public:
- virtual ~CCraftDefManager()
- {
- clear();
- }
- virtual bool getCraftResult(CraftInput &input, CraftOutput &output,
- bool decrementInput, IGameDef *gamedef) const
- {
- output.item = "";
- output.time = 0;
- // If all input items are empty, abort.
- bool all_empty = true;
- for(std::vector<ItemStack>::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<CraftDefinition*>::const_reverse_iterator
- i = m_craft_definitions.rbegin();
- i != m_craft_definitions.rend(); i++)
- {
- CraftDefinition *def = *i;
- /*infostream<<"Checking "<<input.dump()<<std::endl
- <<" against "<<def->dump()<<std::endl;*/
- try {
- if(def->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 "
- <<def->dump()<<std::endl;
- // then go on with the next craft definition
- }
- }
- return false;
- }
- virtual bool getCraftRecipe(CraftInput &input, CraftOutput &output,
- IGameDef *gamedef) const
- {
- CraftOutput tmpout;
- tmpout.item = "";
- tmpout.time = 0;
- // If output item is empty, abort.
- if(output.item.empty())
- return false;
- // Walk crafting definitions from back to front, so that later
- // definitions can override earlier ones.
- for(std::vector<CraftDefinition*>::const_reverse_iterator
- i = m_craft_definitions.rbegin();
- i != m_craft_definitions.rend(); i++)
- {
- CraftDefinition *def = *i;
- /*infostream<<"Checking "<<input.dump()<<std::endl
- <<" against "<<def->dump()<<std::endl;*/
- try {
- tmpout = def->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);
- return true;
- }
- }
- catch(SerializationError &e)
- {
- errorstream<<"getCraftResult: ERROR: "
- <<"Serialization error in recipe "
- <<def->dump()<<std::endl;
- // then go on with the next craft definition
- }
- }
- return false;
- }
- virtual std::vector<CraftDefinition*> getCraftRecipes(CraftOutput &output,
- IGameDef *gamedef) const
- {
- std::vector<CraftDefinition*> recipes_list;
- CraftInput input;
- CraftOutput tmpout;
- tmpout.item = "";
- tmpout.time = 0;
- for(std::vector<CraftDefinition*>::const_reverse_iterator
- i = m_craft_definitions.rbegin();
- i != m_craft_definitions.rend(); i++)
- {
- CraftDefinition *def = *i;
- /*infostream<<"Checking "<<input.dump()<<std::endl
- <<" against "<<def->dump()<<std::endl;*/
- try {
- tmpout = def->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 "
- <<def->dump()<<std::endl;
- // then go on with the next craft definition
- }
- }
- return recipes_list;
- }
- virtual std::string dump() const
- {
- std::ostringstream os(std::ios::binary);
- os<<"Crafting definitions:\n";
- for(std::vector<CraftDefinition*>::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: "
- <<def->dump()<<std::endl;
- m_craft_definitions.push_back(def);
- }
- virtual void clear()
- {
- for(std::vector<CraftDefinition*>::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<CraftDefinition*>::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<<serializeString(tmp_os.str());
- }
- }
- virtual void deSerialize(std::istream &is)
- {
- // Clear everything
- clear();
- // Deserialize
- int version = readU8(is);
- if(version != 0) throw SerializationError(
- "unsupported CraftDefManager version");
- u16 count = readU16(is);
- for(u16 i=0; i<count; i++){
- // Deserialize a string and grab a CraftDefinition from it
- std::istringstream tmp_is(deSerializeString(is), std::ios::binary);
- CraftDefinition *def = CraftDefinition::deSerialize(tmp_is);
- // Register
- registerCraft(def);
- }
- }
- private:
- std::vector<CraftDefinition*> m_craft_definitions;
- };
- IWritableCraftDefManager* createCraftDefManager()
- {
- return new CCraftDefManager();
- }
|