1 #include <msp/io/slice.h>
2 #include <msp/strings/format.h>
3 #include <msp/strings/regex.h>
4 #include "collection.h"
5 #include "packsource.h"
11 void delete_io(Msp::IO::Base *io)
21 void PackSource::add_pack_file(const string &fn)
23 add_pack_file(fn, string());
26 void PackSource::add_pack_file(const string &fn, const string &filter)
29 for(list<Pack>::iterator i=packs.begin(); (!pack && i!=packs.end()); ++i)
30 if(i->get_filename()==fn)
34 packs.push_back(Pack(fn));
36 DataFile::load(*pack, fn);
40 pack->collect_files(pack_files, filter);
41 files.insert(pack_files.begin(), pack_files.end());
42 for(FileMap::const_iterator i=pack_files.begin(); i!=pack_files.end(); ++i)
43 i->second->collect_objects(objects);
46 list<PackSource::FileInfo> PackSource::list_files() const
48 list<FileInfo> result;
49 for(FileMap::const_iterator i=files.begin(); i!=files.end(); ++i)
50 result.push_back(i->second->get_info());
54 bool PackSource::is_loadable(const CollectionItemTypeBase &type, const string &name) const
56 ObjectMap::const_iterator i = objects.find(name);
60 // If the object has a keyword, it must match that of the type
61 if(!i->second->get_keyword().empty() && i->second->get_keyword()!=type.get_keyword())
67 CollectionSource::NameList PackSource::get_names(const CollectionItemTypeBase &type) const
70 for(ObjectMap::const_iterator i=objects.begin(); i!=objects.end(); ++i)
72 if(!i->second->get_keyword().empty())
74 if(i->second->get_keyword()!=type.get_keyword())
77 else if(!type.match_name(i->first))
80 names.push_back(i->first);
86 void PackSource::load(Collection &coll, const CollectionItemTypeBase &type, const string &name) const
88 ObjectMap::const_iterator i = objects.find(name);
92 File &file = i->second->get_file();
97 RefPtr<IO::Base> in = file.open();
98 Parser parser(*in, file.get_full_name());
99 if(file.is_collection())
101 Collection::Loader ldr(coll);
105 type.load_item(coll, parser, name);
108 IO::Seekable *PackSource::open(const string &fn) const
110 FileMap::const_iterator i = files.find(fn);
112 return i->second->open().release();
118 PackSource::Pack::Pack(const string &fn):
123 void PackSource::Pack::collect_files(FileMap &fm, const string &filter) const
127 for(list<File>::const_iterator i=files.begin(); i!=files.end(); ++i)
128 fm[i->get_filename()] = &*i;
133 for(list<File>::const_iterator i=files.begin(); i!=files.end(); ++i)
134 if(re.match(i->get_filename()))
135 fm[i->get_filename()] = &*i;
140 PackSource::File::File(const Pack &p, const string &fn):
149 RefPtr<IO::Seekable> PackSource::File::open() const
151 IO::BufferedFile *io_file = new IO::BufferedFile(pack.get_filename());
152 IO::Slice *io_slice = new IO::Slice(*io_file, pack.get_base_offset()+offset, length);
153 io_slice->signal_deleted.connect(sigc::bind(sigc::ptr_fun(delete_io), io_file));
157 PackSource::FileInfo PackSource::File::get_info() const
160 info.name = filename;
165 string PackSource::File::get_full_name() const
167 return format("%s/%s", pack.get_filename(), filename);
170 void PackSource::File::set_loaded()
175 void PackSource::File::collect_objects(ObjectMap &objs) const
177 for(list<Object>::const_iterator i=objects.begin(); i!=objects.end(); ++i)
178 objs[i->get_name()] = &*i;
182 PackSource::Object::Object(File &f, const string &n, const string &k):
189 PackSource::Pack::Loader::Loader(Pack &p):
190 ObjectLoader<Pack>(p)
192 add("file", &Loader::file);
193 add("base_offset", &Pack::base_offset);
196 void PackSource::Pack::Loader::file(const string &fn)
198 obj.files.push_back(File(obj, fn));
199 load_sub(obj.files.back());
203 PackSource::File::Loader::Loader(File &f):
204 ObjectLoader<File>(f)
206 add("object", &Loader::object);
207 add("slice", &File::offset, &File::length);
210 void PackSource::File::Loader::finish()
214 PackSource::Object ob(obj, obj.filename, string());
215 obj.objects.push_back(ob);
219 void PackSource::File::Loader::object(const string &name, const string &kwd)
221 obj.objects.push_back(PackSource::Object(obj, name, kwd));
222 obj.collection = true;
225 } // namespace DataFile