X-Git-Url: https://git.lizzy.rs/?a=blobdiff_plain;f=src%2Fnodemetadata.cpp;h=b5052c3b8754122460330bd87f17b10661155973;hb=a0edf2849aeebc1fb37ead1177caf471ab4a19d4;hp=0b63d7779c6fe56d2946c50d393aadb86e68f2e7;hpb=c4315a7afafcc9adff945b5e63800340f6c2c008;p=minetest.git diff --git a/src/nodemetadata.cpp b/src/nodemetadata.cpp index 0b63d7779..b5052c3b8 100644 --- a/src/nodemetadata.cpp +++ b/src/nodemetadata.cpp @@ -1,66 +1,75 @@ /* -Minetest-c55 -Copyright (C) 2010-2011 celeron55, Perttu Ahola +Minetest +Copyright (C) 2010-2013 celeron55, Perttu Ahola This program is free software; you can redistribute it and/or modify -it under the terms of the GNU General Public License as published by -the Free Software Foundation; either version 2 of the License, or +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 General Public License for more details. +GNU Lesser General Public License for more details. -You should have received a copy of the GNU General Public License along +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 "nodemetadata.h" -#include "utility.h" #include "exceptions.h" #include "gamedef.h" #include "inventory.h" -#include #include "log.h" +#include "util/serialize.h" +#include "util/basic_macros.h" +#include "constants.h" // MAP_BLOCKSIZE +#include /* NodeMetadata */ -NodeMetadata::NodeMetadata(IGameDef *gamedef): - m_stringvars(), - m_inventory(new Inventory(gamedef->idef())) -{ -} +NodeMetadata::NodeMetadata(IItemDefManager *item_def_mgr): + m_inventory(new Inventory(item_def_mgr)) +{} NodeMetadata::~NodeMetadata() { delete m_inventory; } -void NodeMetadata::serialize(std::ostream &os) const +void NodeMetadata::serialize(std::ostream &os, u8 version, bool disk) const { - int num_vars = m_stringvars.size(); + int num_vars = disk ? m_stringvars.size() : countNonPrivate(); writeU32(os, num_vars); - for(std::map::const_iterator - i = m_stringvars.begin(); i != m_stringvars.end(); i++){ - os<first); - os<second); + for (const auto &sv : m_stringvars) { + bool priv = isPrivate(sv.first); + if (!disk && priv) + continue; + + os << serializeString16(sv.first); + os << serializeString32(sv.second); + if (version >= 2) + writeU8(os, (priv) ? 1 : 0); } m_inventory->serialize(os); } -void NodeMetadata::deSerialize(std::istream &is) +void NodeMetadata::deSerialize(std::istream &is, u8 version) { - m_stringvars.clear(); + clear(); int num_vars = readU32(is); for(int i=0; i= 2) { + if (readU8(is) == 1) + markPrivate(name, true); + } } m_inventory->deSerialize(is); @@ -68,85 +77,123 @@ void NodeMetadata::deSerialize(std::istream &is) void NodeMetadata::clear() { - m_stringvars.clear(); + Metadata::clear(); + m_privatevars.clear(); m_inventory->clear(); } +bool NodeMetadata::empty() const +{ + return Metadata::empty() && m_inventory->getLists().empty(); +} + + +void NodeMetadata::markPrivate(const std::string &name, bool set) +{ + if (set) + m_privatevars.insert(name); + else + m_privatevars.erase(name); +} + +int NodeMetadata::countNonPrivate() const +{ + // m_privatevars can contain names not actually present + // DON'T: return m_stringvars.size() - m_privatevars.size(); + int n = 0; + for (const auto &sv : m_stringvars) { + if (!isPrivate(sv.first)) + n++; + } + return n; +} + /* NodeMetadataList */ -void NodeMetadataList::serialize(std::ostream &os) const +void NodeMetadataList::serialize(std::ostream &os, u8 blockver, bool disk, + bool absolute_pos, bool include_empty) const { /* Version 0 is a placeholder for "nothing to see here; go away." */ - if(m_data.size() == 0){ + u16 count = include_empty ? m_data.size() : countNonEmpty(); + if (count == 0) { writeU8(os, 0); // version return; } - writeU8(os, 1); // version - - u16 count = m_data.size(); + u8 version = (blockver > 27) ? 2 : 1; + writeU8(os, version); writeU16(os, count); - for(std::map::const_iterator + for (NodeMetadataMap::const_iterator i = m_data.begin(); - i != m_data.end(); i++) - { + i != m_data.end(); ++i) { v3s16 p = i->first; NodeMetadata *data = i->second; + if (!include_empty && data->empty()) + continue; - u16 p16 = p.Z*MAP_BLOCKSIZE*MAP_BLOCKSIZE + p.Y*MAP_BLOCKSIZE + p.X; - writeU16(os, p16); - - data->serialize(os); + if (absolute_pos) { + writeS16(os, p.X); + writeS16(os, p.Y); + writeS16(os, p.Z); + } else { + // Serialize positions within a mapblock + u16 p16 = (p.Z * MAP_BLOCKSIZE + p.Y) * MAP_BLOCKSIZE + p.X; + writeU16(os, p16); + } + data->serialize(os, version, disk); } } -void NodeMetadataList::deSerialize(std::istream &is, IGameDef *gamedef) +void NodeMetadataList::deSerialize(std::istream &is, + IItemDefManager *item_def_mgr, bool absolute_pos) { - m_data.clear(); + clear(); u8 version = readU8(is); - - if(version == 0){ + + if (version == 0) { // Nothing return; } - if(version != 1){ - infostream<<__FUNCTION_NAME<<": version "< 2) { + std::string err_str = std::string(FUNCTION_NAME) + + ": version " + itos(version) + " not supported"; + infostream << err_str << std::endl; + throw SerializationError(err_str); } u16 count = readU16(is); - for(u16 i=0; ideSerialize(is); + NodeMetadata *data = new NodeMetadata(item_def_mgr); + data->deSerialize(is, version); m_data[p] = data; } } @@ -156,20 +203,30 @@ NodeMetadataList::~NodeMetadataList() clear(); } -NodeMetadata* NodeMetadataList::get(v3s16 p) +std::vector NodeMetadataList::getAllKeys() { - std::map::const_iterator n = m_data.find(p); - if(n == m_data.end()) - return NULL; + std::vector keys; + keys.reserve(m_data.size()); + for (const auto &it : m_data) + keys.push_back(it.first); + + return keys; +} + +NodeMetadata *NodeMetadataList::get(v3s16 p) +{ + NodeMetadataMap::const_iterator n = m_data.find(p); + if (n == m_data.end()) + return nullptr; return n->second; } void NodeMetadataList::remove(v3s16 p) { NodeMetadata *olddata = get(p); - if(olddata) - { - delete olddata; + if (olddata) { + if (m_is_metadata_owner) + delete olddata; m_data.erase(p); } } @@ -177,16 +234,25 @@ void NodeMetadataList::remove(v3s16 p) void NodeMetadataList::set(v3s16 p, NodeMetadata *d) { remove(p); - m_data.insert(std::make_pair(p, d)); + m_data.emplace(p, d); } void NodeMetadataList::clear() { - for(std::map::iterator - i = m_data.begin(); - i != m_data.end(); i++) - { - delete i->second; + if (m_is_metadata_owner) { + NodeMetadataMap::const_iterator it; + for (it = m_data.begin(); it != m_data.end(); ++it) + delete it->second; } m_data.clear(); } + +int NodeMetadataList::countNonEmpty() const +{ + int n = 0; + for (const auto &it : m_data) { + if (!it.second->empty()) + n++; + } + return n; +}