mirror of
https://github.com/luanti-org/luanti.git
synced 2025-06-27 16:36:03 +00:00
Add ItemStack key-value meta storage
This commit is contained in:
parent
c2e7b1f579
commit
f2aa2c6a98
21 changed files with 459 additions and 123 deletions
|
@ -45,82 +45,16 @@ static content_t content_translate_from_19_to_internal(content_t c_from)
|
|||
return c_from;
|
||||
}
|
||||
|
||||
// If the string contains spaces, quotes or control characters, encodes as JSON.
|
||||
// Else returns the string unmodified.
|
||||
static std::string serializeJsonStringIfNeeded(const std::string &s)
|
||||
{
|
||||
for(size_t i = 0; i < s.size(); ++i)
|
||||
{
|
||||
if(s[i] <= 0x1f || s[i] >= 0x7f || s[i] == ' ' || s[i] == '\"')
|
||||
return serializeJsonString(s);
|
||||
}
|
||||
return s;
|
||||
}
|
||||
|
||||
// Parses a string serialized by serializeJsonStringIfNeeded.
|
||||
static std::string deSerializeJsonStringIfNeeded(std::istream &is)
|
||||
{
|
||||
std::ostringstream tmp_os;
|
||||
bool expect_initial_quote = true;
|
||||
bool is_json = false;
|
||||
bool was_backslash = false;
|
||||
for(;;)
|
||||
{
|
||||
char c = is.get();
|
||||
if(is.eof())
|
||||
break;
|
||||
if(expect_initial_quote && c == '"')
|
||||
{
|
||||
tmp_os << c;
|
||||
is_json = true;
|
||||
}
|
||||
else if(is_json)
|
||||
{
|
||||
tmp_os << c;
|
||||
if(was_backslash)
|
||||
was_backslash = false;
|
||||
else if(c == '\\')
|
||||
was_backslash = true;
|
||||
else if(c == '"')
|
||||
break; // Found end of string
|
||||
}
|
||||
else
|
||||
{
|
||||
if(c == ' ')
|
||||
{
|
||||
// Found end of word
|
||||
is.unget();
|
||||
break;
|
||||
}
|
||||
else
|
||||
{
|
||||
tmp_os << c;
|
||||
}
|
||||
}
|
||||
expect_initial_quote = false;
|
||||
}
|
||||
if(is_json)
|
||||
{
|
||||
std::istringstream tmp_is(tmp_os.str(), std::ios::binary);
|
||||
return deSerializeJsonString(tmp_is);
|
||||
}
|
||||
else
|
||||
return tmp_os.str();
|
||||
}
|
||||
|
||||
|
||||
ItemStack::ItemStack(std::string name_, u16 count_,
|
||||
u16 wear_, std::string metadata_,
|
||||
IItemDefManager *itemdef)
|
||||
ItemStack::ItemStack(const std::string &name_, u16 count_,
|
||||
u16 wear_, IItemDefManager *itemdef)
|
||||
{
|
||||
name = itemdef->getAlias(name_);
|
||||
count = count_;
|
||||
wear = wear_;
|
||||
metadata = metadata_;
|
||||
|
||||
if(name.empty() || count == 0)
|
||||
if (name.empty() || count == 0)
|
||||
clear();
|
||||
else if(itemdef->get(name).type == ITEM_TOOL)
|
||||
else if (itemdef->get(name).type == ITEM_TOOL)
|
||||
count = 1;
|
||||
}
|
||||
|
||||
|
@ -137,7 +71,7 @@ void ItemStack::serialize(std::ostream &os) const
|
|||
parts = 2;
|
||||
if(wear != 0)
|
||||
parts = 3;
|
||||
if(metadata != "")
|
||||
if (!metadata.empty())
|
||||
parts = 4;
|
||||
|
||||
os<<serializeJsonStringIfNeeded(name);
|
||||
|
@ -145,8 +79,10 @@ void ItemStack::serialize(std::ostream &os) const
|
|||
os<<" "<<count;
|
||||
if(parts >= 3)
|
||||
os<<" "<<wear;
|
||||
if(parts >= 4)
|
||||
os<<" "<<serializeJsonStringIfNeeded(metadata);
|
||||
if (parts >= 4) {
|
||||
os << " ";
|
||||
metadata.serialize(os);
|
||||
}
|
||||
}
|
||||
|
||||
void ItemStack::deSerialize(std::istream &is, IItemDefManager *itemdef)
|
||||
|
@ -289,7 +225,7 @@ void ItemStack::deSerialize(std::istream &is, IItemDefManager *itemdef)
|
|||
wear = stoi(wear_str);
|
||||
|
||||
// Read metadata
|
||||
metadata = deSerializeJsonStringIfNeeded(is);
|
||||
metadata.deSerialize(is);
|
||||
|
||||
// In case fields are added after metadata, skip space here:
|
||||
//std::getline(is, tmp, ' ');
|
||||
|
@ -335,7 +271,7 @@ ItemStack ItemStack::addItem(const ItemStack &newitem_,
|
|||
*this = newitem;
|
||||
newitem.clear();
|
||||
}
|
||||
// If item name or metadata differs, bail out
|
||||
// If item name or metadata differs, bail out
|
||||
else if (name != newitem.name
|
||||
|| metadata != newitem.metadata)
|
||||
{
|
||||
|
@ -375,7 +311,7 @@ bool ItemStack::itemFits(const ItemStack &newitem_,
|
|||
{
|
||||
newitem.clear();
|
||||
}
|
||||
// If item name or metadata differs, bail out
|
||||
// If item name or metadata differs, bail out
|
||||
else if (name != newitem.name
|
||||
|| metadata != newitem.metadata)
|
||||
{
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue