// Ryzom - MMORPG Framework
// Copyright (C) 2010 Winch Gate Property Limited
//
// This program is free software: you can redistribute it and/or modify
// it under the terms of the GNU Affero General Public License as
// published by the Free Software Foundation, either version 3 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 Affero General Public License for more details.
//
// You should have received a copy of the GNU Affero General Public License
// along with this program. If not, see .
//-----------------------------------------------------------------------------
// includes
//-----------------------------------------------------------------------------
#include "stdpch.h"
#include "object.h"
#include "utils.h"
#include
namespace R2
{
class CSerializeContext
{
public:
CSerializeContext():_Indent(0){}
sint32 getIndent() const { return _Indent;}
void add() { ++_Indent; }
void sub() { --_Indent; }
private:
sint32 _Indent;
};
class CTableIntegrityChecker
{
public:
CTableIntegrityChecker(const CObjectTable &table) : _Table(table)
{
_Table.checkIntegrity();
}
~CTableIntegrityChecker()
{
_Table.checkIntegrity();
}
private:
const CObjectTable &_Table;
};
#ifdef NL_DEBUG
#define CHECK_TABLE_INTEGRITY CTableIntegrityChecker __cti(*this);
#else
#define CHECK_TABLE_INTEGRITY
#endif
//CObjectFactory *CObjectSerializer::Factory = NULL;
CObjectFactory *CObjectSerializerClient::_ClientObjecFactory = NULL;
static CObject *newTable(CObjectFactory *factory)
{
if (factory)
{
return factory->newBasic("Table");
}
else
{
return new CObjectTable;
}
}
//----------------------- private implementation stuffs ----------------------------------------
static void addTab(std::string& out, sint32 n)
{
for (sint32 i = 0 ; i < n; ++i)
{
out += " ";
}
}
//----------------------- CObject ----------------------------------------
CObject::~CObject()
{
BOMB_IF(_Validation != 0x01020304, "Error (double delete)?", return);
_Validation = 0;
}
CObject::CObject()
{
_Parent = 0;
_Ghost = false;
_Validation = 0x01020304;
}
void CObject::previsit(std::vector &sons)
{
sons.push_back(this);
}
void CObject::visit(IObjectVisitor &visitor)
{
//H_AUTO(R2_CObjectTable_visit)
std::vector sons;
// must work on a copy here, because the 'visit' method may change the current list of sons of this table
// Example of a scenario where this happened :
// - Create a bandit camp
// - onPostCreate (1) is called (see in client : CInstance::onPostCreate)
// - create ghost bandit as sons of current bandit camps, their 'onPostCreate' method is called
// - returns to previous onPostCreate (1) -> the list of sons has changed
// -> as a result, onPostCreated was called twice on instances that were just created, leading to a crash (the onPostCreate method registered
// the new object in some manager, and this registration is only allowed once per object)
previsit(sons);
for(uint k = 0; k < sons.size(); ++k)
{
if (sons[k]) // may become NULL if son was deleted during the visit callback
{
sons[k]->visitInternal(visitor);
}
}
}
void CObject::inPlaceCopy(const CObject &src)
{
//H_AUTO(R2_CObject_inPlaceCopy)
src.inPlaceCopyTo(*this);
}
bool CObject::getGhost() const
{
//H_AUTO(R2_CObject_getGhost)
return _Ghost;
}
void CObject::setGhost(bool ghost)
{
//H_AUTO(R2_CObject_setGhost)
_Ghost = ghost;
}
void CObject::copyMismatchMsg(const CObject &src)
{
//H_AUTO(R2_CObject_copyMismatchMsg)
nlwarning("Can't copy object of type %s into object of type %s", src.getTypeAsString(), this->getTypeAsString());
}
void CObject::inPlaceCopy(const CObjectString &src)
{
//H_AUTO(R2_CObject_inPlaceCopy)
copyMismatchMsg(src);
}
void CObject::inPlaceCopy(const CObjectNumber &src)
{
//H_AUTO(R2_CObject_inPlaceCopy)
copyMismatchMsg(src);
}
void CObject::inPlaceCopy(const CObjectInteger &src)
{
//H_AUTO(R2_CObject_inPlaceCopy)
copyMismatchMsg(src);
}
void CObject::inPlaceCopy(const CObjectTable &src)
{
//H_AUTO(R2_CObject_inPlaceCopy)
copyMismatchMsg(src);
}
bool CObject::isNumber(const std::string & prop) const
{
//H_AUTO(R2_CObject_isNumber)
if (!prop.empty())
{
CObject* attr = getAttr(prop);
if (!attr) return false;
return attr->doIsNumber();
}
return doIsNumber();
}
bool CObject::isInteger(const std::string & prop) const
{
//H_AUTO(R2_CObject_isInteger)
if (!prop.empty())
{
CObject* attr = getAttr(prop);
if (!attr) return false;
return attr->doIsInteger();
}
return doIsInteger();
}
bool CObject::isString(const std::string & prop) const
{
//H_AUTO(R2_CObject_isString)
if (!prop.empty())
{
CObject* attr = getAttr(prop);
if (!attr) return false;
return attr->doIsString();
}
return doIsString();
}
sint32 CObject::findIndex(const std::string &/* key */) const
{
//H_AUTO(R2_CObject_findIndex)
BOMB("Try to use the method findIndex() on a object that is not an CObjectTable", return 0);
return 0;
}
bool CObject::isTable(const std::string & prop) const
{
//H_AUTO(R2_CObject_isTable)
if (!prop.empty())
{
CObject* attr = getAttr(prop);
if (!attr) return false;
return attr->isTable();
}
return doIsTable();
}
bool CObject::isRefId(const std::string & prop /*=""*/) const
{
//H_AUTO(R2_CObject_isRefId)
if (!prop.empty())
{
CObject* attr = getAttr(prop);
if (!attr) return false;
return attr->isTable();
}
return doIsRefId();
}
bool CObject::doIsNumber() const
{
//H_AUTO(R2_CObject_doIsNumber)
return false;
}
bool CObject::doIsInteger() const
{
//H_AUTO(R2_CObject_doIsInteger)
return false;
}
bool CObject::doIsString() const
{
//H_AUTO(R2_CObject_doIsString)
return false;
}
bool CObject::doIsTable() const
{
//H_AUTO(R2_CObject_doIsTable)
return false;
}
bool CObject::doIsRefId() const
{
//H_AUTO(R2_CObject_doIsRefId)
return false;
}
double CObject::toNumber(const std::string & prop) const
{
//H_AUTO(R2_CObject_toNumber)
if (!prop.empty())
{
CObject* attr = getAttr(prop);
if (!attr)
{
BOMB("Try to use the method toNumber() on a NULL Object", return 0);
}
return attr->doToNumber();
}
return doToNumber();
}
sint64 CObject::toInteger(const std::string & prop) const
{
//H_AUTO(R2_CObject_toInteger)
if (!prop.empty())
{
CObject* attr = getAttr(prop);
if (!attr)
{
BOMB("Try to use the method toInteger() on a NULL Object", return 0);
}
return attr->doToInteger();
}
return doToInteger();
}
std::string CObject::toString(const std::string & prop) const
{
//H_AUTO(R2_CObject_toString)
if (!prop.empty())
{
CObject* attr = getAttr(prop);
if (!attr)
{
BOMB(NLMISC::toString("Try to access to the property '%s' that does not exist.",prop.c_str()) , return "");
return "";
}
return attr->doToString();
}
return doToString();
}
CObjectTable* CObject::toTable(const std::string & prop) const
{
//H_AUTO(R2_CObject_toTable)
if (!prop.empty())
{
CObject* attr = getAttr(prop);
if (!attr)
{
BOMB("Try to use the method toTable() on a NULL Object", return 0);
}
return attr->doToTable();
}
return doToTable();
}
CObject* CObject::take(sint32 /* position */)
{
//H_AUTO(R2_CObject_take)
BOMB("Try to use the take function on an object that is not a table", return 0);
return 0;
}
bool CObject::canTake(sint32 /* position */) const
{
//H_AUTO(R2_CObject_canTake)
return false;
}
double CObject::doToNumber() const
{
//H_AUTO(R2_CObject_doToNumber)
BOMB("Try to convert an objet to number without being allowed", return 0);
return 0;
}
sint64 CObject::doToInteger() const
{
//H_AUTO(R2_CObject_doToInteger)
BOMB("Try to convert an objet to integer without being allowed", return 0);
return 0;
}
std::string CObject::doToString() const
{
//H_AUTO(R2_CObject_doToString)
BOMB("Try to convert an objet to string without being allowed", return "");
return "";
}
CObjectTable* CObject::doToTable() const
{
//H_AUTO(R2_CObject_doToTable)
BOMB("Try to convert an objet to string without being allowed", return 0);
return 0;
}
CObject* CObject::getAttr(const std::string & /* name */) const { return 0;}
std::string CObject::getKey(uint32 /* pos */) const{ BOMB("Try to call the function getKey() on an object that is not a table", return ""); return "";}
CObject* CObject::getValueAtPos(uint32 /* pos */) const{ BOMB("Try to call the function getValueAtPos() on an object that is not a table", return 0); return 0;}
uint32 CObject::getSize() const { BOMB("Try to call the function getSize() on an object that is not a table", return 0); return 0; }
CObject* CObject::clone() const { BOMB("Try to call the function clone() on an object that is not a table", return 0); return 0;}
void CObject::add(const std::string & key, CObject* value)
{
//H_AUTO(R2_CObject_add)
insert(key, value, -1);
}
void CObject::add(CObject* value){ add("", value); }
bool CObject::set(const std::string& /* key */, const std::string & /* value */)
{
BOMB("Try to set the value of an object with a string on an object that does not allowed it", return false);
return false;
}
bool CObject::set(const std::string& /* key */, double /* value */)
{
BOMB("Try to set the value of an object with a double on an object that does not allowed it", return false);
return false;
}
bool CObject::set(const std::string& /* key */, sint64 /* value */)
{
BOMB("Try to set the value of an object with an integer on an object that does not allowed it", return false);
return false;
}
bool CObject::setObject(const std::string& /* key */, CObject* /* value */)
{
BOMB("Try to set the value of an object with an object that does not allowed it", return false);
return false;
}
void CObject::serialize(std::string& out) const
{
//H_AUTO(R2_CObject_serialize)
CSerializeContext context;
doSerialize(out, context);
}
bool CObject::insert( const std::string& /* key */, CObject* /* value */, sint32 /* position */)
{
//H_AUTO(R2_CObject_insert)
BOMB("Try to call the function insert() on an object that is not a table", return false);
return false;
}
CObject* CObject::getParent() const
{
//H_AUTO(R2_CObject_getParent)
return _Parent;
}
void CObject::setParent(CObject* parent)
{
//H_AUTO(R2_CObject_setParent)
_Parent = parent;
}
void CObject::add(const std::string& key, const std::string & value)
{
//H_AUTO(R2_CObject_add)
this->add(key, new CObjectString(value));
}
void CObject::add(const std::string& key, double value)
{
//H_AUTO(R2_CObject_add)
this->add(key, new CObjectNumber(value));
}
CObject* CObject::findAttr(const std::string & first) const
{
//H_AUTO(R2_CObject_findAttr)
return getAttr(first);
}
CObject* CObject::findAttr(const std::string & first, const std::string & second) const
{
//H_AUTO(R2_CObject_findAttr)
CObject* ret = getAttr(first);
if (ret) { ret = ret->getAttr(second); }
return ret;
}
CObject* CObject::findAttr(const std::string & first, const std::string & second, const std::string & third) const
{
//H_AUTO(R2_CObject_findAttr)
CObject* ret = getAttr(first);
if (ret) { ret = ret->getAttr(second); }
if (ret) { ret = ret->getAttr(third); }
return ret;
}
CObject* CObject::findAttr(const std::vector& attrNames ) const
{
//H_AUTO(R2_CObject_findAttr)
std::vector::const_iterator first(attrNames.begin()), last(attrNames.end());
CObject* ret = const_cast(this);
for ( ; first != last && ret; ++first)
{
if (ret) { ret = ret->getAttr(*first); }
}
return ret;
}
sint32 CObject::findIndex(const CObject* /* child */) const
{
//H_AUTO(R2_CObject_findIndex)
BOMB("Try to call the function findIndex() on an object that is not a table", return 0);
return 0;
}
bool CObject::getShortestName(std::string &instanceId, std::string &attrName, sint32 &position) const
{
//H_AUTO(R2_CObject_getShortestName)
if (isTable() && getAttr("InstanceId"))
{
instanceId = getAttr("InstanceId")->toString();
attrName = "";
position = -1;
return true;
}
CObject *parent = getParent();
if (!parent)
{
nlassert(0); // TMP : want to see if may possibly happen
return false;
}
if (parent->isTable() && parent->getAttr("InstanceId"))
{
sint32 index = parent->findIndex(this);
nlassert(index != -1);
instanceId = parent->getAttr("InstanceId")->toString();
attrName = parent->getKey(index);
if (attrName.empty())
{
position = index;
}
return true;
}
CObject *parent2 = parent->getParent();
if (!parent2)
{
nlassert(0); // TMP : want to see if may possibly happen
return false;
}
if (parent2->isTable() && parent2->getAttr("InstanceId"))
{
sint32 index2 = parent2->findIndex(parent);
nlassert(index2 != -1);
sint32 index = parent->findIndex(this);
nlassert(index != -1);
if (parent2->getKey(index2).empty())
{
nlassert(0); // TMP : want to see if may possibly happen
return false;
}
instanceId = parent2->getAttr("InstanceId")->toString();
attrName = parent2->getKey(index2);
position = index;
return true;
}
return false;
}
bool CObject::getNameInParent(std::string &instanceId,
std::string &attrName,
sint32 &position) const
{
//H_AUTO(R2_CObject_getNameInParent)
CObject *currParent = this->getParent();
if (!currParent) return false;
if (currParent->findAttr("InstanceId"))
{
// i'm a property in my parent
instanceId = currParent->findAttr("InstanceId")->toString();
position = -1;
attrName = currParent->getKey(currParent->findIndex(this));
return true;
}
sint32 tmpPosition = currParent->findIndex(this);
CObject *nextParent = currParent->getParent();
if (!nextParent) return false;
if (nextParent->findAttr("InstanceId"))
{
instanceId = nextParent->findAttr("InstanceId")->toString();
// i'm a property in my parent
position = tmpPosition;
attrName = nextParent->getKey(nextParent->findIndex(currParent));
return true;
}
return false;
}
//----------------------- CObjectString ----------------------------------------
CObjectString::CObjectString(const std::string & value) : CObject(), _Value(value){}
CObjectRefId::~CObjectRefId()
{
//nlwarning("# Destroying CObjectString 0x%x", (int) this);
}
const char *CObjectString::getTypeAsString() const
{
//H_AUTO(R2_CObjectString_getTypeAsString)
return "String";
}
void CObjectString::visitInternal(IObjectVisitor &visitor)
{
//H_AUTO(R2_CObjectString_visit)
visitor.visit(*this);
}
void CObjectString::doSerialize(std::string& out, CSerializeContext& /* context */) const
{
//H_AUTO(R2_CObjectString_doSerialize)
nlassert(!getGhost());
std::string copy;
std::string::size_type first(0), last(_Value.size());
for (; first != last ; ++first)
{
char c = _Value[first];
if (c != ']' && c != '[')
{
copy += c;
}
else
{
copy += NLMISC::toString("]]..\"%c\"..[[", c);
}
}
out += "[[" + copy + "]]";
}
bool CObjectString::set(const std::string& key, const std::string & value)
{
//H_AUTO(R2_CObjectString_set)
BOMB_IF( key != "", "Try to set the a sub value of an object that does not allowed it", return false);
_Value = value;
return true;
}
bool CObjectString::setObject(const std::string& key, CObject* value)
{
//H_AUTO(R2_CObjectString_setObject)
BOMB_IF( !key.empty() || ! (value->isString() || value->isNumber() || value->isInteger()) , "Try to set the a sub value of an object that does not allowed it", return false);
bool canSet = set(key, value->toString());
if (canSet)
{
setGhost(value->getGhost());
}
return canSet;
}
void CObjectString::inPlaceCopyTo(CObject &dest) const
{
//H_AUTO(R2_CObjectString_inPlaceCopyTo)
dest.inPlaceCopy(*this);
}
void CObjectString::inPlaceCopy(const CObjectString &src)
{
//H_AUTO(R2_CObjectString_inPlaceCopy)
_Value = src._Value;
setGhost(src.getGhost());
}
std::string CObjectString::doToString() const { return _Value;}
CObject* CObjectString::clone() const
{
//H_AUTO(R2_CObjectString_clone)
CObjectString *result = new CObjectString(_Value);
result->setGhost(getGhost());
return result;
}
bool CObjectString::doIsString() const { return true;}
bool CObjectString::equal(const CObject* other) const
{
//H_AUTO(R2_CObjectString_equal)
if (!other || !other->isString()) return false;
std::string otherValue = other->toString();
return _Value == otherValue;
}
//----------------------- CObjectRefId ----------------------------------------
CObjectRefId::CObjectRefId(const std::string & value) : CObjectString(value)
{
}
const char *CObjectRefId::getTypeAsString() const
{
//H_AUTO(R2_CObjectRefId_getTypeAsString)
return "RefId";
}
CObject* CObjectRefId::clone() const
{
//H_AUTO(R2_CObjectRefId_clone)
return new CObjectRefId(*this);
}
void CObjectRefId::visitInternal(IObjectVisitor &visitor)
{
//H_AUTO(R2_CObjectRefId_visit)
visitor.visit(*this);
}
bool CObjectRefId::equal(const CObject* other) const
{
//H_AUTO(R2_CObjectRefId_equal)
if (!other || !other->isRefId()) return false;
std::string otherValue = other->toString();
if (getValue() == otherValue ) return true;
return false;
}
bool CObjectRefId::doIsRefId() const
{
//H_AUTO(R2_CObjectRefId_doIsRefId)
return true;
}
void CObjectRefId::doSerialize(std::string& out, CSerializeContext& /* context */) const
{
//H_AUTO(R2_CObjectRefId_doSerialize)
nlassert(!getGhost());
//out << "r2.RefId([[" << getValue() << "]])";
out += "r2.RefId([[" + getValue() + "]])";
}
//----------------------- CObjectNumber ----------------------------------------
CObjectNumber::CObjectNumber(double value) : CObject(), _Value(value){}
const char *CObjectNumber::getTypeAsString() const
{
return "Number";
}
void CObjectNumber::visitInternal(IObjectVisitor &visitor)
{
//H_AUTO(R2_CObjectNumber_visit)
visitor.visit(*this);
}
void CObjectNumber::inPlaceCopyTo(CObject &dest) const
{
//H_AUTO(R2_CObjectNumber_inPlaceCopyTo)
dest.inPlaceCopy(*this);
}
void CObjectNumber::inPlaceCopy(const CObjectNumber &src)
{
//H_AUTO(R2_CObjectNumber_inPlaceCopy)
_Value = src._Value;
setGhost(src.getGhost());
}
std::string CObjectNumber::doToString() const { return NLMISC::toString("%lf", _Value);}
void CObjectNumber::doSerialize(std::string& out, CSerializeContext& /* context */) const
{
//H_AUTO(R2_CObjectNumber_doSerialize)
nlassert(!getGhost());
//out.precision(15);
std::string value = NLMISC::toString(double(sint64(_Value * 1000.0 + (_Value>=0.0?0.5:-0.5)))/1000.0);
// search for first not 0 value from the end
std::string::size_type pos = value.find_last_not_of('0');
if (pos != std::string::npos)
{
// don't remove character at pos if it's another digit
if (value[pos] != '.') ++pos;
value.erase(pos);
}
out += value;
}
bool CObjectNumber::set(const std::string& key, double value)
{
//H_AUTO(R2_CObjectNumber_set)
BOMB_IF(key != "", "Try to set an element of a table on an object that is not a table", return false);
_Value = value;
return true;
}
bool CObjectNumber::set(const std::string& key, const std::string & value)
{
//H_AUTO(R2_CObjectNumber_set)
//XXX
BOMB_IF(key != "", "Try to set an element of a table on an object that is not a table", return false);
NLMISC::fromString(value, _Value);
return true;
}
double CObjectNumber::doToNumber() const { return _Value; }
CObject* CObjectNumber::clone() const
{
//H_AUTO(R2_CObjectNumber_clone)
CObjectNumber *result = new CObjectNumber(_Value);
result->setGhost(getGhost());
return result;
}
bool CObjectNumber::doIsNumber() const { return true;}
bool CObjectNumber::setObject(const std::string& /* key */, CObject* value)
{
//H_AUTO(R2_CObjectNumber_setObject)
BOMB_IF(!value->isNumber(), NLMISC::toString("Try to set an element of a type '%s' with a value of type '%s' on an object that is not a number", this->getTypeAsString(), value->getTypeAsString()), return false);
_Value = value->toNumber();
setGhost(value->getGhost());
return true;
}
bool CObjectNumber::equal(const CObject* other) const
{
//H_AUTO(R2_CObjectNumber_equal)
if (!other || !other->isNumber()) return false;
double otherValue = other->toNumber();
if (_Value == otherValue) return true;
// if difference between 2 values less than epsilon, we consider they are equals
return fabs(_Value - otherValue) <= std::numeric_limits::epsilon();
}
//----------------------- CObjectInteger ----------------------------------------
CObjectInteger::CObjectInteger(sint64 value) : CObject(), _Value(value){}
const char *CObjectInteger::getTypeAsString() const
{
return "Integer";
}
void CObjectInteger::visitInternal(IObjectVisitor &visitor)
{
//H_AUTO(R2_CObjectInteger_visit)
visitor.visit(*this);
}
void CObjectInteger::inPlaceCopyTo(CObject &dest) const
{
//H_AUTO(R2_CObjectInteger_inPlaceCopyTo)
dest.inPlaceCopy(*this);
}
void CObjectInteger::inPlaceCopy(const CObjectInteger &src)
{
//H_AUTO(R2_CObjectInteger_inPlaceCopy)
_Value = src._Value;
setGhost(src.getGhost());
}
std::string CObjectInteger::doToString() const { return NLMISC::toString("%d"NL_I64, _Value); }
void CObjectInteger::doSerialize(std::string& out, CSerializeContext& /* context */) const
{
//H_AUTO(R2_CObjectInteger_doSerialize)
nlassert(!getGhost());
out += NLMISC::toString(_Value);
}
bool CObjectInteger::set(const std::string& key, sint64 value)
{
//H_AUTO(R2_CObjectInteger_set)
BOMB_IF(key != "", "Try to set an element of a table on an object that is not a table", return false);
_Value = value;
return true;
}
bool CObjectInteger::set(const std::string& key, const std::string & value)
{
//H_AUTO(R2_CObjectInteger_set)
//XXX
BOMB_IF(key != "", "Try to set an element of a table on an object that is not a table", return false);
NLMISC::fromString(value, _Value);
return true;
}
sint64 CObjectInteger::doToInteger() const { return _Value; }
CObject* CObjectInteger::clone() const
{
//H_AUTO(R2_CObjectInteger_clone)
CObjectInteger *result = new CObjectInteger(_Value);
result->setGhost(getGhost());
return result;
}
bool CObjectInteger::doIsInteger() const { return true;}
bool CObjectInteger::setObject(const std::string& /* key */, CObject* value)
{
//H_AUTO(R2_CObjectInteger_setObject)
BOMB_IF(!value->isInteger(), NLMISC::toString("Try to set an element of a type '%s' with a value of type '%s' on an object that is not an integer", this->getTypeAsString(), value->getTypeAsString()), return false);
_Value = value->toInteger();
setGhost(value->getGhost());
return true;
}
bool CObjectInteger::equal(const CObject* other) const
{
//H_AUTO(R2_CObjectInteger_equal)
if (!other || !other->isInteger()) return false;
sint64 otherValue = other->toInteger();
return _Value == otherValue;
}
//----------------------- CObjectTable ----------------------------------------
CObjectTable::CObjectTable() : CObject(){}
void CObjectTable::checkIntegrity() const
{
//H_AUTO(R2_CObjectTable_checkIntegrity)
static volatile bool testWanted = true;
if (!testWanted) return;
if (_Ghost)
{
for(uint k = 0; k < getSize(); ++k)
{
CObject *subObj = getValueAtPos(k);
if (subObj)
{
if (!subObj->getGhost())
{
// dump whole hierarchy
const CObject *topParent = this;
while (topParent->getParent()) topParent = topParent->getParent();
topParent->dump();
nlwarning("Check Integrity failed.");
return;
}
subObj->checkIntegrity();
}
}
}
}
void CObjectTable::setGhost(bool ghost)
{
//H_AUTO(R2_CObjectTable_setGhost)
CHECK_TABLE_INTEGRITY
_Ghost = ghost;
CObject* parent = this->getParent();
if (!ghost && parent)
{
sint32 index = parent->findIndex(this);
if (index != -1)
{
std::string key = parent->getKey( static_cast(index));
if (key == "Ghosts" )
{
return;
}
}
}
for(uint k = 0; k < getSize(); ++k)
{
CObject *subObj = getValueAtPos(k);
if (subObj)
{
subObj->setGhost(ghost);
}
}
}
void CObjectTable::previsit(std::vector &sons)
{
sons.push_back(this);
sons.reserve(sons.size() + getSize());
for(uint k = 0; k < getSize(); ++k)
{
CObject *subObj = getValueAtPos(k);
if (subObj)
{
subObj->previsit(sons);
}
}
}
void CObjectTable::visitInternal(IObjectVisitor &visitor)
{
visitor.visit(*this);
}
void CObjectTable::inPlaceCopyTo(CObject &dest) const
{
//H_AUTO(R2_CObjectTable_inPlaceCopyTo)
CHECK_TABLE_INTEGRITY
dest.inPlaceCopy(*this);
}
void CObjectTable::inPlaceCopy(const CObjectTable &src)
{
//H_AUTO(R2_CObjectTable_inPlaceCopy)
CHECK_TABLE_INTEGRITY
for(uint k = 0; k < src.getSize(); ++k)
{
if (!src.getKey(k).empty())
{
CObject *dest = getAttr(src.getKey(k));
if (!dest)
{
nlwarning("No dest object %s found when copying in place", src.getKey(k).c_str());
}
else
{
src.getValueAtPos(k)->inPlaceCopyTo(*dest);
}
}
else
{
nlwarning("In place copy of objects with a number as key not supported");
}
}
}
CObjectTable::~CObjectTable()
{
{
CHECK_TABLE_INTEGRITY
}
TContainer::iterator first(_Value.begin()), last(_Value.end());
for ( ;first != last; ++first )
{
delete first->second;
first->second = 0;
}
_Value.clear();
}
const char *CObjectTable::getTypeAsString() const
{
//H_AUTO(R2_CObjectTable_getTypeAsString)
return "Table";
}
//complexity less than
void CObjectTable::sort()
{
//H_AUTO(R2_CObjectTable_sort)
CHECK_TABLE_INTEGRITY
std::vector < std::pair > data;
std::vector keyVector;
CObject* keys = getAttr("Keys");
if (!keys || !keys->isTable()) return;
uint32 firstKey = 0;
uint32 lastKey = keys->getSize();
for (; firstKey != lastKey ; ++firstKey)
{
CObject* keyObject = keys->getValueAtPos(firstKey);
if (! keyObject->isString()) return;
std::string key = keyObject->toString();
uint32 firstValue = 0;
uint32 lastValue = (uint32)_Value.size();
for (; firstValue != lastValue; ++firstValue)
{
if ( key == _Value[firstValue].first)
{
data.push_back( _Value[firstValue]);
_Value[firstValue].second = 0;
}
}
}
{
uint32 firstValue = 0;
uint32 lastValue = (uint32)_Value.size();
for (; firstValue != lastValue; ++firstValue)
{
if ( _Value[firstValue].first != "Keys" && _Value[firstValue].second != 0)
{
data.push_back( _Value[firstValue]);
}
}
}
_Value.swap(data);
delete keys;
}
CObject* CObjectTable::clone() const
{
//H_AUTO(R2_CObjectTable_clone)
CHECK_TABLE_INTEGRITY
CObject* ret = new CObjectTable();
TContainer::const_iterator first(_Value.begin()), last(_Value.end());
for ( ;first != last; ++first )
{
BOMB_IF(!first->second, "Try to clone a table with an NULL component", return 0)
nlassert(first->second->getGhost() == this->getGhost());
CObject* clone = first->second->clone();
if (clone) { clone->setParent(0); }
ret->add(first->first, clone);
}
ret->setGhost(getGhost());
#ifdef NL_DEBUG
ret->checkIntegrity();
#endif
return ret;
}
namespace
{
struct CValueIndex
{
CValueIndex(unsigned int index, const std::string& name, bool isTable, bool isDefaultFeature)
:Index(index), Name(name), IsTable(isTable), IsDefaultFeature(isDefaultFeature){}
bool operator < (const CValueIndex& rh) const
{
// container after values
if ( !IsTable && rh.IsTable ) { return true; }
if ( IsTable && !rh.IsTable ) { return false; }
if ( !IsTable)
{
if ( Name == "InstanceId" && (rh.Name != "InstanceId" ) ) { return true;}
if ( Name != "InstanceId" && (rh.Name == "InstanceId" ) ) { return false;}
if ( Name == "Class" && (rh.Name != "Class" ) ) { return true;}
if ( Name != "Class" && (rh.Name == "Class" ) ) { return false;}
if ( Name == "Version" && (rh.Name != "Version" ) ) { return true;}
if ( Name != "Version" && (rh.Name == "Version" ) ) { return false;}
}
if (IsTable && rh.IsTable)
{
if (IsDefaultFeature) { return true; }
if (rh.IsDefaultFeature) { return false; }
return Name < rh.Name;
}
return Name < rh.Name;
}
uint32 Index;
std::string Name;
bool IsTable;
bool IsDefaultFeature;
};
}
void CObjectTable::doSerialize(std::string& out, CSerializeContext& context) const
{
//H_AUTO(R2_CObjectTable_doSerialize)
std::vector indexs;
uint32 i = 0;
uint32 size = (uint32)_Value.size();
indexs.reserve(size);
for (; i < size ; ++i)
{
bool isDefault= false;
if (_Value[i].second->isTable() && _Value[i].second->isString("Class"))
{
std::string cl = _Value[i].second->toString("Class");
if (cl == "DefaultFeature")
{
isDefault = true;
}
else if ( cl == "Act")
{
if ( _Value[i].second->isString("Name")
&& _Value[i].second->isString("LocationId")
&& _Value[i].second->toString("Name") == "Permanent"
&& _Value[i].second->toString("LocationId").empty())
{
isDefault = true;
}
}
}
indexs.push_back( CValueIndex(i, _Value[i].first, _Value[i].second->isTable(), isDefault) );
}
std::stable_sort(indexs.begin(), indexs.end());
CHECK_TABLE_INTEGRITY
if (getGhost())
{
nlwarning("Try to serialize a ghost Component");
return;
}
nlassert(!getGhost());
sint32 indent = context.getIndent();
out += "{";
uint32 j = 0;
context.add();
for (; j < size ; ++j)
{
uint32 i = indexs[j].Index;
static const std::string ghostStr = "Ghost_";
static const std::string::size_type ghostStrSize = ghostStr.size();
std::string key = _Value[i].first;
if (!_Value[i].second->getGhost() && !(key.size() > ghostStrSize && key.substr(0, ghostStrSize) == ghostStr))
{
out += "\n";
addTab(out, context.getIndent());
if (!_Value[i].first.empty())
{
out += _Value[i].first + " = ";
}
_Value[i].second->doSerialize(out, context);
// if (j != size -1)
{
out += ", ";
}
}
}
context.sub();
out += "\n";
addTab(out, indent);
out += "}";
}
CObject* CObjectTable::getAttr(const std::string & name) const
{
//H_AUTO(R2_CObjectTable_getAttr)
CHECK_TABLE_INTEGRITY
//search by position
//XXX
if (name.size() >= 1 && '0' <= name[0] && name[0] <='9')
{
uint32 first2 = 0;
uint32 end2 = (uint32)name.size();
for ( ; first2 != end2 && '0' <= name[first2] && name[first2] <= '9'; ++first2) {}
if (first2 == end2)
{
uint32 position;
NLMISC::fromString(name, position);
if (position < _Value.size())
{
return _Value[position].second;
}
}
}
//search by name
TContainer::const_iterator first(_Value.begin());
TContainer::const_iterator last(_Value.end());
for (; first != last ;++first)
{
if (first->first == name)
{
return first->second;
}
}
return 0;
}
std::string CObjectTable::getKey(uint32 pos) const
{
//H_AUTO(R2_CObjectTable_getKey)
CHECK_TABLE_INTEGRITY
if (pos >= _Value.size())
{
nlwarning(" bad index %d", pos);
return "";
}
return _Value[pos].first;
}
CObject* CObjectTable::getValueAtPos(uint32 pos) const
{
//H_AUTO(R2_CObjectTable_getValue)
if (pos >= _Value.size())
{
nlwarning(" bad index %d", pos);
return NULL;
}
return _Value[pos].second;
}
bool CObjectTable::set(const std::string& key, const std::string & value)
{
//H_AUTO(R2_CObjectTable_set)
CHECK_TABLE_INTEGRITY
CObject *attr = getAttr(key);
if (!attr)
{
CObject *str = new CObjectString(value);
str->setGhost(this->getGhost());
this->add(key, str);
//nlwarning("Can't find attribute '%s' in object of class '%s'", key.c_str(), toString("Class").c_str());
return true;
}
return attr->set("", value);
}
bool CObjectTable::set(const std::string& key, double value)
{
//H_AUTO(R2_CObjectTable_set)
CHECK_TABLE_INTEGRITY
CObject *attr = getAttr(key);
if (!attr)
{
//nlwarning("Can't find attribute '%s' in object of class '%s'", key.c_str(), toString("Class").c_str());
CObject *nbr = new CObjectNumber(value);
nbr->setGhost(this->getGhost());
this->add(key, nbr);
return true;
}
return attr->set("", value);
}
bool CObjectTable::setObject(const std::string& key, CObject* value)
{
//H_AUTO(R2_CObjectTable_setObject)
CHECK_TABLE_INTEGRITY
value->setGhost(this->getGhost());
if (key.empty())
{
clear();
uint32 first = 0;
CObject* table = value;
//check
uint32 last = table->getSize();
for ( ; first != last ; ++first)
{
std::string key1 = table->getKey(first);
CObject* value1 = table->getValueAtPos(first);
add(key1, value1->clone());
}
return true;
//getAttrId
//remove
}
else
{
CObject* attr= getAttr(key);
if (!attr)
{
this->add(key, value->clone());
return true;
}
if (attr){ return attr->setObject("", value); }
}
return false;
}
void CObjectTable::clear()
{
//H_AUTO(R2_CObjectTable_clear)
CHECK_TABLE_INTEGRITY
TContainer::iterator first(_Value.begin());
TContainer::iterator last(_Value.end());
for (; first != last ;++first)
{
delete first->second;
}
_Value.clear();
}
uint32 CObjectTable::getSize() const
{
//H_AUTO(R2_CObjectTable_getSize)
return (uint32)_Value.size();
}
bool CObjectTable::doIsTable() const { return true;}
CObjectTable* CObjectTable::doToTable() const { return const_cast(this);}
sint32 CObjectTable::findIndex(const CObject* child) const
{
//H_AUTO(R2_CObjectTable_findIndex)
CHECK_TABLE_INTEGRITY
uint32 first = 0, last = (uint32)_Value.size();
for (; first != last && _Value[first].second != child ; ++first){}
if (first == last) return -1;
return first;
}
sint32 CObjectTable::findIndex(const std::string &key) const
{
//H_AUTO(R2_CObjectTable_findIndex)
CHECK_TABLE_INTEGRITY
uint32 first = 0, last = (uint32)_Value.size();
for (; first != last && _Value[first].first != key ; ++first){}
if (first == last) return -1;
return first;
}
CObject* CObjectTable::take(sint32 position)
{
//H_AUTO(R2_CObjectTable_take)
CHECK_TABLE_INTEGRITY
BOMB_IF(!( -1 <= position && position < static_cast(_Value.size())), "Try to take an element that does not exist", return 0);
if (0 <= position && position < static_cast(_Value.size()))
{
CObject* child = _Value[ static_cast(position) ].second;
_Value.erase(_Value.begin() + static_cast(position));
child->setParent(0);
return child;
}
else if (position == -1)
{
CObject* parent = getParent();
if (!parent)
{
return this;
}
uint32 pos = getParent()->findIndex(this);
BOMB_IF(pos >= getParent()->getSize(), "Try to take an element that does not exist", return 0);
CObject* child = getParent()->take(pos);
return child;
}
return 0;
}
bool CObjectTable::canTake(sint32 position) const
{
//H_AUTO(R2_CObjectTable_canTake)
CHECK_TABLE_INTEGRITY
if(!( -1 <= position && position < static_cast(_Value.size())))
{
return false;
}
if (position == -1)
{
CObject* parent = getParent();
if (parent) //try to take the root of a tree
{
return true;
}
uint32 pos = parent->findIndex(this);
BOMB_IF(pos >= getParent()->getSize(), "Try to take an element that does not exist", return false);
return getParent()->canTake(pos);
}
return true;
}
bool CObjectTable::insert(const std::string& key, CObject* value, sint32 position)
{
//H_AUTO(R2_CObjectTable_insert)
CHECK_TABLE_INTEGRITY
uint32 count = (uint32)_Value.size();
BOMB_IF(!( -1 <= position && position <= static_cast(count)), "Try to take an element that does not exist", return false);
BOMB_IF(!value, "Try to insert a Null value", return false);
BOMB_IF(value->getParent() != 0, "Try to insert an element that not at the root of the tree.", return false);
value->setParent(this);
// inherit the 'ghost' flag
if (this->getGhost())
{
value->setGhost(true);
}
if (0<= position && position < static_cast(count))
{
_Value.insert(_Value.begin() + position, std::pair(key, value));
}
else
{
_Value.push_back(std::pair(key, value));
}
return true;
}
bool CObjectTable::equal(const CObject* other) const
{
//H_AUTO(R2_CObjectTable_equal)
CHECK_TABLE_INTEGRITY
if (!other || !other->isTable()) return false;
#ifdef NL_DEBUG
other->checkIntegrity();
#endif
uint32 size;
size = getSize();
if (size != other->getSize()) return false;
uint i = 0;
for ( i=0; i!= size ; ++i )
{
std::string key = other->getKey(i);
CObject* value = other->getValueAtPos(i);
if (key != this->getKey(i)) return false;
if ( !value->equal(this->getValueAtPos(i))) return false;
}
return true;
}
//----------------------- CObjectTable ----------------------------------------
CTypedObject::CTypedObject(const std::string & type): CObjectTable()
{
if (!type.empty())
{
add("Class", type);
}
}
bool CTypedObject::isOk() const { return true;}
//-------------------------------
//----------------------
std::string CNameGiver::getNewName(const std::string & type, sint32 id)
{
//H_AUTO(R2_CNameGiver_getNewName)
//XX To change
// std::stringstream ss;
// std::string ret = type;
// if (id == -1)
// {
// id = getNewId(type);
// }
// ss << ret << "_" << id;
// return ss.str();
std::string ret = type;
if (id == -1)
{
id = getNewId(type);
}
return ret + "_" + NLMISC::toString(id);
}
sint32 CNameGiver::getNewId(const std::string & type)
{
//H_AUTO(R2_CNameGiver_getNewId)
std::map< std::string, sint32>::const_iterator found = _Value.find(type);
if (found == _Value.end())
{
_Value[type] = 0;
}
return ++_Value[type];
}
CNameGiver::CNameGiver()
{
}
void CNameGiver::setMaxId(const std::string& eid,sint32 id)
{
//H_AUTO(R2_CNameGiver_setMaxId)
_Value[eid]=id;
}
sint32 CNameGiver::getMaxId(const std::string& eid)
{
//H_AUTO(R2_CNameGiver_getMaxId)
return _Value[eid];
}
void CNameGiver::clear()
{
//H_AUTO(R2_CNameGiver_clear)
_Value.clear();
}
//----------------------------------------------
void CObjectFactory::clear()
{
//H_AUTO(R2_CObjectFactory_clear)
_NameGiver->clear();
}
CObject* CObjectFactory::newBasic(const std::string & type)
{
//H_AUTO(R2_CObjectFactory_newBasic)
if (type == "RefId")
{
return new CObjectRefId("");
}
else
if (type == "String")
{
return new CObjectString("");
}
else if (type == "Number")
{
return new CObjectNumber(0);
}
else if (type == "Table")
{
return new CObjectTable();
}
return 0;
}
CObject* CObjectFactory::newAvanced(const std::string & type)
{
//H_AUTO(R2_CObjectFactory_newAvanced)
CObjectGenerator* ret = getGenerator(type);
if (ret) { return ret->instanciate(this); }
return 0;
}
void CObjectFactory::registerGenerator(CObject* objectClass)
{
//H_AUTO(R2_CObjectFactory_registerGenerator)
nlassert(objectClass->isString("Name"));
std::string classType = objectClass->toString("Name");
bool exist=_Map.insert(std::make_pair(classType,new CObjectGenerator(objectClass, this))).second;
if (!exist)
{
nlwarning("Generator with name %s already exists", classType.c_str());
}
}
CObjectFactory::CObjectFactory(const std::string & prefix)
{
//H_AUTO(R2_CObjectFactory_CObjectFactory)
_NameGiver = new CNameGiver();
_Prefix = prefix;
}
void CObjectFactory::setPrefix(const std::string & prefix)
{
//H_AUTO(R2_CObjectFactory_setPrefix)
_Prefix = prefix;
}
CObjectFactory::~CObjectFactory()
{
delete _NameGiver;
//delete map
std::map::iterator it = _Map.begin();
while( it != _Map.end() )
delete (*it++).second;
_Map.clear();
}
void CObjectFactory::setMaxId(const std::string& eid,sint32 id)
{
//H_AUTO(R2_CObjectFactory_setMaxId)
_NameGiver->setMaxId(eid,id);
}
sint32 CObjectFactory::getMaxId(const std::string& eid) const
{
//H_AUTO(R2_CObjectFactory_getMaxId)
return _NameGiver->getMaxId(eid);
}
CObject* CObjectFactory::newComponent(const std::string & type)
{
//H_AUTO(R2_CObjectFactory_newComponent)
CObject* ret = 0;
ret = newBasic(type);
if (ret) return ret;
ret = newAvanced(type);
if (!ret)
{
nlwarning("Component not found : %s", type.c_str());
}
return ret;
}
std::string CObjectFactory::getNewName(const std::string& type) const
{
//H_AUTO(R2_CObjectFactory_getNewName)
if (type.empty())
{
return _NameGiver->getNewName(_Prefix);
}
return _NameGiver->getNewName(type);
}
//-------------------------
CClass::CClass(const std::string & classType):CObjectTable()
{
CObjectTable* prop = new CObjectTable(); // never accessed by client
add("name", classType);
add("prop", prop);
_ClassType = classType;
}
void CClass::addAttribute(const std::string & name, const std::string & type)
{
//H_AUTO(R2_CClass_addAttribute)
findAttr("Prop")->add( "", new CClassAttribute(name, type));
}
void CClass::addAttribute(const std::string & name, const std::string & type, const std::string & value)
{
//H_AUTO(R2_CClass_addAttribute)
findAttr("Prop")->add( "", new CClassAttribute(name, type, value));
}
//-------------
CObjectGenerator::~CObjectGenerator()
{
delete _ObjectClass;
TDefaultValues::iterator first(_DefaultValues.begin()), last(_DefaultValues.end());
for (;first != last; ++first)
{
CObject* data = first->second;
delete data;
}
}
CObjectGenerator * CObjectFactory::getGenerator(const std::string & type)
{
//H_AUTO(R2_CObjectFactory_getGenerator)
std::map::const_iterator found(_Map.find(type));
if (found != _Map.end())
{
CObjectGenerator* ret = found->second;
return ret;
}
return 0;
}
void CObjectGenerator::createDefaultValues(CObjectFactory* factory)
{
//H_AUTO(R2_CObjectGenerator_createDefaultValues)
if (!_ObjectClass) return;
CObject* prop = _ObjectClass->getAttr("Prop");
nlassert(prop);
uint32 first = 0;
uint32 last = prop->getSize();
for ( ; first != last ; ++first )
{
CObject* found = prop->getValueAtPos(first);
if (found && found->isString("DefaultValue") && found->isString("Name") )
{
std::string type = found->toString("Type");
std::string name = found->toString("Name");
CObject* instance = factory->newComponent(type);
if (!instance)
{
nlwarning(" Can't create component of type %s", type.c_str());
return;
}
std::string value = found->toString("DefaultValue");
instance->set("", value);
_DefaultValues.insert( std::make_pair(name, instance));
}
}
}
std::string CObjectGenerator::getBaseClass() const
{
//H_AUTO(R2_CObjectGenerator_getBaseClass)
if (!_ObjectClass) return "";
if ( _ObjectClass->isString("BaseClass"))
{
return _ObjectClass->toString("BaseClass");
}
return "";
}
CObject* CObjectGenerator::getDefaultValue(const std::string & propName) const
{
//H_AUTO(R2_CObjectGenerator_getDefaultValue)
TDefaultValues::const_iterator found ( _DefaultValues.find(propName) );
if (found != _DefaultValues.end())
{
return found->second;
}
return 0;
}
CObject* CObjectGenerator::instanciate(CObjectFactory* factory) const
{
//H_AUTO(R2_CObjectGenerator_instanciate)
CObject* toRet = newTable(factory);
CObject* objectClassType = _ObjectClass->getAttr("Name");
nlassert( objectClassType && objectClassType->isString());
std::string classType = objectClassType->toString();
toRet->add("Class", classType);
CObject* prop = _ObjectClass->getAttr("Prop");
nlassert(prop);
uint32 first = 0;
uint32 last = prop->getSize();
for ( ; first != last ; ++first )
{
CObject* found = prop->getValueAtPos(first);
if (found && found->isString("Type") )
{
CObject*defaultInBase = found->getAttr("DefaultInBase");
if (!defaultInBase || (defaultInBase->isInteger() && defaultInBase->toInteger() != 1))
{
std::string type = found->toString("Type");
/*
:XXX:
Default Value = NIL
*/
CObject* instance = factory->newComponent(type);
if (!instance)
{
nlwarning(" Can't create component of type %s", type.c_str());
}
else
{
nlassert(instance);
std::string name = found->toString("Name");
toRet->add(name, instance);
if (found->isString("DefaultValue"))
{
std::string value = found->toString("DefaultValue");
instance->set("", value);
}
if (name == "InstanceId")
{
std::string value = factory->getNewName();
instance->set("", value);
}
else if (name == "Id")
{
std::string value = factory->getNewName(classType);
instance->set("", value);
}
}
}
}
else
{
if (!found)
{
nlwarning("Field 'Type' not found for a property in class %s", classType.c_str());
}
else
{
nlwarning("Field 'Type' shoud has type 'String' (class = %s)", classType.c_str());
}
}
}
return toRet;
}
void CObjectString::dump(const std::string prefix, uint depth) const
{
//H_AUTO(R2_CObjectString_dump)
std::string result(depth * 4, ' ');
result += NLMISC::toString("%sString, ptr = Ox%p, value = %s, ghost = %s", prefix.c_str(), this, _Value.c_str(), _Ghost ? "true" : "false");
nlwarning(result.c_str());
}
void CObjectNumber::dump(const std::string prefix, uint depth) const
{
//H_AUTO(R2_CObjectNumber_dump)
std::string result(depth * 4, ' ');
result += NLMISC::toString("%sNumber, ptr = 0x%p, value = %lf, ghost = %s", prefix.c_str(), this, _Value, _Ghost ? "true" : "false");
nlwarning(result.c_str());
}
void CObjectInteger::dump(const std::string prefix, uint depth) const
{
//H_AUTO(R2_CObjectInteger_dump)
std::string result(depth * 4, ' ');
result += NLMISC::toString("%sInteger, ptr = 0x%p, value = %d"NL_I64", ghost = %s", prefix.c_str(), this, _Value, _Ghost ? "true" : "false");
nlwarning(result.c_str());
}
void CObjectTable::dump(const std::string prefix, uint depth) const
{
//H_AUTO(R2_CObjectTable_dump)
std::string result(depth * 4, ' ');
result += NLMISC::toString("%sTable, ptr = 0x%p, , ghost = %s", prefix.c_str(), this, _Ghost ? "true" : "false");
nlwarning(result.c_str());
for(uint k = 0; k < _Value.size(); ++k)
{
std::string prefix = NLMISC::toString("Index = %d, key = %s ", (int) k, _Value[k].first.c_str());
_Value[k].second->dump(prefix, depth + 1);
}
}
enum
{
ObjectNull, ObjectString, ObjectNumber, ObjectTable,
ObjectNumberZero, ObjectNumberSInt32, ObjectNumberUInt32, ObjectNumberSInt16, ObjectNumberUInt16, ObjectNumberSInt8, ObjectNumberUInt8, ObjectNumberFloat,
ObjectStringEmpty, ObjectString8, ObjectString16,
ObjectTablePosition, ObjectTableNpc, ObjectTableNpcCustom, ObjectTableWayPoint, ObjectTableRegionVertex,
ObjectTableRegion, ObjectTableRoad,
ObjectTableNpcGrpFeature, ObjectTableBehavior,
ObjectTableActivitySequence, ObjectTableActivityStep,
ObjectTableChatSequence, ObjectTableChatStep,ObjectTableChatAction,
ObjectTableLogicEntityAction,
ObjectTableActionStep, ObjectTableActionType, ObjectTableEventType, ObjectTableLogicEntityReaction,
ObjectTableTextManagerEntry, ObjectTableConditionStep, ObjectTableConditionType,
ObjectTableRtAct, ObjectTableRtNpcGrp, ObjectTableRtNpc, ObjectTableRtPosition,
RtAiState, ObjectTableRtAiState, ObjectTableRtNpcEventHandler, ObjectTableRtNpcEventHandlerAction,
ObjectRefIdEmpty, ObjectRefId8, ObjectRefId16, ObjectRefId,
ObjectNumberDouble, ObjectHeaderTag // If we "sort" this list we bust remove old session save
};
const double sint8Min = -128.;
const double sint8Max = 127.;
const double uint8Max = 255.;
const double sint16Min = -32768.;
const double sint16Max = 32767.;
const double uint16Max = 65535.;
const double sint32Min = -2147483648.;
const double sint32Max = 2147483647.;
const double uint32Max = 4294967295.;
uint32 CObject::instanceIdToUint32(const std::string& instanceId)
{
//H_AUTO(R2_CObject_instanceIdToUint32)
if (instanceId.empty()) return 0;
uint32 size = (uint32)instanceId.size();
if ( instanceId.substr(0, 6) != "Client")
{
nlwarning("R2Share: Wrong InstanceId(%s)", instanceId.c_str());
return 0;
}
std::string::size_type clientIdIt= instanceId.find("_", 6);
if (clientIdIt == std::string::npos)
{
nlwarning("R2Share: Wrong InstanceId(%s)", instanceId.c_str());
return 0;
}
std::string clientIdStr = instanceId.substr(6, clientIdIt-6);
std::string componentIdStr = instanceId.substr(clientIdIt+1, size - clientIdIt);
bool ko;
uint32 clientId;
ko = NLMISC::fromString(clientIdStr, clientId);
if (!ko)
{
nlwarning("R2Share: Wrong InstanceId(%s)", instanceId.c_str());
return 0;
}
uint32 componentId;
ko = NLMISC::fromString(componentIdStr, componentId);
if (!ko)
{
nlwarning("R2Share: Wrong InstanceId(%s)", instanceId.c_str());
return 0;
}
return (0xff000000 & clientId << 24) | (componentId & 0x00ffffff);
}
std::string CObject::uint32ToInstanceId(uint32 id)
{
//H_AUTO(R2_CObject_uint32ToInstanceId)
if (id == 0) return "";
uint32 clientId = (id>> 24) & 0x000000ff;
uint32 componentId = (id & 0x00ffffff);
return NLMISC::toString("Client%d_%d", clientId, componentId);
}
static void writeNumber( NLMISC::IStream& stream, double theValue)
{
double value = theValue;
double absValue = fabs(value);
uint8 type;
// It's 0
if (absValue <= std::numeric_limits::epsilon())
{
type = ObjectNumberZero;
stream.serial(type);
return;
}
double integral;
double fractional = modf(absValue, &integral);
// It is an integral type (no fractional part)
if ( fractional <= std::numeric_limits::epsilon() )
{
bool pos = 0.0 <= value;
// positif
if (pos)
{
if (integral <= uint8Max)
{
uint8 uint8value = static_cast(value);
type = ObjectNumberUInt8;
stream.serial(type);
stream.serial( uint8value);
return;
}
if (integral <= uint16Max)
{
uint16 uint16value = static_cast(value);
type = ObjectNumberUInt16;
stream.serial(type);
stream.serial(uint16value);
return;
}
if (integral <= uint32Max)
{
uint32 uint32value = static_cast(value);
type = ObjectNumberUInt32;
stream.serial(type);
stream.serial(uint32value);
return;
}
}
//negatif
else
{
if ( sint8Min <= integral && integral <= sint8Max)
{
sint8 sint8value = static_cast(value);
type = ObjectNumberSInt8;
stream.serial(type);
stream.serial( sint8value);
return;
}
if ( sint16Min <= integral && integral <= sint16Max)
{
sint16 sint16value = static_cast(value);
type = ObjectNumberSInt16;
stream.serial(type);
stream.serial( sint16value);
return;
}
if ( sint32Min <= integral && integral <= sint32Max)
{
sint32 sint32value = static_cast(value);
type = ObjectNumberSInt32;
stream.serial(type);
stream.serial( sint32value);
return;
}
}
}
//Default case
// Float are evil: you loose too much precision
type = ObjectNumberDouble;
double fValue = value;
stream.serial(type);
stream.serial(fValue);
}
static void writeInteger( NLMISC::IStream& stream, sint64 theValue)
{
sint64 value = theValue;
uint8 type;
// It's 0
if (value == 0)
{
type = ObjectNumberZero;
stream.serial(type);
return;
}
bool pos = value >= 0;
// positif
if (pos)
{
if (value <= uint8Max)
{
uint8 uint8value = static_cast(value);
type = ObjectNumberUInt8;
stream.serial(type);
stream.serial( uint8value);
return;
}
if (value <= uint16Max)
{
uint16 uint16value = static_cast(value);
type = ObjectNumberUInt16;
stream.serial(type);
stream.serial(uint16value);
return;
}
if (value <= uint32Max)
{
uint32 uint32value = static_cast(value);
type = ObjectNumberUInt32;
stream.serial(type);
stream.serial(uint32value);
return;
}
}
//negatif
else
{
if ( sint8Min <= value && value <= sint8Max)
{
sint8 sint8value = static_cast(value);
type = ObjectNumberSInt8;
stream.serial(type);
stream.serial( sint8value);
return;
}
if ( sint16Min <= value && value <= sint16Max)
{
sint16 sint16value = static_cast(value);
type = ObjectNumberSInt16;
stream.serial(type);
stream.serial( sint16value);
return;
}
}
// Default case
type = ObjectNumberSInt32;
sint32 fValue = (sint32)value;
stream.serial(type);
stream.serial(fValue);
}
static void serialStringInstanceId( NLMISC::IStream& stream, CObject*& data)
{
if (!stream.isReading())
{
uint32 instanceId = CObject::instanceIdToUint32(data->toString());
stream.serial(instanceId);
}
else
{
uint32 instanceId;
stream.serial(instanceId);
std::string strInstanceId = CObject::uint32ToInstanceId(instanceId);
data = new CObjectString(strInstanceId);
}
}
void CObjectSerializer::serialStringInstanceId( NLMISC::IStream& stream, std::string& data)
{
//H_AUTO(R2_CObjectSerializer_serialStringInstanceId)
if (!stream.isReading())
{
uint32 instanceId = CObject::instanceIdToUint32(data);
stream.serial(instanceId);
}
else
{
uint32 instanceId;
stream.serial(instanceId);
data= CObject::uint32ToInstanceId(instanceId);
}
}
void CObjectSerializer::swap(CObjectSerializer& other)
{
//H_AUTO(R2_CObjectSerializer_swap)
std::swap(this->_Data, other._Data);
std::swap(this->_Compressed, other._Compressed);
std::swap(this->_MustUncompress, other._MustUncompress);
std::swap(this->_CompressedBuffer, other._CompressedBuffer);
std::swap(this->_CompressedLen, other._CompressedLen);
std::swap(this->_UncompressedLen, other._UncompressedLen);
std::swap(this->Level, other.Level);
std::swap(this->Log, other.Log);
}
static void serialNumberFixedPoint( NLMISC::IStream& stream, CObject*& data, CObjectSerializer* serializer )
{
if (serializer->getVersion() == 0)
{
nlwarning("Using oldScenario Version (must only be use when loading) old scenarios scenario session");
if (!stream.isReading())
{
sint32 v = static_cast(data->toNumber()* 64);
stream.serial(v);
}
else
{
sint32 v;
stream.serial(v);
data = new CObjectNumber( 1.0 * v / 64.0);
}
}
else
{
if (!stream.isReading())
{
double val = data->toNumber();
sint32 v = static_cast( val * 1000.0 + (val >= 0.0? 0.5: -0.5));
stream.serial(v);
// nldebug("serial > %f %d ", val, v);
}
else
{
sint32 v;
stream.serial(v);
double val = double(v) / 1000.0;
data = new CObjectNumber( val );
// nldebug("serial < %f %d", val, v);
}
}
}
class CClassSerializer;
class CObjectSerializerImpl : public NLMISC::CSingleton
{
public:
~CObjectSerializerImpl();
CObjectSerializerImpl();
// release singleton
static void releaseInstance();
void registerSerializer(CClassSerializer* serializer);
void serialImpl(NLMISC::IStream& stream, CObject*& data, CObjectSerializer* serializer, bool init = false);
private:
std::map _ClassSerializersById;
std::map _ClassSerializersByName;
};
class CClassSerializer
{
public:
virtual void serialClass(NLMISC::IStream& stream, CObject*& data, CObjectSerializer* serializer)
{
if (!stream.isReading())
{
//serial other prop;
std::vector optionalPropFoundIndex;
std::vector otherPropFoundIndex;
std::vector valuePropFoundIndex;
/*
make difference between properties
*/
{
uint32 first=0;
uint last=data->getSize();
for (; first != last; ++first)
{
std::string key = data->getKey(first);
if (key == "Class")
{
}
else if ( key.empty() )
{
valuePropFoundIndex.push_back(first);
}
else if (std::find(_NeededProp.begin(), _NeededProp.end(), key) == _NeededProp.end())
{
std::vector::const_iterator optionalFound = std::find(_OptionalProp.begin(), _OptionalProp.end(), key);
if (optionalFound != _OptionalProp.end())
{
optionalPropFoundIndex.push_back(first);
}
else
{
otherPropFoundIndex.push_back(first);
}
}
}
}
uint8 id = getId();
stream.serial(id);
uint8 next = 0;
if (!_NeededProp.empty()) next |= 1;
if (!optionalPropFoundIndex.empty()) next |= 2;
if (!valuePropFoundIndex.empty()) next |= 4;
if (!otherPropFoundIndex.empty()) next |= 8;
stream.serial(next);
/*
Serial needed property
*/
if (!_NeededProp.empty())
{
//no need to put a number we already known how meany object must be
std::vector::const_iterator first(_NeededProp.begin()), last(_NeededProp.end());
uint initLength = stream.getPos();
serializer->Level +=2;
for ( ; first != last ; ++first)
{
//*first => Property key ()
CObject* value = data->getAttr(*first);
if( !value )
{
nlwarning( "About to serialize a NULL value at key %s, className %s!",
first->c_str(), _ClassName.c_str() );
}
if (serializer->Log) { nldebug("R2NET: (%u) Field '%s'", serializer->Level, first->c_str());}
onSerial(stream, *first, value, serializer);
}
serializer->Level -=2;
uint endLength = stream.getPos();
if (serializer->Log) { nldebug("R2NET: (%u) Needed Properties sent %u bytes", serializer->Level +1, endLength - initLength);}
}
// serial optional properties
if (!optionalPropFoundIndex.empty())
{
uint initLength = stream.getPos();
serializer->Level +=2;
uint32 first = 0, last = (uint32)optionalPropFoundIndex.size();
uint8 optionalSize = static_cast(last);
stream.serial(optionalSize);
for (; first != last; ++first)
{
std::string key = data->getKey(optionalPropFoundIndex[first]);
uint32 uiKey = 0, lastLocal = (uint32)_OptionalProp.size();
for ( ; uiKey != lastLocal && _OptionalProp[uiKey]!=key ; ++uiKey){}
uint8 shortKey = static_cast(uiKey);
stream.serial(shortKey);
CObject*value = data->getValueAtPos(optionalPropFoundIndex[first]);
if (serializer->Log) { nldebug("R2NET: (%u) Field '%s'", serializer->Level, key.c_str());}
onSerial(stream, key, value, serializer);
}
serializer->Level -=2;
uint endLength = stream.getPos();
if (serializer->Log) { nldebug("R2NET: (%u) Optional Properties sent %u bytes", serializer->Level +1, endLength - initLength);}
}
// serial value properties
if (!valuePropFoundIndex.empty())
{
uint initLength = stream.getPos();
serializer->Level +=2;
uint32 first = 0, last = (uint32)valuePropFoundIndex.size();
uint16 last16 = static_cast(last);
stream.serial(last16);
for (; first != last; ++first)
{
CObject* value = data->getValueAtPos(valuePropFoundIndex[first]);
if (serializer->Log) { nldebug("R2NET: (%u) Field [%u]", serializer->Level, first);}
CObjectSerializerImpl::getInstance().serialImpl(stream, value, serializer);
}
serializer->Level -=2;
uint endLength = stream.getPos();
if (serializer->Log) { nldebug("R2NET: (%u) Values Properties sent %u bytes", serializer->Level +1, endLength - initLength);}
}
// serial other properties
if (!otherPropFoundIndex.empty())
{
uint initLength = stream.getPos();
serializer->Level +=2;
uint32 first = 0, last = (uint32)otherPropFoundIndex.size();
uint16 last16 = static_cast(last);
stream.serial(last16);
for (; first != last; ++first)
{
uint32 keyIndex = otherPropFoundIndex[first];
std::string key = data->getKey(keyIndex);
CObject* value = data->getValueAtPos(keyIndex);
stream.serial(key);
if (serializer->Log) { nldebug("R2NET: (%u) Field '%s'", serializer->Level, key.c_str());}
CObjectSerializerImpl::getInstance().serialImpl(stream, value, serializer);
}
serializer->Level -=2;
uint endLength = stream.getPos();
if (serializer->Log) { nldebug("R2NET: (%u) Other Properties sent %u bytes", serializer->Level +1, endLength - initLength);}
}
}
else
{
data = newTable(serializer->Factory);
data->add("Class", new CObjectString(_ClassName));
uint8 next;
stream.serial(next);
// Needed pop
if( (next & 1) )
{
std::vector::const_iterator first(_NeededProp.begin()), last(_NeededProp.end());
for ( ; first != last ; ++first)
{
//*first => Property key ()
CObject* value = 0;
std::string key = *first;
onSerial(stream, key, value, serializer);
if (!value)
{
nlwarning( "Error the needed param '%s' of Class '%s' is a NULL value.",
_ClassName.c_str(), first->c_str() );
}
else
{
data->add(key, value);
}
}
}
// Optional value
if ( (next & 2) )
{
uint8 optionalSize;
stream.serial(optionalSize);
uint first(0), last(optionalSize);
for ( ; first != last ; ++first)
{
CObject* value;
uint8 keyIndex;
stream.serial(keyIndex);
std::string key = _OptionalProp[keyIndex];
onSerial(stream, key, value, serializer);
if (!value)
{
nlwarning("R2Share: Stream error");
}
data->add(key, value);
}
}
// Table value
if ( (next & 4) )
{
uint16 arrayValuesSize;
stream.serial(arrayValuesSize);
uint first(0), last(arrayValuesSize);
for ( ; first != last ; ++first)
{
CObject* value;
onSerial(stream, "", value, serializer);
if (!value)
{
nlwarning("R2Share: Stream error");
}
data->add("", value);
}
}
// Other Value
if ( (next & 8) )
{
uint16 otherPropSize;
stream.serial(otherPropSize);
uint first(0), last(otherPropSize);
for ( ; first != last ; ++first)
{
std::string key;
stream.serial(key);
CObject* value=0;
onSerial(stream, key, value, serializer);
if (!value)
{
nlwarning("R2Share: Stream error");
}
else
{
data->add(key, value);
}
}
}
}
}
virtual void onSerial(NLMISC::IStream& stream, const std::string& key, CObject*& data, CObjectSerializer* serializer)
{
if (key == "InstanceId") { serialStringInstanceId(stream, data); return; }
CObjectSerializerImpl::getInstance().serialImpl(stream, data, serializer);
}
virtual ~CClassSerializer(){}
uint8 getId() const { return _Type; }
std::string getClassName() const { return _ClassName; }
protected:
std::vector _NeededProp;
std::vector _OptionalProp;
std::string _ClassName;
uint8 _Type;
};
//------------------
class CNpcSerializer : public CClassSerializer
{
public:
CNpcSerializer();
virtual void onSerial(NLMISC::IStream& stream, const std::string& key, CObject*& data, CObjectSerializer* serializer);
};
CNpcSerializer::CNpcSerializer()
{
static const char* neededProp[] = { "InstanceId", "Base", "Position", "Angle", "Behavior" };
static const char* optionalProp[] = { "GabaritHeight", "GabaritTorsoWidth","GabaritArmsWidth", "GabaritLegsWidth", "GabaritBreastSize",
"HairType", "HairColor", "Tattoo", "EyesColor", "MorphTarget1", "MorphTarget2", "MorphTarget3", "MorphTarget4", "MorphTarget5",
"MorphTarget6", "MorphTarget7", "MorphTarget8", "Sex", "JacketModel", "TrouserModel", "FeetModel", "HandsModel", "ArmModel",
"WeaponRightHand", "WeaponLeftHand", "JacketColor", "ArmColor", "HandsColor", "TrouserColor", "FeetColor", "Function","Level",
"Profile", "Speed", "Aggro", "PlayerAttackable", "BotAttackable" };
_ClassName = "Npc";
_Type = ObjectTableNpc;
const uint32 last = sizeof(neededProp) / sizeof(neededProp[0]) ;
std::vector tmp(&neededProp[0], &neededProp[last]);
_NeededProp.swap(tmp);
std::vector tmp2(&optionalProp[0], &optionalProp[ sizeof(optionalProp) / sizeof(optionalProp[0]) ]);
_OptionalProp.swap(tmp2);
}
void CNpcSerializer::onSerial(NLMISC::IStream& stream, const std::string& key, CObject*& data, CObjectSerializer* serializer)
{
//H_AUTO(R2_CNpcSerializer_onSerial)
if (key == "Angle") { serialNumberFixedPoint(stream, data, serializer); return; }
this->CClassSerializer::onSerial(stream, key, data, serializer);
}
//------------------
class CNpcCustomSerializer : public CClassSerializer
{
public:
CNpcCustomSerializer();
virtual void onSerial(NLMISC::IStream& stream, const std::string& key, CObject*& data, CObjectSerializer* serializer);
};
CNpcCustomSerializer::CNpcCustomSerializer()
{
static const char* neededProp[] = { "InstanceId", "Base", "Position", "Angle", "Behavior",
"GabaritHeight", "GabaritTorsoWidth","GabaritArmsWidth", "GabaritLegsWidth", "GabaritBreastSize",
"HairType", "HairColor", "Tattoo", "EyesColor", "MorphTarget1", "MorphTarget2", "MorphTarget3", "MorphTarget4", "MorphTarget5",
"MorphTarget6", "MorphTarget7", "MorphTarget8", "JacketModel", "TrouserModel", "FeetModel", "HandsModel", "ArmModel",
"JacketColor", "ArmColor", "HandsColor", "TrouserColor", "FeetColor" };
static const char* optionalProp[] = {
"Name", "Profile", "Speed", "Aggro", "PlayerAttackable", "BotAttackable", "Function", "Level", "WeaponRightHand", "WeaponLeftHand", "Sex" };
_ClassName = "NpcCustom";
_Type = ObjectTableNpcCustom;
const uint32 last = sizeof(neededProp) / sizeof(neededProp[0]) ;
std::vector tmp(&neededProp[0], &neededProp[last]);
_NeededProp.swap(tmp);
std::vector tmp2(&optionalProp[0], &optionalProp[ sizeof(optionalProp) / sizeof(optionalProp[0]) ]);
_OptionalProp.swap(tmp2);
}
// TODO -> Lot of property to bitstream
void CNpcCustomSerializer::onSerial(NLMISC::IStream& stream, const std::string& key, CObject*& data, CObjectSerializer* serializer)
{
//H_AUTO(R2_CNpcCustomSerializer_onSerial)
if (key == "Angle") { serialNumberFixedPoint(stream, data, serializer); return; }
this->CClassSerializer::onSerial(stream, key, data, serializer);
}
//--------------------------------
#define R2_CLASS_SERIALIZER_NEEDED(Name, NeededProp ) \
class C##Name##Serializer : public CClassSerializer \
{\
public:\
C##Name##Serializer()\
{ \
_ClassName = #Name;\
_Type = ObjectTable##Name;\
const uint32 last = sizeof(NeededProp) / sizeof(NeededProp[0]) ;\
std::vector tmp(&NeededProp[0], &NeededProp[last]);\
_NeededProp.swap(tmp);\
}\
};
static const char* ActivitySequenceNeededProp[] = {"InstanceId","Name","Repeating","Components"};
R2_CLASS_SERIALIZER_NEEDED(ActivitySequence, ActivitySequenceNeededProp);
static const char* ActivityStepNeededProp[] = {"InstanceId","Type","TimeLimitValue","EventsIds","Chat", "ActivityZoneId", "Name", "TimeLimit", "Activity"};
R2_CLASS_SERIALIZER_NEEDED(ActivityStep, ActivityStepNeededProp);
static const char* ChatSequenceNeededProp[] = {"InstanceId","Name", "Components"};
R2_CLASS_SERIALIZER_NEEDED(ChatSequence, ChatSequenceNeededProp);
static const char* ChatStepNeededProp[] = {"InstanceId","Time", "Actions", "Name"};
R2_CLASS_SERIALIZER_NEEDED(ChatStep, ChatStepNeededProp);
static const char* ChatActionNeededProp[] = {"InstanceId","Emote", "Who", "Facing","Says" };
R2_CLASS_SERIALIZER_NEEDED(ChatAction, ChatActionNeededProp);
static const char* LogicEntityActionNeededProp[] = {"InstanceId","Conditions", "Actions", "Event", "Name"};
R2_CLASS_SERIALIZER_NEEDED(LogicEntityAction, LogicEntityActionNeededProp);
static const char* ActionStepNeededProp[] = {"InstanceId","Entity"};
R2_CLASS_SERIALIZER_NEEDED(ActionStep, ActionStepNeededProp);
static const char* ActionTypeNeededProp[] = {"InstanceId","Type", "Value"};
R2_CLASS_SERIALIZER_NEEDED(ActionType, ActionTypeNeededProp);
static const char* EventTypeNeededProp[] = {"InstanceId","Type", "Value"};
R2_CLASS_SERIALIZER_NEEDED(EventType, EventTypeNeededProp);
static const char* LogicEntityReactionNeededProp[] = {"InstanceId","LogicEntityAction", "ActionStep", "Name"};
R2_CLASS_SERIALIZER_NEEDED(LogicEntityReaction, LogicEntityReactionNeededProp);
static const char* TextManagerEntryNeededProp[] = {"InstanceId","Count", "Text"};
R2_CLASS_SERIALIZER_NEEDED(TextManagerEntry, TextManagerEntryNeededProp);
static const char* ConditionStepNeededProp[] = {"InstanceId","Entity", "Condition"};
R2_CLASS_SERIALIZER_NEEDED(ConditionStep, ConditionStepNeededProp);
static const char* ConditionTypeNeededProp[] = {"InstanceId","Type", "Value"};
R2_CLASS_SERIALIZER_NEEDED(ConditionType, ConditionTypeNeededProp);
//--------------------------------
class CPositionSerializer : public CClassSerializer
{
public:
CPositionSerializer();
virtual void onSerial(NLMISC::IStream& stream, const std::string& key, CObject*& data, CObjectSerializer* serializer);
};
CPositionSerializer::CPositionSerializer()
{
static const char* neededProp[] = {"InstanceId", "x", "y", "z"};
_ClassName = "Position";
_Type = ObjectTablePosition;
std::vector tmp(&(neededProp[0]), &(neededProp[4]));
_NeededProp.swap(tmp);
}
void CPositionSerializer::onSerial(NLMISC::IStream& stream, const std::string& key, CObject*& data, CObjectSerializer* serializer)
{
//H_AUTO(R2_CPositionSerializer_onSerial)
if (key == "x" || key == "y" || key == "z") { serialNumberFixedPoint(stream, data, serializer); return; }
this->CClassSerializer::onSerial(stream, key, data, serializer);
}
//--------------------------------
class CWayPointSerializer : public CClassSerializer
{
public:
CWayPointSerializer()
{
static const char* neededProp[] = {"InstanceId", "Position"};
_ClassName = "WayPoint";
_Type = ObjectTableWayPoint;
std::vector tmp(&(neededProp[0]), &(neededProp[2]));
_NeededProp.swap(tmp);
}
};
class CRegionVertexSerializer : public CClassSerializer
{
public:
CRegionVertexSerializer()
{
static const char* neededProp[] = {"InstanceId", "Position"};
_ClassName = "RegionVertex";
_Type = ObjectTableRegionVertex;
std::vector tmp(&(neededProp[0]), &(neededProp[2]));
_NeededProp.swap(tmp);
}
};
class CRegionSerializer : public CClassSerializer
{
public:
CRegionSerializer()
{
static const char* neededProp[] = {"InstanceId", "Name", "Points"};
_ClassName = "Region";
_Type = ObjectTableRegion;
std::vector tmp(&(neededProp[0]), &(neededProp[2]));
_NeededProp.swap(tmp);
}
};
class CRoadSerializer : public CClassSerializer
{
public:
CRoadSerializer()
{
static const char* neededProp[] = {"InstanceId", "Name", "Points"};
_ClassName = "Road";
_Type = ObjectTableRoad;
std::vector tmp(&(neededProp[0]), &(neededProp[2]));
_NeededProp.swap(tmp);
}
};
//------------------------------
class CNpcGrpFeatureSerializer : public CClassSerializer
{
public:
CNpcGrpFeatureSerializer()
{
_ClassName = "NpcGrpFeature";
_Type = ObjectTableNpcGrpFeature;
static const char* neededProp[] = {"InstanceId", "Name", "Components", "ActivitiesId"};
static uint32 nbNeededProp = sizeof(neededProp) / sizeof(neededProp[0]);
std::vector tmp(&(neededProp[0]), &(neededProp[nbNeededProp]));
_NeededProp.swap(tmp);
}
};
//--------------------------------------------
class CBehaviorSerializer : public CClassSerializer
{
public:
CBehaviorSerializer()
{
_ClassName = "Behavior";
_Type = ObjectTableBehavior;
static const char* neededProp[] = {"InstanceId", "Type", "ZoneId"};
static uint32 nbNeededProp = sizeof(neededProp) / sizeof(neededProp[0]);
std::vector tmp(&(neededProp[0]), &(neededProp[nbNeededProp]));
_NeededProp.swap(tmp);
}
};
//----------------------------------------------
class CRtSerializer : public CClassSerializer
{
public:
void onSerial(NLMISC::IStream& stream, const std::string& key, CObject*& data, CObjectSerializer* serializer)
{
if (key == "Id") { serialRtId(stream, data); return; }
this->CClassSerializer::onSerial(stream, key, data, serializer);
}
void serialRtId( NLMISC::IStream& stream, CObject*& data)
{
if (!stream.isReading())
{
uint32 id = 0;
std::string str = data->toString();
std::string toFind = _ClassName;
toFind += "_";
std::string::size_type pos = str.find(toFind);
if (pos != std::string::npos)
{
NLMISC::fromString(str.substr(toFind.size()), id);
}
stream.serial(id);
}
else
{
uint32 id;
stream.serial(id);
std::string strInstanceId = NLMISC::toString("%s_%d", _ClassName.c_str(), id);
data = new CObjectString(strInstanceId);
}
}
};
//----------------------------------------------
class CRtActSerializer : public CRtSerializer
{
public:
CRtActSerializer()
{
_ClassName = "RtAct";
_Type = ObjectTableRtAct;
static const char* neededProp[] = {"Id", "NpcGrps", "FaunaGrps", "AiStates", "Npcs", "Events", "Actions"};
static uint32 nbNeededProp = sizeof(neededProp) / sizeof(neededProp[0]);
std::vector tmp(&(neededProp[0]), &(neededProp[nbNeededProp]));
_NeededProp.swap(tmp);
}
};
class CRtNpcGrpSerializer : public CRtSerializer
{
public:
CRtNpcGrpSerializer()
{
_ClassName = "RtNpcGrp";
_Type = ObjectTableRtNpcGrp;
static const char* neededProp[] = {"Id", "Name", "Children", "AutoSpawn", "BotChat_parameters", "BotEquipment",
"BotSheetClient", "BotVerticalPos", "Count", "GrpKeywords", "GrpParameters"};
static uint32 nbNeededProp = sizeof(neededProp) / sizeof(neededProp[0]);
std::vector tmp(&(neededProp[0]), &(neededProp[nbNeededProp]));
_NeededProp.swap(tmp);
}
};
class CRtNpcSerializer : public CRtSerializer
{
public:
CRtNpcSerializer()
{
_ClassName = "RtNpc";
_Type = ObjectTableRtNpc;
static const char* neededProp[] = {"Id", "Name", "Children", "ChatParameters", "Equipment", "IsStuck",
"Keywords", "Keywords", "Sheet", "SheetClient", "BotVerticalPos", "Angle", "Pt"};
static uint32 nbNeededProp = sizeof(neededProp) / sizeof(neededProp[0]);
std::vector tmp(&(neededProp[0]), &(neededProp[nbNeededProp]));
_NeededProp.swap(tmp);
}
};
class CRtPositionSerializer : public CRtSerializer
{
public:
CRtPositionSerializer()
{
_ClassName = "RtPosition";
_Type = ObjectTableRtPosition;
static const char* neededProp[] = {"x", "y", "z"};
static uint32 nbNeededProp = sizeof(neededProp) / sizeof(neededProp[0]);
std::vector tmp(&(neededProp[0]), &(neededProp[nbNeededProp]));
_NeededProp.swap(tmp);
}
};
class CRtAiStateSerializer : public CRtSerializer
{
public:
CRtAiStateSerializer()
{
_ClassName = "RtAiState";
_Type = ObjectTableRtAiState;
static const char* neededProp[] = {"Id", "Name", "Children", "AiActivity", "AiMovement", "AiProfileParams", "Keywords","VerticalPos", "Pts" };
static uint32 nbNeededProp = sizeof(neededProp) / sizeof(neededProp[0]);
std::vector tmp(&(neededProp[0]), &(neededProp[nbNeededProp]));
_NeededProp.swap(tmp);
}
};
class CRtNpcEventHandlerSerializer : public CRtSerializer
{
public:
CRtNpcEventHandlerSerializer()
{
_ClassName = "RtNpcEventHandler";
_Type = ObjectTableRtNpcEventHandler;
static const char* neededProp[] = {"Id", "Name", "Event", "StatesByName", "GroupsByName", "ActionsId"};
static uint32 nbNeededProp = sizeof(neededProp) / sizeof(neededProp[0]);
std::vector tmp(&(neededProp[0]), &(neededProp[nbNeededProp]));
_NeededProp.swap(tmp);
}
};
class CRtNpcEventHandlerActionSerializer : public CRtSerializer
{
public:
CRtNpcEventHandlerActionSerializer()
{
_ClassName = "RtNpcEventHandlerAction";
_Type = ObjectTableRtNpcEventHandlerAction;
static const char* neededProp[] = {"Id", "Action", "Name", "Parameters", "Children", "Weight"};
static uint32 nbNeededProp = sizeof(neededProp) / sizeof(neededProp[0]);
std::vector tmp(&(neededProp[0]), &(neededProp[nbNeededProp]));
_NeededProp.swap(tmp);
}
};
//-------------------------------------------
/*
class CXXXSerializer : public CClassSerializer
{
public:
CNpcGrpFeatureSerializer()
{
_ClassName = "XXX";
_Type = ObjectTableXXX;
static const char* neededProp[] = {"InstanceId", "Type", "ZoneId"};
static uint32 nbNeededProp = sizeof(neededProp) / sizeof(neededProp[0]);
std::vector tmp(&(neededProp[0]), &(neededProp[nbNeededProp]));
_NeededProp.swap(tmp);
}
};
*/
//------------------------------
//--------------------------------
CObjectSerializerImpl::~CObjectSerializerImpl()
{
std::map::iterator first(_ClassSerializersById.begin()), last(_ClassSerializersById.end());
for ( ; first != last ; ++first)
{
delete first->second;
}
_ClassSerializersById.clear();
}
CObjectSerializerImpl::CObjectSerializerImpl()
{
registerSerializer( new CNpcSerializer() );
registerSerializer( new CNpcCustomSerializer() );
registerSerializer( new CNpcGrpFeatureSerializer());
registerSerializer( new CPositionSerializer());
registerSerializer( new CWayPointSerializer());
registerSerializer( new CRegionVertexSerializer());
registerSerializer( new CRegionSerializer());
registerSerializer( new CRoadSerializer());
// registerSerializer( new CNpcGrpFeatureSerializer());
registerSerializer( new CBehaviorSerializer());
registerSerializer( new CRtActSerializer());
registerSerializer( new CRtNpcGrpSerializer());
registerSerializer( new CRtNpcSerializer());
registerSerializer( new CRtAiStateSerializer());
registerSerializer( new CRtNpcEventHandlerSerializer());
registerSerializer( new CRtNpcEventHandlerActionSerializer());
registerSerializer( new CActivitySequenceSerializer());
registerSerializer( new CActivityStepSerializer());
registerSerializer( new CChatSequenceSerializer());
registerSerializer( new CChatStepSerializer());
registerSerializer( new CChatActionSerializer());
registerSerializer( new CLogicEntityActionSerializer());
registerSerializer( new CActionStepSerializer());
registerSerializer( new CActionTypeSerializer());
registerSerializer( new CEventTypeSerializer());
registerSerializer( new CLogicEntityReactionSerializer());
registerSerializer( new CTextManagerEntrySerializer());
registerSerializer( new CConditionTypeSerializer());
registerSerializer( new CConditionStepSerializer());
}
void CObjectSerializerImpl::releaseInstance()
{
if( Instance )
{
delete Instance;
Instance = NULL;
}
}
void CObjectSerializerImpl::registerSerializer(CClassSerializer* serializer)
{
bool insert = _ClassSerializersById.insert(std::make_pair(serializer->getId(), serializer)).second;
if (!insert)
{
nlinfo("R2Share: Prototype register two time");
}
insert = _ClassSerializersByName.insert(std::make_pair(serializer->getClassName(), serializer)).second;
if (!insert)
{
nlinfo("R2Share: Prototype register two time");
}
}
void CObjectSerializerImpl::serialImpl(NLMISC::IStream& stream, CObject*& data, CObjectSerializer* serializer, bool init)
{
//H_AUTO(R2_CObjectSerializerImpl_serialImpl)
uint8 type;
if (!stream.isReading())
{
if (init)
{
type = ObjectHeaderTag;
uint32 version = 1;
stream.serial(type);
stream.serial(version);
serializer->setVersion(version);
}
if (data == 0)
{
type = ObjectNull;
uint initLength = stream.getPos();
stream.serial(type);
uint endLength = stream.getPos();
if (serializer->Log) { nldebug("R2NET: (%u) Null sent %u bytes", serializer->Level, endLength - initLength);}
}
else if (data->isInteger())
{
uint initLength = stream.getPos();
writeInteger(stream, data->toInteger());
uint endLength = stream.getPos();
if (serializer->Log) { nldebug("R2NET: (%u) Integer sent %u bytes", serializer->Level, endLength - initLength); }
}
else if (data->isNumber())
{
uint initLength = stream.getPos();
writeNumber(stream, data->toNumber());
uint endLength = stream.getPos();
if (serializer->Log) { nldebug("R2NET: (%u) Number sent %u bytes", serializer->Level, endLength - initLength); }
}
else if(data->isString() || data->isRefId())
{
uint initLength = stream.getPos();
std::string value = data->toString();
uint32 size = (uint32)value.size();
if (size == 0)
{
/*if (data->isRefId())
{
nlwarning("Serializing an object of type 'empty RefId'");
}*/
type = data->isRefId() ? ObjectRefIdEmpty : ObjectStringEmpty; // NB : 'isString()' would be true in both test because CObjectRefId derives from CObjectString
stream.serial(type);
}
else if (size < static_cast(uint8Max) )
{
/*if (data->isRefId())
{
nlwarning("Serializing an object of type 'RefId8'");
}*/
type = data->isRefId() ? ObjectRefId8 : ObjectString8; // NB : 'isString()' would be true in both test because CObjectRefId derives from CObjectString
uint8 size8 = static_cast(size);
stream.serial(type);
stream.serial(size8);
stream.serialBuffer((uint8*)(&(value[0])), size8);
}
else if (size < static_cast(uint16Max) )
{
/*if (data->isRefId())
{
nlwarning("Serializing an object of type 'RefId16'");
}*/
type = data->isRefId() ? ObjectRefId16 : ObjectString16; // NB : 'isString()' would be true in both test because CObjectRefId derives from CObjectString
uint16 size16 = static_cast(size);
stream.serial(type);
stream.serial(size16);
stream.serialBuffer((uint8*)(&(value[0])), size16);
}
else
{
//default very big string
type = data->isRefId() ? ObjectRefId : ObjectString; // NB : 'isString()' would be true in both test because CObjectRefId derives from CObjectString
stream.serial(type);
stream.serial(value);
}
uint endLength = stream.getPos();
if (serializer->Log) { nldebug("R2NET: (%u) String Send %u bytes", serializer->Level, endLength - initLength); }
return;
}
else if (data->isTable())
{
uint initLength = stream.getPos();
std::string className;
if ( data->isString("Class"))
{
className = data->toString("Class");
}
if (!className.empty())
{
std::map::const_iterator found(_ClassSerializersByName.find(className));
if (found != _ClassSerializersByName.end() )
{
found->second->serialClass(stream, data, serializer);
uint endLength = stream.getPos();
if (serializer->Log) { nldebug("R2NET: (%u) Class '%s' sent %u bytes", serializer->Level, className.c_str(), endLength - initLength); }
return;
}
}
type = ObjectTable;
stream.serial(type);
uint32 size = data->getSize();
stream.serial(size);
for (uint first = 0; first != size; ++first)
{
std::string key = data->getKey(first);
CObject* value = data->getValueAtPos(first);
stream.serial(key);
++ (serializer->Level);
if (serializer->Log) { nldebug("R2NET: (%u) Field '%s'", serializer->Level, key.c_str());}
serialImpl(stream, value, serializer);
-- (serializer->Level);
}
uint endLength = stream.getPos();
if (serializer->Log)
{
if (className.empty())
{
nldebug("R2NET: (%u) Table sent %u bytes",serializer->Level, endLength - initLength);
}
else
{
nldebug("R2NET: (%u) Generic Class(%s) sent %u bytes", serializer->Level, className.c_str(), endLength - initLength);
}
}
// nlstop;
}
}
else
{
uint8 type = ObjectNull;
stream.serial(type);
// NB nico : use factory here instead of plain 'new' because client may use derived classes internally
// NB 2 : if server is created locally, not a problem because derived object fonctinnality
switch(type)
{
case ObjectHeaderTag:
{
uint32 version = 0;
stream.serial(version);
serializer->setVersion(version);
this->serialImpl(stream, data, serializer);
return;
}
case ObjectNull:
{
data = 0;
return;
}
case ObjectNumber:
{
double value;
stream.serial(value);
data = serializer->Factory ? serializer->Factory->newBasic("Number") : new CObjectNumber(0);
((CObjectNumber *) data)->set("", value);
return;
}
case ObjectNumberZero:
{
data = serializer->Factory ? serializer->Factory->newBasic("Number") : new CObjectNumber(0);
((CObjectNumber *) data)->set("", 0.0);
return;
}
case ObjectNumberSInt32:
{
sint32 value;
stream.serial(value);
data = serializer->Factory ? serializer->Factory->newBasic("Number") : new CObjectNumber(0);
((CObjectNumber *) data)->set("", value);
return;
}
case ObjectNumberUInt32:
{
uint32 value;
stream.serial(value);
data = serializer->Factory ? serializer->Factory->newBasic("Number") : new CObjectNumber(0);
((CObjectNumber *) data)->set("", value);
return;
}
case ObjectNumberSInt16:
{
sint16 value;
stream.serial(value);
data = serializer->Factory ? serializer->Factory->newBasic("Number") : new CObjectNumber(0);
((CObjectNumber *) data)->set("", value);
return;
}
case ObjectNumberUInt16:
{
uint16 value;
stream.serial(value);
data = serializer->Factory ? serializer->Factory->newBasic("Number") : new CObjectNumber(0);
((CObjectNumber *) data)->set("", value);
return;
}
case ObjectNumberSInt8:
{
sint8 value;
stream.serial(value);
data = serializer->Factory ? serializer->Factory->newBasic("Number") : new CObjectNumber(0);
((CObjectNumber *) data)->set("", value);
return;
}
case ObjectNumberUInt8:
{
uint8 value;
stream.serial(value);
data = serializer->Factory ? serializer->Factory->newBasic("Number") : new CObjectNumber(0);
((CObjectNumber *) data)->set("", value);
return;
}
// Do not remove this or it would be impossible to load old session
case ObjectNumberFloat:
{
float value;
stream.serial(value);
data = serializer->Factory ? serializer->Factory->newBasic("Number") : new CObjectNumber(0);
((CObjectNumber *) data)->set("", value);
return;
}
case ObjectNumberDouble:
{
double value;
stream.serial(value);
data = serializer->Factory ? serializer->Factory->newBasic("Number") : new CObjectNumber(0);
((CObjectNumber *) data)->set("", value);
return;
}
case ObjectStringEmpty:
{
data = serializer->Factory ? serializer->Factory->newBasic("String") : new CObjectString("");
return;
}
case ObjectString8:
{
std::string value;
uint8 size;
stream.serial(size);
value.resize(size);
stream.serialBuffer((uint8*)(&(value[0])), size);
data = serializer->Factory ? serializer->Factory->newBasic("String") : new CObjectString("");
((CObjectString *) data)->set("", value);
return;
}
case ObjectString16:
{
std::string value;
uint16 size;
stream.serial(size);
value.resize(size);
stream.serialBuffer((uint8*)(&(value[0])), size);
data = serializer->Factory ? serializer->Factory->newBasic("String") : new CObjectString("");
((CObjectString *) data)->set("", value);
break;
}
case ObjectString:
{
std::string value;
stream.serial(value);
data = serializer->Factory ? serializer->Factory->newBasic("String") : new CObjectString("");
((CObjectString *) data)->set("", value);
break;
}
case ObjectRefId:
{
std::string value;
stream.serial(value);
data = serializer->Factory ? serializer->Factory->newBasic("RefId") : new CObjectRefId("");
((CObjectRefId *) data)->set("", value);
break;
}
case ObjectRefIdEmpty:
{
data = serializer->Factory ? serializer->Factory->newBasic("RefId") : new CObjectRefId("");
return;
}
case ObjectRefId8:
{
std::string value;
uint8 size;
stream.serial(size);
value.resize(size);
stream.serialBuffer((uint8*)(&(value[0])), size);
data = serializer->Factory ? serializer->Factory->newBasic("RefId") : new CObjectRefId("");
((CObjectRefId *) data)->set("", value);
return;
}
case ObjectRefId16:
{
std::string value;
uint16 size;
stream.serial(size);
value.resize(size);
stream.serialBuffer((uint8*)(&(value[0])), size);
data = serializer->Factory ? serializer->Factory->newBasic("RefId") : new CObjectRefId("");
((CObjectRefId *) data)->set("", value);
}
break;
case ObjectTable:
{
uint32 size;
stream.serial(size);
data = newTable(serializer->Factory);
uint32 first;
for (first = 0 ; first != size; ++first)
{
std::string key;
stream.serial(key);
CObject* value=0;
serialImpl(stream,value, serializer);
data->add(key, value);
}
}
break;
default:
std::map::const_iterator found(_ClassSerializersById.find(type));
if (found != _ClassSerializersById.end())
{
found->second->serialClass(stream, data, serializer);
return;
}
BOMB("ClassSerializer not found: can not read data", return);
}
}
}
void CObjectSerializer::releaseInstance()
{
//H_AUTO(R2_CObjectSerializer_releaseInstance)
CObjectSerializerImpl::releaseInstance();
}
void CObjectSerializer::serial(NLMISC::IStream& stream)
{
//H_AUTO(R2_CObjectSerializer_serial)
stream.serial(_Compressed);
if ( stream.isReading() )
{
_MustUncompress = _Compressed;
}
if (!_Compressed)
{
CObjectSerializerImpl::getInstance().serialImpl(stream, _Data, this, true);
}
else
{
stream.serial(_CompressedLen);
stream.serial(_UncompressedLen);
if ( stream.isReading() )
{
_CompressedBuffer = new uint8[_CompressedLen];
}
stream.serialBuffer(_CompressedBuffer, _CompressedLen);
}
}
CObject* CObjectSerializer::getData() const
{
//H_AUTO(R2_CObjectSerializer_getData)
if (_Compressed && _MustUncompress) { uncompress(); };
if (_Data) return _Data->clone();
return 0;
}
CObjectSerializer::CObjectSerializer(CObjectFactory *factory, CObject* data)
: Factory(factory),
Level(0),
Log(false)
{
Log = false;
_CompressedBuffer = 0;
_CompressedLen = 0;
_UncompressedLen = 0;
_Compressed = false;
_MustUncompress = false;
_Version = 0;
if (data)
{
_Data = data->clone();
}
else
{
_Data = 0;
}
}
CObjectSerializer::~CObjectSerializer()
{
if (_CompressedBuffer) { delete [] _CompressedBuffer; _CompressedBuffer = 0;}
delete _Data;
}
CObjectSerializer::CObjectSerializer(const CObjectSerializer& /* lh */)
{
//H_AUTO(R2_CObjectSerializer_CObjectSerializer)
nlassert(0);
}
CObjectSerializer& CObjectSerializer::operator=(const CObjectSerializer& /* rh */)
{
nlassert(0);
return *this;
}
void CObjectSerializer::setData(CObject* data)
{
//H_AUTO(R2_CObjectSerializer_setData)
nlassert(!_Compressed);
if (data)
{
_Data = data->clone();
}
else
{
_Data = 0;
}
}
void CObjectSerializer::compress()
{
//H_AUTO(R2_CObjectSerializer_compress)
NLMISC::CMemStream buffer;
if (buffer.isReading()) buffer.invert();
uint32 init = buffer.length();
CObjectSerializerImpl::getInstance().serialImpl(buffer, _Data, this, true);
uint32 length = buffer.length() - init;
uLongf destLen = length + length / 1000 + 12;
Bytef *dest = new Bytef[destLen];
int ok = ::compress(dest, &destLen, (Bytef *)buffer.buffer(), buffer.length());
if (ok != Z_OK)
{
delete [] dest;
nlwarning("Error while compressing data stream");
return;
}
// Compress data only if shortest
if ( length < destLen)
{
delete [] dest;
return;
}
_Compressed = true;
_CompressedBuffer = (uint8*) dest;
_CompressedLen = destLen;
_UncompressedLen = length;
nlinfo("Compress Data from %u to %u",_UncompressedLen, _CompressedLen );
}
void CObjectSerializer::uncompress() const
{
//H_AUTO(R2_CObjectSerializer_uncompress)
const_cast(this)->uncompressImpl();
}
void CObjectSerializer::uncompressImpl()
{
//H_AUTO(R2_CObjectSerializer_uncompressImpl)
if (_Compressed && _MustUncompress)
{
_MustUncompress = false;
if ( _Data )
{
delete _Data;
_Data = 0;
}
Bytef* data = new Bytef[_UncompressedLen];
uLongf dataLen = _UncompressedLen;
sint32 state = ::uncompress (data, &dataLen ,
reinterpret_cast(_CompressedBuffer), _CompressedLen);
if (state != Z_OK)
{
delete[] data;
nlwarning("Error while uncompressing data stream.");
return;
}
if (_UncompressedLen != dataLen)
{
nlwarning("Error error in data stream.");
}
_UncompressedLen = dataLen;
NLMISC::CMemStream buffer;
if (buffer.isReading()) buffer.invert();
buffer.serialBuffer((uint8*)data, _UncompressedLen);
buffer.invert();
buffer.seek(0, NLMISC::IStream::begin);
CObjectSerializerImpl::getInstance().serialImpl(buffer, _Data, this, true);
delete[] data;
}
}
}