]> git.tdb.fi Git - libs/gui.git/blob - source/graphics/imageloader.cpp
Make it possible to load an image into an externally allocated buffer
[libs/gui.git] / source / graphics / imageloader.cpp
1 #include <msp/core/refptr.h>
2 #include <msp/io/file.h>
3 #include <msp/strings/format.h>
4 #include "bmploader.h"
5 #include "imageloader.h"
6 #ifdef WITH_LIBPNG
7 #include "png/pngloader.h"
8 #endif
9 #ifdef WITH_LIBJPEG
10 #include "jpeg/jpegloader.h"
11 #endif
12 #ifdef WITH_DEVIL
13 #include "devil/devilloader.h"
14 #endif
15 #ifdef WITH_QUARTZ
16 #include "quartz/quartzloader.h"
17 #endif
18
19 using namespace std;
20
21 namespace Msp {
22 namespace Graphics {
23
24 ImageLoader::ImageLoader():
25         source(0),
26         state(INITIAL)
27 { }
28
29 ImageLoader::~ImageLoader()
30 {
31         delete source;
32 }
33
34 ImageLoader *ImageLoader::open_file(const string &fn)
35 {
36         try
37         {
38                 RefPtr<IO::BufferedFile> file = new IO::BufferedFile(fn);
39                 ImageLoader *loader = open_io(*file);
40                 loader->source = file.release();
41                 return loader;
42         }
43         catch(const unsupported_image_format &e)
44         {
45                 throw unsupported_image_format(format("%s: %s", fn, e.what()));
46         }
47 }
48
49 ImageLoader *ImageLoader::open_io(IO::Seekable &io)
50 {
51         Registry &registry = get_registry();
52         if(registry.changed)
53         {
54                 registry.changed = false;
55                 registry.loaders.sort(signature_size_compare);
56         }
57
58         if(registry.loaders.empty())
59                 throw unsupported_image_format("no loaders");
60
61         vector<char> sig_buf(registry.loaders.back()->get_signature_size());
62         unsigned sig_len = io.read(&sig_buf[0], sig_buf.size());
63         string signature(sig_buf.begin(), sig_buf.end());
64
65         ImageLoader *loader = 0;
66         for(list<RegisterBase *>::const_iterator i=registry.loaders.begin(); (!loader && i!=registry.loaders.end()); ++i)
67                 if((*i)->detect(signature))
68                         loader = (*i)->create(io);
69
70         io.seek(0, IO::S_BEG);
71
72         if(!loader)
73         {
74                 string sig_hex;
75                 for(unsigned i=0; i<sig_len; ++i)
76                 {
77                         if(i)
78                                 sig_hex += ' ';
79                         sig_hex += format("%02X", static_cast<unsigned char>(sig_buf[i]));
80                 }
81                 throw unsupported_image_format(sig_hex);
82         }
83
84         return loader;
85 }
86
87 void ImageLoader::load(Image::Data &data)
88 {
89         if(state>=FINISHED)
90                 throw logic_error("already loaded");
91
92         if(state<HEADERS_LOADED)
93                 load_headers_(data);
94         if(!data.pixels)
95                 data.pixels = data.owned_pixels = new char[data.stride*data.height];
96         load_pixels_(data);
97         state = FINISHED;
98 }
99
100 void ImageLoader::load_headers(Image::Data &data)
101 {
102         if(state>=HEADERS_LOADED)
103                 throw logic_error("headers already loaded");
104
105         load_headers_(data);
106         state = HEADERS_LOADED;
107 }
108
109 ImageLoader::Registry &ImageLoader::get_registry()
110 {
111         static Registry registry;
112         static bool initialized = false;
113         if(!initialized)
114         {
115                 initialized = true;
116                 register_loader<BmpLoader>();
117 #ifdef WITH_LIBPNG
118                 register_loader<PngLoader>();
119 #endif
120 #ifdef WITH_LIBJPEG
121                 register_loader<JpegLoader>();
122 #endif
123 #ifdef WITH_DEVIL
124                 register_loader<DevilLoader>();
125 #endif
126 #ifdef WITH_QUARTZ
127                 register_loader<QuartzLoader>();
128 #endif
129         }
130         return registry;
131 }
132
133 bool ImageLoader::signature_size_compare(RegisterBase *r1, RegisterBase *r2)
134 {
135         return r1->get_signature_size()<r2->get_signature_size();
136 }
137
138
139 ImageLoader::Registry::Registry():
140         changed(false)
141 { }
142
143 ImageLoader::Registry::~Registry()
144 {
145         for(list<RegisterBase *>::iterator i=loaders.begin(); i!=loaders.end(); ++i)
146                 delete *i;
147 }
148
149 } // namespace Graphics
150 } // namespace Msp