X-Git-Url: http://git.tdb.fi/?a=blobdiff_plain;f=source%2Fgraphics%2Fimageloader.cpp;h=55d2dfb1b8f8c3f4cab2ebd3c9bcd93a0a0e547e;hb=b99a9eb342d0f6ba5509c6d9f8ab0b0b5d5d2979;hp=965c7b3cb78dbaa76c659efd0ff3be60a9a00787;hpb=5715968065256daa4c43d5365cf5e268d75e007a;p=libs%2Fgui.git diff --git a/source/graphics/imageloader.cpp b/source/graphics/imageloader.cpp index 965c7b3..55d2dfb 100644 --- a/source/graphics/imageloader.cpp +++ b/source/graphics/imageloader.cpp @@ -1,8 +1,10 @@ +#include "imageloader.h" +#include #include #include #include +#include #include "bmploader.h" -#include "imageloader.h" #ifdef WITH_LIBPNG #include "png/pngloader.h" #endif @@ -21,17 +23,20 @@ using namespace std; namespace Msp { namespace Graphics { -bool ImageLoader::registered_loaders_changed = false; - -ImageLoader::ImageLoader(): - source(0) -{ } - ImageLoader::~ImageLoader() { delete source; } +bool ImageLoader::detect_signature(const string &sig) +{ + Registry ®istry = get_registry(); + for(const RegisterBase *r: registry.loaders) + if(r->detect(sig)) + return true; + return false; +} + ImageLoader *ImageLoader::open_file(const string &fn) { try @@ -49,36 +54,23 @@ ImageLoader *ImageLoader::open_file(const string &fn) ImageLoader *ImageLoader::open_io(IO::Seekable &io) { - (void)RegisteredImageLoader::reg; -#ifdef WITH_LIBPNG - (void)RegisteredImageLoader::reg; -#endif -#ifdef WITH_LIBJPEG - (void)RegisteredImageLoader::reg; -#endif -#ifdef WITH_DEVIL - (void)RegisteredImageLoader::reg; -#endif -#ifdef WITH_QUARTZ - (void)RegisteredImageLoader::reg; -#endif - - list &loaders = get_registered_loaders(); - if(registered_loaders_changed) + Registry ®istry = get_registry(); + if(registry.changed) { - registered_loaders_changed = false; - loaders.sort(signature_size_compare); + registry.changed = false; + sort(registry.loaders, [](RegisterBase *r1, RegisterBase *r2){ + return r1->get_signature_size()get_signature_size(); + }); } - if(loaders.empty()) + if(registry.loaders.empty()) throw unsupported_image_format("no loaders"); - vector sig_buf(loaders.back()->get_signature_size()); - unsigned sig_len = io.read(&sig_buf[0], sig_buf.size()); - string signature(sig_buf.begin(), sig_buf.end()); + string signature(registry.loaders.back()->get_signature_size(), 0); + unsigned sig_len = io.read(&signature[0], signature.size()); - ImageLoader *loader = 0; - for(list::const_iterator i=loaders.begin(); (!loader && i!=loaders.end()); ++i) + ImageLoader *loader = nullptr; + for(auto i=registry.loaders.begin(); (!loader && i!=registry.loaders.end()); ++i) if((*i)->detect(signature)) loader = (*i)->create(io); @@ -88,34 +80,64 @@ ImageLoader *ImageLoader::open_io(IO::Seekable &io) { string sig_hex; for(unsigned i=0; i(sig_buf[i])); - } + append(sig_hex, " ", format("%02X", static_cast(signature[i]))); throw unsupported_image_format(sig_hex); } return loader; } - -ImageLoader::RegisterBase::RegisterBase() +void ImageLoader::load(Image::Data &data) { - get_registered_loaders().push_back(this); - registered_loaders_changed = true; + if(state>=FINISHED) + throw logic_error("already loaded"); + + if(state=HEADERS_LOADED) + throw logic_error("headers already loaded"); + + load_headers_(data); + state = HEADERS_LOADED; +} -list &ImageLoader::get_registered_loaders() +ImageLoader::Registry &ImageLoader::get_registry() { - static list regs; - return regs; + static Registry registry; + static bool initialized = false; + if(!initialized) + { + initialized = true; + register_loader(); +#ifdef WITH_LIBPNG + register_loader(); +#endif +#ifdef WITH_LIBJPEG + register_loader(); +#endif +#ifdef WITH_DEVIL + register_loader(); +#endif +#ifdef WITH_QUARTZ + register_loader(); +#endif + } + return registry; } -bool ImageLoader::signature_size_compare(RegisterBase *r1, RegisterBase *r2) + +ImageLoader::Registry::~Registry() { - return r1->get_signature_size()get_signature_size(); + for(auto l: loaders) + delete l; } } // namespace Graphics