X-Git-Url: http://git.tdb.fi/?a=blobdiff_plain;f=source%2Fcollection.h;h=fa3a7de9dbe25441417f988fbe16a5235cbf638d;hb=d84673be1c2fe8bd32ddaa7877529855cad6daa0;hp=6ddbe6ed65c980d035b30160ad1eca18c0390f1f;hpb=9c0da3059eed8470325477d99a0d56f0948534fd;p=libs%2Fdatafile.git diff --git a/source/collection.h b/source/collection.h index 6ddbe6e..fa3a7de 100644 --- a/source/collection.h +++ b/source/collection.h @@ -72,6 +72,7 @@ private: */ struct ItemKeywordBase { + virtual ~ItemKeywordBase() { } virtual void add_to_loader(Loader &) const { }; }; @@ -105,12 +106,15 @@ private: virtual ~ItemCreatorBase() { } template - S *create(Collection &coll, const std::string &name) + bool create(Collection &coll, const std::string &name, S *&ptr) { ItemCreatorBridge *creator=dynamic_cast *>(this); if(creator) - return creator->create(coll, name); - return 0; + { + ptr=creator->create(coll, name); + return true; + } + return false; } }; @@ -189,7 +193,7 @@ public: void add(const std::string &name, T *d) { if(items.count(name)) - throw KeyError("Duplicate key '"+name+"' in collection"); + throw KeyError("Duplicate key in collection", name); items[name]=new Item::Type>(d); } @@ -198,19 +202,19 @@ public: Gets an object of a specific type from the collection. */ template - T &get(const std::string &name) const + T *get(const std::string &name) const { typedef typename RemoveConst::Type NCT; ItemMap::const_iterator i=items.find(name); if(i==items.end()) - throw KeyError("Item '"+name+"' not found in collection"); + throw KeyError("Item not found in collection", name); const Item *item=dynamic_cast *>(i->second); if(!item) - throw TypeError("Item '"+name+"' is not of correct type"); + throw TypeError("Type mismatch on item '"+name+"'"); - return *item->data; + return item->data; } /** @@ -219,7 +223,7 @@ public: invoked. */ template - T &get(const std::string &name) + T *get(const std::string &name) { typedef typename RemoveConst::Type NCT; @@ -227,20 +231,23 @@ public: if(i==items.end()) { for(ItemCreatorSeq::iterator j=creators.begin(); j!=creators.end(); ++j) - if(NCT *d=(*j)->create(*this, name)) + { + NCT *d=0; + if((*j)->create(*this, name, d)) { // We already know that the item didn't exist yet items[name]=new Item(d); - return *d; + return d; } - throw KeyError("Item '"+name+"' not found in collection"); + } + throw KeyError("Item not found in collection", name); } const Item *item=dynamic_cast *>(i->second); if(!item) - throw TypeError("Item '"+name+"' is not of correct type"); + throw TypeError("Type mismatch on item '"+name+"'"); - return *item->data; + return item->data; } /** @@ -277,6 +284,22 @@ public: */ bool contains(const std::string &n) const; + /** + Returns the name of an item in the collection. + */ + template + const std::string &get_name(T *d) const + { + typedef typename RemoveConst::Type NCT; + + for(ItemMap::const_iterator i=items.begin(); i!=items.end(); ++i) + if(Item *item=dynamic_cast *>(i->second)) + if(item->data==d) + return i->first; + + throw KeyError("Item not found in collection"); + } + protected: /** Adds a type that can be loaded from datafiles.