#include "tidentifiable.h"
#include <set>
namespace
{
class IdentifierTable
{ // singleton
unsigned long m_lastId;
std::map<unsigned long, TIdentifiable *> m_table;
std::set<TIdentifiable *> m_objects;
IdentifierTable() : m_lastId(0) {}
public:
static IdentifierTable *instance()
{
// NON DEVE MORIRE
//static IdentifierTable _instance;
//return &_instance;
static IdentifierTable *_instance = 0;
if (!_instance)
_instance = new IdentifierTable;
return _instance;
}
unsigned long getNextId()
{
return ++m_lastId;
}
void insert(TIdentifiable *o)
{
unsigned long id = o->getIdentifier();
std::map<unsigned long, TIdentifiable *>::iterator it = m_table.find(id);
if (it != m_table.end()) {
if (it->second == o)
return;
m_objects.erase(it->second);
it->second = o;
} else {
m_table[id] = o;
}
m_objects.insert(o);
}
void erase(TIdentifiable *o)
{
unsigned long id = o->getIdentifier();
m_table.erase(id);
m_objects.erase(o);
}
TIdentifiable *fetch(unsigned long id)
{
std::map<unsigned long, TIdentifiable *>::iterator it = m_table.find(id);
return it == m_table.end() ? 0 : it->second;
}
};
} // namespace
TIdentifiable::TIdentifiable()
: m_id(0)
{
}
TIdentifiable::~TIdentifiable()
{
if (m_id != 0)
IdentifierTable::instance()->erase(this);
}
TIdentifiable::TIdentifiable(const TIdentifiable &src)
: m_id(src.m_id)
{
}
const TIdentifiable &TIdentifiable::operator=(const TIdentifiable &src)
{
if (src.m_id != m_id && m_id != 0)
IdentifierTable::instance()->erase(this);
m_id = src.m_id;
return *this;
}
void TIdentifiable::setIdentifier(unsigned long id)
{
bool wasStored = m_id > 0 && IdentifierTable::instance()->fetch(m_id) == this;
if (m_id != id && m_id != 0)
IdentifierTable::instance()->erase(this);
m_id = id;
if (wasStored)
IdentifierTable::instance()->insert(this);
}
void TIdentifiable::setNewIdentifier()
{
setIdentifier(IdentifierTable::instance()->getNextId());
}
void TIdentifiable::storeByIdentifier()
{
assert(getIdentifier() >= 1);
IdentifierTable::instance()->insert(this);
}
TIdentifiable *TIdentifiable::fetchByIdentifier(unsigned long id)
{
return IdentifierTable::instance()->fetch(id);
}