2020-04-03 02:57:01 -04:00
|
|
|
// BlockType.cpp
|
2012-06-14 09:06:06 -04:00
|
|
|
|
2020-04-03 02:57:01 -04:00
|
|
|
// Implements the helper functions for converting Block Type string to int etc.
|
2012-06-14 09:06:06 -04:00
|
|
|
|
|
|
|
#include "Globals.h"
|
2014-10-23 09:15:10 -04:00
|
|
|
#include "IniFile.h"
|
2012-09-23 18:09:57 -04:00
|
|
|
#include "Item.h"
|
2013-08-16 04:48:19 -04:00
|
|
|
#include "Mobs/Monster.h"
|
2012-06-14 09:06:06 -04:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class cBlockIDMap
|
|
|
|
{
|
2012-09-08 12:38:27 -04:00
|
|
|
// Making the map case-insensitive:
|
|
|
|
struct Comparator
|
|
|
|
{
|
2014-08-03 05:57:05 -04:00
|
|
|
bool operator ()(const AString & a_Item1, const AString & a_Item2) const
|
2012-09-08 12:38:27 -04:00
|
|
|
{
|
|
|
|
return (NoCaseCompare(a_Item1, a_Item2) > 0);
|
|
|
|
}
|
|
|
|
} ;
|
2016-02-05 16:45:45 -05:00
|
|
|
|
2012-09-08 12:38:27 -04:00
|
|
|
typedef std::map<AString, std::pair<short, short>, Comparator> ItemMap;
|
2016-02-05 16:45:45 -05:00
|
|
|
|
2012-06-14 09:06:06 -04:00
|
|
|
public:
|
2015-03-31 09:50:03 -04:00
|
|
|
|
2012-09-08 12:08:29 -04:00
|
|
|
cBlockIDMap(void)
|
2012-06-14 09:06:06 -04:00
|
|
|
{
|
2013-10-25 05:15:44 -04:00
|
|
|
cIniFile Ini;
|
|
|
|
if (!Ini.ReadFile("items.ini"))
|
2012-09-08 12:08:29 -04:00
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
2013-10-25 05:38:14 -04:00
|
|
|
int KeyID = Ini.FindKey("Items");
|
2012-09-08 12:08:29 -04:00
|
|
|
if (KeyID == cIniFile::noID)
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
2013-10-25 05:38:14 -04:00
|
|
|
int NumValues = Ini.GetNumValues(KeyID);
|
|
|
|
for (int i = 0; i < NumValues; i++)
|
2012-09-08 12:08:29 -04:00
|
|
|
{
|
2013-10-25 05:38:14 -04:00
|
|
|
AString Name = Ini.GetValueName(KeyID, i);
|
2012-09-08 12:08:29 -04:00
|
|
|
if (Name.empty())
|
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
AString Value = Ini.GetValue(KeyID, i);
|
|
|
|
AddToMap(Name, Value);
|
|
|
|
} // for i - Ini.Values[]
|
2012-06-14 09:06:06 -04:00
|
|
|
}
|
2016-02-05 16:45:45 -05:00
|
|
|
|
|
|
|
|
2012-06-14 09:06:06 -04:00
|
|
|
int Resolve(const AString & a_ItemName)
|
|
|
|
{
|
2012-09-08 12:08:29 -04:00
|
|
|
ItemMap::iterator itr = m_Map.find(a_ItemName);
|
|
|
|
if (itr == m_Map.end())
|
|
|
|
{
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
return itr->second.first;
|
|
|
|
}
|
2016-02-05 16:45:45 -05:00
|
|
|
|
|
|
|
|
2012-09-08 12:08:29 -04:00
|
|
|
bool ResolveItem(const AString & a_ItemName, cItem & a_Item)
|
|
|
|
{
|
2013-10-18 12:13:20 -04:00
|
|
|
// Split into parts divided by either ':' or '^'
|
|
|
|
AStringVector Split = StringSplitAndTrim(a_ItemName, ":^");
|
|
|
|
if (Split.empty())
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
ItemMap::iterator itr = m_Map.find(Split[0]);
|
2012-09-08 12:08:29 -04:00
|
|
|
if (itr != m_Map.end())
|
|
|
|
{
|
2013-10-18 12:13:20 -04:00
|
|
|
// Resolved as a string, assign the type and the default damage / count
|
2012-09-08 12:08:29 -04:00
|
|
|
a_Item.m_ItemType = itr->second.first;
|
|
|
|
a_Item.m_ItemDamage = itr->second.second;
|
2012-09-08 17:20:36 -04:00
|
|
|
if (a_Item.m_ItemDamage == -1)
|
|
|
|
{
|
|
|
|
a_Item.m_ItemDamage = 0;
|
|
|
|
}
|
2012-09-08 12:08:29 -04:00
|
|
|
}
|
2013-10-18 12:13:20 -04:00
|
|
|
else
|
2012-09-08 12:08:29 -04:00
|
|
|
{
|
2013-10-18 12:13:20 -04:00
|
|
|
// Not a resolvable string, try pure numbers: "45:6", "45^6" etc.
|
2015-05-24 07:56:56 -04:00
|
|
|
if (!StringToInteger(Split[0], a_Item.m_ItemType))
|
2013-10-18 12:13:20 -04:00
|
|
|
{
|
|
|
|
// Parsing the number failed
|
|
|
|
return false;
|
|
|
|
}
|
2012-09-08 12:08:29 -04:00
|
|
|
}
|
2016-02-05 16:45:45 -05:00
|
|
|
|
2013-10-18 12:13:20 -04:00
|
|
|
// Parse the damage, if present:
|
2012-09-08 12:08:29 -04:00
|
|
|
if (Split.size() < 2)
|
|
|
|
{
|
2013-10-18 12:13:20 -04:00
|
|
|
// Not present, set the item as valid and return success:
|
2012-09-08 17:33:53 -04:00
|
|
|
a_Item.m_ItemCount = 1;
|
2012-09-08 12:08:29 -04:00
|
|
|
return true;
|
|
|
|
}
|
2015-05-24 07:56:56 -04:00
|
|
|
|
|
|
|
if (!StringToInteger(Split[1], a_Item.m_ItemDamage))
|
2012-09-08 12:08:29 -04:00
|
|
|
{
|
|
|
|
// Parsing the number failed
|
|
|
|
return false;
|
|
|
|
}
|
2012-09-08 17:33:53 -04:00
|
|
|
a_Item.m_ItemCount = 1;
|
2012-09-08 12:08:29 -04:00
|
|
|
return true;
|
2012-06-14 09:06:06 -04:00
|
|
|
}
|
2016-02-05 16:45:45 -05:00
|
|
|
|
|
|
|
|
2012-09-08 12:08:29 -04:00
|
|
|
AString Desolve(short a_ItemType, short a_ItemDamage)
|
2012-06-14 09:06:06 -04:00
|
|
|
{
|
2012-09-08 17:49:27 -04:00
|
|
|
// First try an exact match, both ItemType and ItemDamage ("birchplanks=5:2"):
|
2012-09-08 12:08:29 -04:00
|
|
|
for (ItemMap::iterator itr = m_Map.begin(), end = m_Map.end(); itr != end; ++itr)
|
|
|
|
{
|
|
|
|
if ((itr->second.first == a_ItemType) && (itr->second.second == a_ItemDamage))
|
|
|
|
{
|
|
|
|
return itr->first;
|
|
|
|
}
|
|
|
|
} // for itr - m_Map[]
|
2016-02-05 16:45:45 -05:00
|
|
|
|
2012-09-08 17:49:27 -04:00
|
|
|
// There is no exact match, try matching ItemType only ("planks=5"):
|
|
|
|
if (a_ItemDamage == 0)
|
|
|
|
{
|
|
|
|
for (ItemMap::iterator itr = m_Map.begin(), end = m_Map.end(); itr != end; ++itr)
|
|
|
|
{
|
|
|
|
if ((itr->second.first == a_ItemType) && (itr->second.second == -1))
|
|
|
|
{
|
|
|
|
return itr->first;
|
|
|
|
}
|
|
|
|
} // for itr - m_Map[]
|
|
|
|
}
|
|
|
|
|
|
|
|
// No match at all, synthesize a string ("5:1"):
|
2012-09-08 12:08:29 -04:00
|
|
|
AString res;
|
|
|
|
if (a_ItemDamage == -1)
|
|
|
|
{
|
|
|
|
Printf(res, "%d", a_ItemType);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
Printf(res, "%d:%d", a_ItemType, a_ItemDamage);
|
|
|
|
}
|
|
|
|
return res;
|
2012-06-14 09:06:06 -04:00
|
|
|
}
|
2016-02-05 16:45:45 -05:00
|
|
|
|
|
|
|
|
2012-06-14 09:06:06 -04:00
|
|
|
protected:
|
2012-09-08 12:08:29 -04:00
|
|
|
ItemMap m_Map;
|
2016-02-05 16:45:45 -05:00
|
|
|
|
|
|
|
|
2012-09-08 12:08:29 -04:00
|
|
|
void AddToMap(const AString & a_Name, const AString & a_Value)
|
|
|
|
{
|
|
|
|
AStringVector Split = StringSplit(a_Value, ":");
|
|
|
|
if (Split.size() == 1)
|
|
|
|
{
|
|
|
|
Split = StringSplit(a_Value, "^");
|
|
|
|
}
|
|
|
|
if (Split.empty())
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
2015-05-24 07:56:56 -04:00
|
|
|
short ItemType;
|
|
|
|
if (!StringToInteger(Split[0], ItemType))
|
|
|
|
{
|
|
|
|
ASSERT(!"Invalid item type");
|
|
|
|
}
|
|
|
|
short ItemDamage = -1;
|
|
|
|
if (Split.size() > 1 && !StringToInteger(Split[1], ItemDamage))
|
|
|
|
{
|
|
|
|
ASSERT(!"Invalid item damage");
|
|
|
|
}
|
2012-09-08 12:08:29 -04:00
|
|
|
m_Map[a_Name] = std::make_pair(ItemType, ItemDamage);
|
|
|
|
}
|
2012-06-14 09:06:06 -04:00
|
|
|
} ;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2020-05-10 12:16:49 -04:00
|
|
|
static cBlockIDMap & GetBlockIDMap()
|
|
|
|
{
|
|
|
|
static cBlockIDMap IDMap;
|
|
|
|
return IDMap;
|
|
|
|
}
|
2012-06-14 09:06:06 -04:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2012-09-08 12:08:29 -04:00
|
|
|
/*
|
|
|
|
// Quick self-test:
|
|
|
|
class Tester
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
Tester(void)
|
|
|
|
{
|
|
|
|
cItem Item;
|
|
|
|
gsBlockIDMap.ResolveItem("charcoal", Item);
|
|
|
|
AString Charcoal = gsBlockIDMap.Desolve(Item.m_ItemType, Item.m_ItemDamage);
|
|
|
|
ASSERT(Charcoal == "charcoal");
|
|
|
|
}
|
|
|
|
} test;
|
|
|
|
//*/
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2014-11-27 15:19:52 -05:00
|
|
|
int BlockStringToType(const AString & a_BlockTypeString)
|
2012-06-14 09:06:06 -04:00
|
|
|
{
|
|
|
|
int res = atoi(a_BlockTypeString.c_str());
|
|
|
|
if ((res != 0) || (a_BlockTypeString.compare("0") == 0))
|
|
|
|
{
|
|
|
|
// It was a valid number, return that
|
|
|
|
return res;
|
|
|
|
}
|
2016-02-05 16:45:45 -05:00
|
|
|
|
2020-05-10 12:16:49 -04:00
|
|
|
return GetBlockIDMap().Resolve(TrimString(a_BlockTypeString));
|
2012-06-14 09:06:06 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2018-07-26 17:24:36 -04:00
|
|
|
|
2012-06-14 09:06:06 -04:00
|
|
|
bool StringToItem(const AString & a_ItemTypeString, cItem & a_Item)
|
|
|
|
{
|
2014-10-29 10:47:43 -04:00
|
|
|
AString ItemName = TrimString(a_ItemTypeString);
|
|
|
|
if (ItemName.substr(0, 10) == "minecraft:")
|
|
|
|
{
|
|
|
|
ItemName = ItemName.substr(10);
|
|
|
|
}
|
2015-03-31 09:50:03 -04:00
|
|
|
|
2020-05-10 12:16:49 -04:00
|
|
|
return GetBlockIDMap().ResolveItem(ItemName, a_Item);
|
2012-09-08 12:08:29 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
AString ItemToString(const cItem & a_Item)
|
|
|
|
{
|
2020-05-10 12:16:49 -04:00
|
|
|
return GetBlockIDMap().Desolve(a_Item.m_ItemType, a_Item.m_ItemDamage);
|
2012-09-08 12:08:29 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
AString ItemTypeToString(short a_ItemType)
|
|
|
|
{
|
2020-05-10 12:16:49 -04:00
|
|
|
return GetBlockIDMap().Desolve(a_ItemType, -1);
|
2012-06-14 09:06:06 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2012-09-20 09:25:54 -04:00
|
|
|
AString ItemToFullString(const cItem & a_Item)
|
|
|
|
{
|
|
|
|
AString res;
|
2013-01-11 23:46:01 -05:00
|
|
|
Printf(res, "%s:%d * %d", ItemToString(a_Item).c_str(), a_Item.m_ItemDamage, a_Item.m_ItemCount);
|
2012-09-20 09:25:54 -04:00
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2013-10-11 04:12:36 -04:00
|
|
|
cItem GetIniItemSet(cIniFile & a_IniFile, const char * a_Section, const char * a_Key, const char * a_Default)
|
|
|
|
{
|
|
|
|
AString ItemStr = a_IniFile.GetValueSet(a_Section, a_Key, a_Default);
|
|
|
|
cItem res;
|
|
|
|
if (!StringToItem(ItemStr, res))
|
|
|
|
{
|
|
|
|
res.Empty();
|
|
|
|
}
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2013-11-18 17:30:34 -05:00
|
|
|
|