/* * * (C) 2003-2014 Anope Team * Contact us at team@anope.org * * Please read COPYING and README for further details. * */ #pragma once #include "anope.h" #include "service.h" #include "logger.h" class Extensible; class CoreExport ExtensibleBase : public Service { protected: std::map items; ExtensibleBase(Module *m, const Anope::string &n); ExtensibleBase(Module *m, const Anope::string &t, const Anope::string &n); ~ExtensibleBase(); public: virtual void Unset(Extensible *obj) anope_abstract; }; class CoreExport Extensible { public: std::vector extension_items; virtual ~Extensible(); template T* GetExt(const Anope::string &name); bool HasExtOK(const Anope::string &name); template T* Extend(const Anope::string &name, const T &what); template void ShrinkOK(const Anope::string &name); }; template class ExtensibleItem : public ExtensibleBase { public: ExtensibleItem(Module *m, const Anope::string &n) : ExtensibleBase(m, n) { } ExtensibleItem(Module *m, const Anope::string &t, const Anope::string &n) : ExtensibleBase(m, t, n) { } ~ExtensibleItem() { while (!items.empty()) { std::map::iterator it = items.begin(); Extensible *obj = it->first; T *value = static_cast(it->second); auto it2 = std::find(obj->extension_items.begin(), obj->extension_items.end(), this); if (it2 != obj->extension_items.end()) obj->extension_items.erase(it2); items.erase(it); delete value; } } T* Set(Extensible *obj, const T &value) { T* t = new T(value); Unset(obj); items[obj] = t; obj->extension_items.push_back(this); return t; } void Unset(Extensible *obj) override { T *value = Get(obj); items.erase(obj); auto it = std::find(obj->extension_items.begin(), obj->extension_items.end(), this); if (it != obj->extension_items.end()) obj->extension_items.erase(it); delete value; } T* Get(Extensible *obj) { std::map::const_iterator it = items.find(obj); if (it != items.end()) return static_cast(it->second); return nullptr; } bool HasExt(Extensible *obj) { return items.find(obj) != items.end(); } T* Require(Extensible *obj) { T* t = Get(obj); if (t) return t; return Set(obj, T()); } }; template struct ExtensibleRef : ServiceReference> { ExtensibleRef(const Anope::string &n) : ServiceReference>("Extensible", n) { } ExtensibleRef(const Anope::string &t, const Anope::string &n) : ServiceReference>(t, n) { } }; template T* Extensible::GetExt(const Anope::string &name) { ExtensibleRef ref(name); if (ref) return ref->Get(this); Log(LOG_DEBUG) << "GetExt for nonexistent type " << name << " on " << static_cast(this); return NULL; } template T* Extensible::Extend(const Anope::string &name, const T &what) { ExtensibleRef ref(name); if (ref) { ref->Set(this, what); return ref->Get(this); } Log(LOG_DEBUG) << "Extend for nonexistent type " << name << " on " << static_cast(this); return NULL; } template //XXX void Extensible::ShrinkOK(const Anope::string &name) { ExtensibleRef ref(name); if (ref) ref->Unset(this); else Log(LOG_DEBUG) << "Shrink for nonexistent type " << name << " on " << static_cast(this); }