Add ItemStack key-value meta storage

This commit is contained in:
rubenwardy 2017-01-31 19:49:01 +00:00
parent c2e7b1f579
commit f2aa2c6a98
21 changed files with 459 additions and 123 deletions

View File

@ -164,6 +164,7 @@ LOCAL_SRC_FILES := \
jni/src/inventory.cpp \ jni/src/inventory.cpp \
jni/src/inventorymanager.cpp \ jni/src/inventorymanager.cpp \
jni/src/itemdef.cpp \ jni/src/itemdef.cpp \
jni/src/itemstackmetadata.cpp \
jni/src/keycode.cpp \ jni/src/keycode.cpp \
jni/src/light.cpp \ jni/src/light.cpp \
jni/src/localplayer.cpp \ jni/src/localplayer.cpp \
@ -305,6 +306,7 @@ LOCAL_SRC_FILES += \
jni/src/script/lua_api/l_env.cpp \ jni/src/script/lua_api/l_env.cpp \
jni/src/script/lua_api/l_inventory.cpp \ jni/src/script/lua_api/l_inventory.cpp \
jni/src/script/lua_api/l_item.cpp \ jni/src/script/lua_api/l_item.cpp \
jni/src/script/lua_api/l_itemstackmeta.cpp\
jni/src/script/lua_api/l_mainmenu.cpp \ jni/src/script/lua_api/l_mainmenu.cpp \
jni/src/script/lua_api/l_mapgen.cpp \ jni/src/script/lua_api/l_mapgen.cpp \
jni/src/script/lua_api/l_metadata.cpp \ jni/src/script/lua_api/l_metadata.cpp \

View File

@ -1411,7 +1411,7 @@ the entity itself.
* `direction` is a unit vector, pointing from the source of the punch to * `direction` is a unit vector, pointing from the source of the punch to
the punched object. the punched object.
* `damage` damage that will be done to entity * `damage` damage that will be done to entity
Return value of this function will determin if damage is done by this function Return value of this function will determin if damage is done by this function
(retval true) or shall be done by engine (retval false) (retval true) or shall be done by engine (retval false)
To punch an entity/object in Lua, call: To punch an entity/object in Lua, call:
@ -1427,9 +1427,9 @@ Node Metadata
------------- -------------
The instance of a node in the world normally only contains the three values The instance of a node in the world normally only contains the three values
mentioned in "Nodes". However, it is possible to insert extra data into a mentioned in "Nodes". However, it is possible to insert extra data into a
node. It is called "node metadata"; See "`NodeMetaRef`". node. It is called "node metadata"; See `NodeMetaRef`.
Metadata contains two things: Node metadata contains two things:
* A key-value store * A key-value store
* An inventory * An inventory
@ -1467,6 +1467,18 @@ Example stuff:
} }
}) })
Item Metadata
-------------
Item stacks can store metadata too. See `ItemStackMetaRef`.
Item metadata only contains a key-value store.
Example stuff:
local meta = stack:get_meta()
meta:set_string("key", "value")
print(dump(meta:to_table()))
Formspec Formspec
-------- --------
Formspec defines a menu. Currently not much else than inventories are Formspec defines a menu. Currently not much else than inventories are
@ -2774,9 +2786,8 @@ These functions return the leftover itemstack.
Class reference Class reference
--------------- ---------------
### `NodeMetaRef` ### `MetaDataRef`
Node metadata: reference extra data and functionality stored in a node. See `NodeMetaRef` and `ItemStackMetaRef`.
Can be gotten via `minetest.get_meta(pos)`.
#### Methods #### Methods
* `set_string(name, value)` * `set_string(name, value)`
@ -2785,13 +2796,29 @@ Can be gotten via `minetest.get_meta(pos)`.
* `get_int(name)` * `get_int(name)`
* `set_float(name, value)` * `set_float(name, value)`
* `get_float(name)` * `get_float(name)`
* `get_inventory()`: returns `InvRef` * `to_table()`: returns `nil` or a table with keys:
* `to_table()`: returns `nil` or `{fields = {...}, inventory = {list1 = {}, ...}}` * `fields`: key-value storage
* `inventory`: `{list1 = {}, ...}}` (NodeMetaRef only)
* `from_table(nil or {})` * `from_table(nil or {})`
* Any non-table value will clear the metadata * Any non-table value will clear the metadata
* See "Node Metadata" * See "Node Metadata" for an example
* returns `true` on success * returns `true` on success
### `NodeMetaRef`
Node metadata: reference extra data and functionality stored in a node.
Can be gotten via `minetest.get_meta(pos)`.
#### Methods
* All methods in MetaDataRef
* `get_inventory()`: returns `InvRef`
### `ItemStackMetaRef`
ItemStack metadata: reference extra data and functionality stored in a stack.
Can be gotten via `item:get_meta()`.
#### Methods
* All methods in MetaDataRef
### `NodeTimerRef` ### `NodeTimerRef`
Node Timers: a high resolution persistent per-node timer. Node Timers: a high resolution persistent per-node timer.
Can be gotten via `minetest.get_node_timer(pos)`. Can be gotten via `minetest.get_node_timer(pos)`.

View File

@ -415,6 +415,7 @@ set(common_SRCS
inventory.cpp inventory.cpp
inventorymanager.cpp inventorymanager.cpp
itemdef.cpp itemdef.cpp
itemstackmetadata.cpp
light.cpp light.cpp
log.cpp log.cpp
map.cpp map.cpp

View File

@ -938,7 +938,7 @@ void GenericCAO::addToScene(scene::ISceneManager *smgr,
if(m_prop.textures.size() >= 1){ if(m_prop.textures.size() >= 1){
infostream<<"textures[0]: "<<m_prop.textures[0]<<std::endl; infostream<<"textures[0]: "<<m_prop.textures[0]<<std::endl;
IItemDefManager *idef = m_client->idef(); IItemDefManager *idef = m_client->idef();
ItemStack item(m_prop.textures[0], 1, 0, "", idef); ItemStack item(m_prop.textures[0], 1, 0, idef);
m_wield_meshnode = new WieldMeshSceneNode( m_wield_meshnode = new WieldMeshSceneNode(
smgr->getRootSceneNode(), smgr, -1); smgr->getRootSceneNode(), smgr, -1);

View File

@ -139,7 +139,7 @@ static std::vector<ItemStack> craftGetItems(
for (std::vector<std::string>::size_type i = 0; for (std::vector<std::string>::size_type i = 0;
i < items.size(); i++) { i < items.size(); i++) {
result.push_back(ItemStack(std::string(items[i]), (u16)1, result.push_back(ItemStack(std::string(items[i]), (u16)1,
(u16)0, "", gamedef->getItemDefManager())); (u16)0, gamedef->getItemDefManager()));
} }
return result; return result;
} }
@ -1126,4 +1126,3 @@ IWritableCraftDefManager* createCraftDefManager()
{ {
return new CCraftDefManager(); return new CCraftDefManager();
} }

View File

@ -45,82 +45,16 @@ static content_t content_translate_from_19_to_internal(content_t c_from)
return c_from; return c_from;
} }
// If the string contains spaces, quotes or control characters, encodes as JSON. ItemStack::ItemStack(const std::string &name_, u16 count_,
// Else returns the string unmodified. u16 wear_, IItemDefManager *itemdef)
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)
{ {
name = itemdef->getAlias(name_); name = itemdef->getAlias(name_);
count = count_; count = count_;
wear = wear_; wear = wear_;
metadata = metadata_;
if(name.empty() || count == 0) if (name.empty() || count == 0)
clear(); clear();
else if(itemdef->get(name).type == ITEM_TOOL) else if (itemdef->get(name).type == ITEM_TOOL)
count = 1; count = 1;
} }
@ -137,7 +71,7 @@ void ItemStack::serialize(std::ostream &os) const
parts = 2; parts = 2;
if(wear != 0) if(wear != 0)
parts = 3; parts = 3;
if(metadata != "") if (!metadata.empty())
parts = 4; parts = 4;
os<<serializeJsonStringIfNeeded(name); os<<serializeJsonStringIfNeeded(name);
@ -145,8 +79,10 @@ void ItemStack::serialize(std::ostream &os) const
os<<" "<<count; os<<" "<<count;
if(parts >= 3) if(parts >= 3)
os<<" "<<wear; os<<" "<<wear;
if(parts >= 4) if (parts >= 4) {
os<<" "<<serializeJsonStringIfNeeded(metadata); os << " ";
metadata.serialize(os);
}
} }
void ItemStack::deSerialize(std::istream &is, IItemDefManager *itemdef) void ItemStack::deSerialize(std::istream &is, IItemDefManager *itemdef)
@ -289,7 +225,7 @@ void ItemStack::deSerialize(std::istream &is, IItemDefManager *itemdef)
wear = stoi(wear_str); wear = stoi(wear_str);
// Read metadata // Read metadata
metadata = deSerializeJsonStringIfNeeded(is); metadata.deSerialize(is);
// In case fields are added after metadata, skip space here: // In case fields are added after metadata, skip space here:
//std::getline(is, tmp, ' '); //std::getline(is, tmp, ' ');
@ -335,7 +271,7 @@ ItemStack ItemStack::addItem(const ItemStack &newitem_,
*this = newitem; *this = newitem;
newitem.clear(); newitem.clear();
} }
// If item name or metadata differs, bail out // If item name or metadata differs, bail out
else if (name != newitem.name else if (name != newitem.name
|| metadata != newitem.metadata) || metadata != newitem.metadata)
{ {
@ -375,7 +311,7 @@ bool ItemStack::itemFits(const ItemStack &newitem_,
{ {
newitem.clear(); newitem.clear();
} }
// If item name or metadata differs, bail out // If item name or metadata differs, bail out
else if (name != newitem.name else if (name != newitem.name
|| metadata != newitem.metadata) || metadata != newitem.metadata)
{ {

View File

@ -23,6 +23,7 @@ with this program; if not, write to the Free Software Foundation, Inc.,
#include "debug.h" #include "debug.h"
#include "itemdef.h" #include "itemdef.h"
#include "irrlichttypes.h" #include "irrlichttypes.h"
#include "itemstackmetadata.h"
#include <istream> #include <istream>
#include <ostream> #include <ostream>
#include <string> #include <string>
@ -32,10 +33,10 @@ struct ToolCapabilities;
struct ItemStack struct ItemStack
{ {
ItemStack(): name(""), count(0), wear(0), metadata("") {} ItemStack(): name(""), count(0), wear(0) {}
ItemStack(std::string name_, u16 count_, ItemStack(const std::string &name_, u16 count_,
u16 wear, std::string metadata_, u16 wear, IItemDefManager *itemdef);
IItemDefManager *itemdef);
~ItemStack() {} ~ItemStack() {}
// Serialization // Serialization
@ -61,7 +62,7 @@ struct ItemStack
name = ""; name = "";
count = 0; count = 0;
wear = 0; wear = 0;
metadata = ""; metadata.clear();
} }
void add(u16 n) void add(u16 n)
@ -166,7 +167,7 @@ struct ItemStack
std::string name; std::string name;
u16 count; u16 count;
u16 wear; u16 wear;
std::string metadata; ItemStackMetadata metadata;
}; };
class InventoryList class InventoryList
@ -313,4 +314,3 @@ private:
}; };
#endif #endif

43
src/itemstackmetadata.cpp Normal file
View File

@ -0,0 +1,43 @@
#include "itemstackmetadata.h"
#include "util/serialize.h"
#include "util/strfnd.h"
#define DESERIALIZE_START '\x01'
#define DESERIALIZE_KV_DELIM '\x02'
#define DESERIALIZE_PAIR_DELIM '\x03'
#define DESERIALIZE_START_STR "\x01"
#define DESERIALIZE_KV_DELIM_STR "\x02"
#define DESERIALIZE_PAIR_DELIM_STR "\x03"
void ItemStackMetadata::serialize(std::ostream &os) const
{
std::ostringstream os2;
os2 << DESERIALIZE_START;
for (StringMap::const_iterator
it = m_stringvars.begin();
it != m_stringvars.end(); ++it) {
os2 << it->first << DESERIALIZE_KV_DELIM
<< it->second << DESERIALIZE_PAIR_DELIM;
}
os << serializeJsonStringIfNeeded(os2.str());
}
void ItemStackMetadata::deSerialize(std::istream &is)
{
std::string in = deSerializeJsonStringIfNeeded(is);
m_stringvars.clear();
if (!in.empty() && in[0] == DESERIALIZE_START) {
Strfnd fnd(in);
fnd.to(1);
while (!fnd.at_end()) {
std::string name = fnd.next(DESERIALIZE_KV_DELIM_STR);
std::string var = fnd.next(DESERIALIZE_PAIR_DELIM_STR);
m_stringvars[name] = var;
}
} else {
// BACKWARDS COMPATIBILITY
m_stringvars[""] = in;
}
}

35
src/itemstackmetadata.h Normal file
View File

@ -0,0 +1,35 @@
/*
Minetest
Copyright (C) 2010-2013 rubenwardy <rubenwardy@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.
*/
#ifndef ITEMSTACKMETADATA_HEADER
#define ITEMSTACKMETADATA_HEADER
#include "metadata.h"
class Inventory;
class IItemDefManager;
class ItemStackMetadata : public Metadata
{
public:
void serialize(std::ostream &os) const;
void deSerialize(std::istream &is);
};
#endif

View File

@ -22,6 +22,8 @@ with this program; if not, write to the Free Software Foundation, Inc.,
#include "gamedef.h" #include "gamedef.h"
#include "log.h" #include "log.h"
#include <sstream> #include <sstream>
#include "constants.h" // MAP_BLOCKSIZE
#include <sstream>
/* /*
Metadata Metadata

View File

@ -25,35 +25,37 @@ with this program; if not, write to the Free Software Foundation, Inc.,
#include <vector> #include <vector>
#include "util/string.h" #include "util/string.h"
/*
NodeMetadata stores arbitary amounts of data for special blocks.
Used for furnaces, chests and signs.
There are two interaction methods: inventory menu and text input.
Only one can be used for a single metadata, thus only inventory OR
text input should exist in a metadata.
*/
class Inventory;
class IItemDefManager;
class Metadata class Metadata
{ {
public: public:
void clear(); virtual ~Metadata() {}
bool empty() const;
// Generic key/value store virtual void clear();
virtual bool empty() const;
bool operator==(const Metadata &other) const;
inline bool operator!=(const Metadata &other) const
{
return !(*this == other);
}
//
// Key-value related
//
size_t size() const;
bool contains(const std::string &name) const;
const std::string &getString(const std::string &name, u16 recursion = 0) const; const std::string &getString(const std::string &name, u16 recursion = 0) const;
void setString(const std::string &name, const std::string &var); void setString(const std::string &name, const std::string &var);
// Support variable names in values
const std::string &resolveString(const std::string &str, u16 recursion = 0) const;
const StringMap &getStrings() const const StringMap &getStrings() const
{ {
return m_stringvars; return m_stringvars;
} }
private: // Add support for variable names in values
const std::string &resolveString(const std::string &str, u16 recursion = 0) const;
protected:
StringMap m_stringvars; StringMap m_stringvars;
}; };
#endif #endif

View File

@ -824,11 +824,32 @@ ItemStack read_item(lua_State* L, int index,Server* srv)
std::string name = getstringfield_default(L, index, "name", ""); std::string name = getstringfield_default(L, index, "name", "");
int count = getintfield_default(L, index, "count", 1); int count = getintfield_default(L, index, "count", 1);
int wear = getintfield_default(L, index, "wear", 0); int wear = getintfield_default(L, index, "wear", 0);
std::string metadata = getstringfield_default(L, index, "metadata", "");
return ItemStack(name, count, wear, metadata, idef); ItemStack istack(name, count, wear, idef);
}
else lua_getfield(L, index, "metadata");
{
// Support old metadata format by checking type
int fieldstable = lua_gettop(L);
if (lua_istable(L, fieldstable)) {
lua_pushnil(L);
while (lua_next(L, fieldstable) != 0) {
// key at index -2 and value at index -1
std::string key = lua_tostring(L, -2);
size_t value_len;
const char *value_cs = lua_tolstring(L, -1, &value_len);
std::string value(value_cs, value_len);
istack.metadata.setString(name, value);
lua_pop(L, 1); // removes value, keeps key for next iteration
}
} else {
// BACKWARDS COMPATIBLITY
std::string value = getstringfield_default(L, index, "metadata", "");
istack.metadata.setString("", value);
}
return istack;
} else {
throw LuaError("Expecting itemstack, itemstring, table or nil"); throw LuaError("Expecting itemstack, itemstring, table or nil");
} }
} }

View File

@ -5,6 +5,7 @@ set(common_SCRIPT_LUA_API_SRCS
${CMAKE_CURRENT_SOURCE_DIR}/l_env.cpp ${CMAKE_CURRENT_SOURCE_DIR}/l_env.cpp
${CMAKE_CURRENT_SOURCE_DIR}/l_inventory.cpp ${CMAKE_CURRENT_SOURCE_DIR}/l_inventory.cpp
${CMAKE_CURRENT_SOURCE_DIR}/l_item.cpp ${CMAKE_CURRENT_SOURCE_DIR}/l_item.cpp
${CMAKE_CURRENT_SOURCE_DIR}/l_itemstackmeta.cpp
${CMAKE_CURRENT_SOURCE_DIR}/l_mapgen.cpp ${CMAKE_CURRENT_SOURCE_DIR}/l_mapgen.cpp
${CMAKE_CURRENT_SOURCE_DIR}/l_metadata.cpp ${CMAKE_CURRENT_SOURCE_DIR}/l_metadata.cpp
${CMAKE_CURRENT_SOURCE_DIR}/l_nodemeta.cpp ${CMAKE_CURRENT_SOURCE_DIR}/l_nodemeta.cpp

View File

@ -284,7 +284,7 @@ int ModApiEnvMod::l_place_node(lua_State *L)
return 1; return 1;
} }
// Create item to place // Create item to place
ItemStack item(ndef->get(n).name, 1, 0, "", idef); ItemStack item(ndef->get(n).name, 1, 0, idef);
// Make pointed position // Make pointed position
PointedThing pointed; PointedThing pointed;
pointed.type = POINTEDTHING_NODE; pointed.type = POINTEDTHING_NODE;

View File

@ -18,6 +18,7 @@ with this program; if not, write to the Free Software Foundation, Inc.,
*/ */
#include "lua_api/l_item.h" #include "lua_api/l_item.h"
#include "lua_api/l_itemstackmeta.h"
#include "lua_api/l_internal.h" #include "lua_api/l_internal.h"
#include "common/c_converter.h" #include "common/c_converter.h"
#include "common/c_content.h" #include "common/c_content.h"
@ -137,16 +138,28 @@ int LuaItemStack::l_set_wear(lua_State *L)
return 1; return 1;
} }
// get_meta(self) -> string
int LuaItemStack::l_get_meta(lua_State *L)
{
NO_MAP_LOCK_REQUIRED;
LuaItemStack *o = checkobject(L, 1);
ItemStackMetaRef::create(L, &o->m_stack);
return 1;
}
// DEPRECATED
// get_metadata(self) -> string // get_metadata(self) -> string
int LuaItemStack::l_get_metadata(lua_State *L) int LuaItemStack::l_get_metadata(lua_State *L)
{ {
NO_MAP_LOCK_REQUIRED; NO_MAP_LOCK_REQUIRED;
LuaItemStack *o = checkobject(L, 1); LuaItemStack *o = checkobject(L, 1);
ItemStack &item = o->m_stack; ItemStack &item = o->m_stack;
lua_pushlstring(L, item.metadata.c_str(), item.metadata.size()); const std::string &value = item.metadata.getString("");
lua_pushlstring(L, value.c_str(), value.size());
return 1; return 1;
} }
// DEPRECATED
// set_metadata(self, string) // set_metadata(self, string)
int LuaItemStack::l_set_metadata(lua_State *L) int LuaItemStack::l_set_metadata(lua_State *L)
{ {
@ -156,7 +169,7 @@ int LuaItemStack::l_set_metadata(lua_State *L)
size_t len = 0; size_t len = 0;
const char *ptr = luaL_checklstring(L, 2, &len); const char *ptr = luaL_checklstring(L, 2, &len);
item.metadata.assign(ptr, len); item.metadata.setString("", std::string(ptr, len));
lua_pushboolean(L, true); lua_pushboolean(L, true);
return 1; return 1;
@ -211,8 +224,24 @@ int LuaItemStack::l_to_table(lua_State *L)
lua_setfield(L, -2, "count"); lua_setfield(L, -2, "count");
lua_pushinteger(L, item.wear); lua_pushinteger(L, item.wear);
lua_setfield(L, -2, "wear"); lua_setfield(L, -2, "wear");
lua_pushlstring(L, item.metadata.c_str(), item.metadata.size());
lua_setfield(L, -2, "metadata"); if (item.metadata.size() == 1 && item.metadata.contains("")) {
const std::string &value = item.metadata.getString("");
lua_pushlstring(L, value.c_str(), value.size());
lua_setfield(L, -2, "metadata");
} else {
lua_newtable(L);
const StringMap &fields = item.metadata.getStrings();
for (StringMap::const_iterator it = fields.begin();
it != fields.end(); ++it) {
const std::string &name = it->first;
const std::string &value = it->second;
lua_pushlstring(L, name.c_str(), name.size());
lua_pushlstring(L, value.c_str(), value.size());
lua_settable(L, -3);
}
lua_setfield(L, -2, "metadata");
}
} }
return 1; return 1;
} }
@ -443,6 +472,7 @@ const luaL_reg LuaItemStack::methods[] = {
luamethod(LuaItemStack, set_count), luamethod(LuaItemStack, set_count),
luamethod(LuaItemStack, get_wear), luamethod(LuaItemStack, get_wear),
luamethod(LuaItemStack, set_wear), luamethod(LuaItemStack, set_wear),
luamethod(LuaItemStack, get_meta),
luamethod(LuaItemStack, get_metadata), luamethod(LuaItemStack, get_metadata),
luamethod(LuaItemStack, set_metadata), luamethod(LuaItemStack, set_metadata),
luamethod(LuaItemStack, clear), luamethod(LuaItemStack, clear),

View File

@ -56,9 +56,14 @@ private:
// set_wear(self, number) // set_wear(self, number)
static int l_set_wear(lua_State *L); static int l_set_wear(lua_State *L);
// get_meta(self) -> string
static int l_get_meta(lua_State *L);
// DEPRECATED
// get_metadata(self) -> string // get_metadata(self) -> string
static int l_get_metadata(lua_State *L); static int l_get_metadata(lua_State *L);
// DEPRECATED
// set_metadata(self, string) // set_metadata(self, string)
static int l_set_metadata(lua_State *L); static int l_set_metadata(lua_State *L);

View File

@ -0,0 +1,115 @@
/*
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 "lua_api/l_itemstackmeta.h"
#include "lua_api/l_internal.h"
#include "common/c_content.h"
/*
NodeMetaRef
*/
ItemStackMetaRef* ItemStackMetaRef::checkobject(lua_State *L, int narg)
{
luaL_checktype(L, narg, LUA_TUSERDATA);
void *ud = luaL_checkudata(L, narg, className);
if (!ud)
luaL_typerror(L, narg, className);
return *(ItemStackMetaRef**)ud; // unbox pointer
}
Metadata* ItemStackMetaRef::getmeta(bool auto_create)
{
return &istack->metadata;
}
void ItemStackMetaRef::clearMeta()
{
istack->metadata.clear();
}
void ItemStackMetaRef::reportMetadataChange()
{
// TODO
}
// Exported functions
// garbage collector
int ItemStackMetaRef::gc_object(lua_State *L) {
ItemStackMetaRef *o = *(ItemStackMetaRef **)(lua_touserdata(L, 1));
delete o;
return 0;
}
// Creates an NodeMetaRef and leaves it on top of stack
// Not callable from Lua; all references are created on the C side.
void ItemStackMetaRef::create(lua_State *L, ItemStack *istack)
{
ItemStackMetaRef *o = new ItemStackMetaRef(istack);
//infostream<<"NodeMetaRef::create: o="<<o<<std::endl;
*(void **)(lua_newuserdata(L, sizeof(void *))) = o;
luaL_getmetatable(L, className);
lua_setmetatable(L, -2);
}
void ItemStackMetaRef::Register(lua_State *L)
{
lua_newtable(L);
int methodtable = lua_gettop(L);
luaL_newmetatable(L, className);
int metatable = lua_gettop(L);
lua_pushliteral(L, "__metatable");
lua_pushvalue(L, methodtable);
lua_settable(L, metatable); // hide metatable from Lua getmetatable()
lua_pushliteral(L, "metadata_class");
lua_pushlstring(L, className, strlen(className));
lua_settable(L, metatable);
lua_pushliteral(L, "__index");
lua_pushvalue(L, methodtable);
lua_settable(L, metatable);
lua_pushliteral(L, "__gc");
lua_pushcfunction(L, gc_object);
lua_settable(L, metatable);
lua_pop(L, 1); // drop metatable
luaL_openlib(L, 0, methods, 0); // fill methodtable
lua_pop(L, 1); // drop methodtable
// Cannot be created from Lua
//lua_register(L, className, create_object);
}
const char ItemStackMetaRef::className[] = "ItemStackMetaRef";
const luaL_reg ItemStackMetaRef::methods[] = {
luamethod(MetaDataRef, get_string),
luamethod(MetaDataRef, set_string),
luamethod(MetaDataRef, get_int),
luamethod(MetaDataRef, set_int),
luamethod(MetaDataRef, get_float),
luamethod(MetaDataRef, set_float),
luamethod(MetaDataRef, to_table),
luamethod(MetaDataRef, from_table),
{0,0}
};

View File

@ -0,0 +1,59 @@
/*
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.
*/
#ifndef L_ITEMSTACKMETA_H_
#define L_ITEMSTACKMETA_H_
#include "lua_api/l_base.h"
#include "lua_api/l_metadata.h"
#include "irrlichttypes_bloated.h"
#include "inventory.h"
class ItemStackMetaRef : public MetaDataRef
{
private:
ItemStack *istack;
static const char className[];
static const luaL_reg methods[];
static ItemStackMetaRef *checkobject(lua_State *L, int narg);
virtual Metadata* getmeta(bool auto_create);
virtual void clearMeta();
virtual void reportMetadataChange();
// Exported functions
// garbage collector
static int gc_object(lua_State *L);
public:
ItemStackMetaRef(ItemStack *istack): istack(istack) {}
~ItemStackMetaRef() {}
// Creates an ItemStackMetaRef and leaves it on top of stack
// Not callable from Lua; all references are created on the C side.
static void create(lua_State *L, ItemStack *istack);
static void Register(lua_State *L);
};
#endif

View File

@ -28,6 +28,7 @@ with this program; if not, write to the Free Software Foundation, Inc.,
#include "lua_api/l_env.h" #include "lua_api/l_env.h"
#include "lua_api/l_inventory.h" #include "lua_api/l_inventory.h"
#include "lua_api/l_item.h" #include "lua_api/l_item.h"
#include "lua_api/l_itemstackmeta.h"
#include "lua_api/l_mapgen.h" #include "lua_api/l_mapgen.h"
#include "lua_api/l_nodemeta.h" #include "lua_api/l_nodemeta.h"
#include "lua_api/l_nodetimer.h" #include "lua_api/l_nodetimer.h"
@ -94,6 +95,7 @@ void GameScripting::InitializeModApi(lua_State *L, int top)
// Register reference classes (userdata) // Register reference classes (userdata)
InvRef::Register(L); InvRef::Register(L);
ItemStackMetaRef::Register(L);
LuaAreaStore::Register(L); LuaAreaStore::Register(L);
LuaItemStack::Register(L); LuaItemStack::Register(L);
LuaPerlinNoise::Register(L); LuaPerlinNoise::Register(L);

View File

@ -354,6 +354,55 @@ std::string deSerializeJsonString(std::istream &is)
return os.str(); return os.str();
} }
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;
}
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();
}
//// ////
//// String/Struct conversions //// String/Struct conversions
//// ////

View File

@ -405,6 +405,13 @@ std::string serializeJsonString(const std::string &plain);
// Reads a string encoded in JSON format // Reads a string encoded in JSON format
std::string deSerializeJsonString(std::istream &is); std::string deSerializeJsonString(std::istream &is);
// If the string contains spaces, quotes or control characters, encodes as JSON.
// Else returns the string unmodified.
std::string serializeJsonStringIfNeeded(const std::string &s);
// Parses a string serialized by serializeJsonStringIfNeeded.
std::string deSerializeJsonStringIfNeeded(std::istream &is);
// Creates a string consisting of the hexadecimal representation of `data` // Creates a string consisting of the hexadecimal representation of `data`
std::string serializeHexString(const std::string &data, bool insert_spaces=false); std::string serializeHexString(const std::string &data, bool insert_spaces=false);