X-Git-Url: http://git.tdb.fi/?a=blobdiff_plain;f=source%2Fgraphics%2Fimageloader.cpp;h=55d2dfb1b8f8c3f4cab2ebd3c9bcd93a0a0e547e;hb=b99a9eb342d0f6ba5509c6d9f8ab0b0b5d5d2979;hp=6cdaef912e65062740c284559b3851c742bd102b;hpb=0466fc5fe92ae03189f45f8872fb070ef7022290;p=libs%2Fgui.git diff --git a/source/graphics/imageloader.cpp b/source/graphics/imageloader.cpp index 6cdaef9..55d2dfb 100644 --- a/source/graphics/imageloader.cpp +++ b/source/graphics/imageloader.cpp @@ -1,28 +1,42 @@ +#include "imageloader.h" +#include #include #include #include -#include "imageloader.h" +#include +#include "bmploader.h" #ifdef WITH_LIBPNG #include "png/pngloader.h" #endif +#ifdef WITH_LIBJPEG +#include "jpeg/jpegloader.h" +#endif #ifdef WITH_DEVIL #include "devil/devilloader.h" #endif +#ifdef WITH_QUARTZ +#include "quartz/quartzloader.h" +#endif using namespace std; namespace Msp { namespace Graphics { -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 @@ -40,37 +54,91 @@ ImageLoader *ImageLoader::open_file(const string &fn) ImageLoader *ImageLoader::open_io(IO::Seekable &io) { - char sig_buf[8]; - unsigned sig_len = io.read(sig_buf, sizeof(sig_buf)); - string sig(sig_buf, sig_len); + Registry ®istry = get_registry(); + if(registry.changed) + { + registry.changed = false; + sort(registry.loaders, [](RegisterBase *r1, RegisterBase *r2){ + return r1->get_signature_size()get_signature_size(); + }); + } + + if(registry.loaders.empty()) + throw unsupported_image_format("no loaders"); + + string signature(registry.loaders.back()->get_signature_size(), 0); + unsigned sig_len = io.read(&signature[0], signature.size()); + + ImageLoader *loader = nullptr; + for(auto i=registry.loaders.begin(); (!loader && i!=registry.loaders.end()); ++i) + if((*i)->detect(signature)) + loader = (*i)->create(io); + io.seek(0, IO::S_BEG); - // TODO register loader classes automatically - ImageLoader *loader = 0; - if(0) - ; -#ifdef WITH_LIBPNG - else if(PngLoader::detect(sig)) - loader = new PngLoader(io); -#endif -#ifdef WITH_DEVIL - else if(DevilLoader::detect(sig)) - loader = new DevilLoader(io); -#endif - else + if(!loader) { 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; } +void ImageLoader::load(Image::Data &data) +{ + if(state>=FINISHED) + throw logic_error("already loaded"); + + if(state=HEADERS_LOADED) + throw logic_error("headers already loaded"); + + load_headers_(data); + state = HEADERS_LOADED; +} + +ImageLoader::Registry &ImageLoader::get_registry() +{ + 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; +} + + +ImageLoader::Registry::~Registry() +{ + for(auto l: loaders) + delete l; +} + } // namespace Graphics } // namespace Msp