]> git.tdb.fi Git - libs/datafile.git/commitdiff
Rework the list and containment queries
authorMikko Rasa <tdb@tdb.fi>
Wed, 5 Dec 2012 19:16:00 +0000 (21:16 +0200)
committerMikko Rasa <tdb@tdb.fi>
Wed, 5 Dec 2012 19:16:00 +0000 (21:16 +0200)
Most of the implementations are now in .cpp, and they should deal better
with base classes.

source/collection.cpp
source/collection.h

index 2333aaf2726082a6214d2b950aa4570ebc1ea64b..fbe894a8bb4b073c0baa227655bd2d3ecf3781e3 100644 (file)
@@ -35,6 +35,35 @@ const Variant &Collection::get_var(const string &name, const CollectionItemTypeB
        return get_item(items, name);
 }
 
+void Collection::gather_items(list<const Variant *> *vars, list<string> *names, const CollectionItemTypeBase &type, bool include_sources) const
+{
+       for(ItemMap::const_iterator i=items.begin(); i!=items.end(); ++i)
+               if(type.check_item_type(i->second))
+               {
+                       if(vars)
+                               vars->push_back(&i->second);
+                       if(names)
+                               names->push_back(i->first);
+               }
+
+       if(include_sources && names)
+               gather_names_from_sources(*names, type);
+}
+
+unsigned Collection::get_status(const string &name, const CollectionItemTypeBase &type) const
+{
+       ItemMap::const_iterator i = items.find(name);
+       if(i==items.end())
+       {
+               for(SourceList::const_iterator j=sources.begin(); j!=sources.end(); ++j)
+                       if((*j)->is_loadable(type, name))
+                               return 2;
+               return 0;
+       }
+
+       return type.check_item_type(i->second);
+}
+
 CollectionItemTypeBase *Collection::get_type_for_item(const Variant &var) const
 {
        for(TypeList::const_iterator i=types.begin(); i!=types.end(); ++i)
@@ -48,6 +77,28 @@ void Collection::add_source(CollectionSource &s)
        sources.push_back(&s);
 }
 
+void Collection::gather_names_from_sources(list<string> &names, const CollectionItemTypeBase &type) const
+{
+       for(SourceList::const_iterator i=sources.begin(); i!=sources.end(); ++i)
+       {
+               std::list<std::string> available_names = (*i)->get_names(type);
+               for(std::list<std::string>::iterator j=available_names.begin(); j!=available_names.end(); ++j)
+                       if(!items.count(*j))
+                               names.push_back(*j);
+       }
+}
+
+void Collection::load_items_from_sources(const CollectionItemTypeBase &type)
+{
+       for(SourceList::const_iterator i=sources.begin(); i!=sources.end(); ++i)
+       {
+               std::list<std::string> available_names = (*i)->get_names(type);
+               for(std::list<std::string>::iterator j=available_names.begin(); j!=available_names.end(); ++j)
+                       if(!items.count(*j))
+                               (*i)->load(*this, type, *j);
+       }
+}
+
 
 Collection::Loader::Loader(Collection &c):
        coll(c)
index 6e6900cece1b0c3428e666e918e2e901b6a4541f..d5853f2ea6cb4a3fba658dd5912e148f419cb8e4 100644 (file)
@@ -138,30 +138,23 @@ private:
        T &extract(const Variant &var) const;
 
        template<typename T>
-       void collect_items(std::list<T *> *objects, std::list<std::string> *names, std::list<std::string> *future_names) const
+       std::list<T *> extract_list(const std::list<const Variant *> &vars) const
        {
-               typedef RefPtr<typename RemoveConst<T>::Type> RPNCT;
+               std::list<T *> result;
+               for(std::list<const Variant *>::const_iterator i=vars.begin(); i!=vars.end(); ++i)
+                       result.push_back(&extract<T>(**i));
+               return result;
+       }
 
-               for(ItemMap::const_iterator i=items.begin(); i!=items.end(); ++i)
-                       if(i->second.check_type<RPNCT>())
-                       {
-                               if(objects)
-                                       objects->push_back(i->second.value<RPNCT>().get());
-                               if(names)
-                                       names->push_back(i->first);
-                       }
-
-               if(future_names)
-                       if(CollectionItemTypeBase *type = get_type<T>())
-                       {
-                               for(SourceList::const_iterator i=sources.begin(); i!=sources.end(); ++i)
-                               {
-                                       std::list<std::string> available_names = (*i)->get_names(*type);
-                                       for(std::list<std::string>::iterator j=available_names.begin(); j!=available_names.end(); ++j)
-                                               if(!items.count(*j))
-                                                       future_names->push_back(*j);
-                               }
-                       }
+       void gather_items(std::list<const Variant *> *, std::list<std::string> *, const CollectionItemTypeBase &, bool) const;
+
+       template<typename T>
+       void gather_items(std::list<const Variant *> *vars, std::list<std::string> *names, const CollectionItemTypeBase *type, bool include_sources) const
+       {
+               if(type || (type = get_type<T>()))
+                       gather_items(vars, names, *type, include_sources);
+               else
+                       gather_items(vars, names, CollectionItemType<T>(), false);
        }
 
 public:
@@ -169,9 +162,9 @@ public:
        template<typename T>
        std::list<std::string> get_names() const
        {
-               std::list<std::string> result;
-               collect_items<T>(0, &result, 0);
-               return result;
+               std::list<std::string> names;
+               gather_items<typename RemoveConst<T>::Type>(0, &names, 0, false);
+               return names;
        }
 
        /** Returns a list of the names of objects of one type in the collection or
@@ -179,18 +172,18 @@ public:
        template<typename T>
        std::list<std::string> get_names()
        {
-               std::list<std::string> result;
-               collect_items<T>(0, &result, &result);
-               return result;
+               std::list<std::string> names;
+               gather_items<typename RemoveConst<T>::Type>(0, &names, 0, true);
+               return names;
        }
 
        /// Returns a list of objects of one type in the collection.
        template<typename T>
        std::list<T *> get_list() const
        {
-               std::list<T *> result;
-               collect_items<T>(&result, 0, 0);
-               return result;
+               std::list<const Variant *> vars;
+               gather_items<typename RemoveConst<T>::Type>(&vars, 0, 0, false);
+               return extract_list<T>(vars);
        }
 
        /** Returns a list of objects of one type, loading them from sources if
@@ -198,48 +191,40 @@ public:
        template<typename T>
        std::list<T *> get_list()
        {
-               std::list<T *> result;
-               std::list<std::string> future;
-               collect_items<T>(&result, 0, &future);
-               for(std::list<std::string>::iterator i=future.begin(); i!=future.end(); ++i)
-                       result.push_back(&get<T>(*i));
-               return result;
+               CollectionItemTypeBase *type = get_type<typename RemoveConst<T>::Type>();
+               if(type)
+                       load_items_from_sources(*type);
+
+               std::list<const Variant *> vars;
+               gather_items<typename RemoveConst<T>::Type>(&vars, 0, type, true);
+               return extract_list<T>(vars);
        }
 
 private:
+       unsigned get_status(const std::string &, const CollectionItemTypeBase &) const;
+
        template<typename T>
        unsigned get_status(const std::string &name) const
        {
-               ItemMap::const_iterator i = items.find(name);
-               if(i==items.end())
-               {
-                       if(CollectionItemTypeBase *type = get_type<T>())
-                       {
-                               for(SourceList::const_iterator j=sources.begin(); j!=sources.end(); ++j)
-                                       if((*j)->is_loadable(*type, name))
-                                               return 2;
-                       }
-                       return 0;
-               }
-
-               typedef RefPtr<typename RemoveConst<T>::Type> RPNCT;
-               if(!i->second.check_type<RPNCT>())
-                       return 0;
+               // XXX Should go through all applicable types
+               if(CollectionItemTypeBase *type = get_type<T>())
+                       return get_status(name, *type);
 
-               return 1;
+               ItemMap::const_iterator i = items.find(name);
+               return (i!=items.end() && i->second.check_type<RefPtr<T> >());
        }
 
 public:
        /// Checks whether a typed object exists in the collection.
        template<typename T>
        bool contains(const std::string &name) const
-       { return get_status<T>(name)==1; }
+       { return get_status<typename RemoveConst<T>::Type>(name)==1; }
 
        /** Checks whether a typed object exists in the collection or is loadable
        from a source. */
        template<typename T>
        bool contains(const std::string &name)
-       { return get_status<T>(name)>0; }
+       { return get_status<typename RemoveConst<T>::Type>(name)>0; }
 
        /// Returns the name of an item in the collection.
        template<typename T>
@@ -270,6 +255,10 @@ protected:
        CollectionItemTypeBase *get_type_for_item(const Variant &) const;
 
        void add_source(CollectionSource &);
+
+       void gather_names_from_sources(std::list<std::string> &, const CollectionItemTypeBase &) const;
+
+       void load_items_from_sources(const CollectionItemTypeBase &);
 };
 
 template<typename T>