163 lines
3.8 KiB
C++
163 lines
3.8 KiB
C++
/*
|
|
Copyright (c) 2013 yvt
|
|
|
|
This file is part of OpenSpades.
|
|
|
|
OpenSpades is free software: you can redistribute it and/or modify
|
|
it under the terms of the GNU General Public License as published by
|
|
the Free Software Foundation, either version 3 of the License, or
|
|
(at your option) any later version.
|
|
|
|
OpenSpades is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with OpenSpades. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
//#include "FltkImageReader.h"
|
|
#include "IBitmapCodec.h"
|
|
#include "Debug.h"
|
|
#include "Exception.h"
|
|
#include "IStream.h"
|
|
#include "Bitmap.h"
|
|
#include <FL/Fl_PNG_Image.H>
|
|
#include <FL/Fl_JPEG_Image.H>
|
|
#include <string.h>
|
|
|
|
namespace spades {
|
|
class FltkImageReader: public IBitmapCodec {
|
|
public:
|
|
virtual std::string GetName() = 0;
|
|
|
|
virtual Fl_RGB_Image *LoadFltkImage(const std::string& data) = 0;
|
|
|
|
virtual bool CanLoad() { return true; }
|
|
virtual bool CanSave() { return false; }
|
|
|
|
virtual bool CheckExtension(const std::string&) = 0;
|
|
|
|
virtual Bitmap *Load(IStream *stream){
|
|
SPADES_MARK_FUNCTION();
|
|
|
|
// read all
|
|
std::string data = stream->ReadAllBytes();
|
|
|
|
// copy to buffer
|
|
Fl_Image *img = LoadFltkImage(data);
|
|
|
|
SPAssert(img);
|
|
SPAssert(img->count() >= 1);
|
|
|
|
const unsigned char* inPixels =
|
|
(const unsigned char *)img->data()[0];
|
|
int depth = img->d();
|
|
int width = img->w();
|
|
int height = img->h();
|
|
int pitch = width * depth + img->ld();
|
|
|
|
Handle<Bitmap> bmp;
|
|
try{
|
|
bmp.Set(new Bitmap(width, height), false);
|
|
}catch(...){
|
|
delete img;
|
|
throw;
|
|
}
|
|
try{
|
|
unsigned char *outPixels = (unsigned char *)bmp->GetPixels();
|
|
|
|
if(pitch == width * 4 && depth == 4){
|
|
// if the format matches the requirement of Bitmap,
|
|
// just use it
|
|
memcpy(outPixels, inPixels, pitch * height);
|
|
} else {
|
|
// convert
|
|
const unsigned char* line;
|
|
for(int y = 0; y < height; y++){
|
|
line = inPixels;
|
|
for(int x = 0; x < width; x++){
|
|
uint8_t r, g, b, a;
|
|
switch(depth){
|
|
case 1:
|
|
r = g = b = *(line++);
|
|
a = 255;
|
|
break;
|
|
case 2:
|
|
r = g = b = *(line++);
|
|
a = *(line++);
|
|
break;
|
|
case 3:
|
|
r = *(line++);
|
|
g = *(line++);
|
|
b = *(line++);
|
|
a = 255;
|
|
break;
|
|
case 4:
|
|
r = *(line++);
|
|
g = *(line++);
|
|
b = *(line++);
|
|
a = *(line++);
|
|
break;
|
|
default:
|
|
SPAssert(false);
|
|
}
|
|
*(outPixels++) = r;
|
|
*(outPixels++) = g;
|
|
*(outPixels++) = b;
|
|
*(outPixels++) = a;
|
|
}
|
|
inPixels += pitch;
|
|
}
|
|
}
|
|
delete img;
|
|
return bmp.Unmanage();
|
|
}catch(...){
|
|
delete img;
|
|
throw;
|
|
}
|
|
}
|
|
|
|
virtual void Save(IStream *, Bitmap *) {
|
|
SPADES_MARK_FUNCTION();
|
|
SPNotImplemented();
|
|
}
|
|
};
|
|
|
|
class FltkPngReader: public FltkImageReader {
|
|
public:
|
|
virtual std::string GetName(){
|
|
return "FLTK PNG Reader";
|
|
}
|
|
|
|
virtual Fl_RGB_Image *LoadFltkImage(const std::string& data) {
|
|
return new Fl_PNG_Image(NULL, (const unsigned char *)data.data(),
|
|
data.size());
|
|
}
|
|
|
|
virtual bool CheckExtension(const std::string& fn) {
|
|
return EndsWith(fn, ".png");
|
|
}
|
|
} pngReader;
|
|
|
|
class FltkJpegReader: public FltkImageReader {
|
|
public:
|
|
virtual std::string GetName(){
|
|
return "FLTK JPEG Reader";
|
|
}
|
|
|
|
virtual Fl_RGB_Image *LoadFltkImage(const std::string& data) {
|
|
return new Fl_JPEG_Image(NULL, (const unsigned char *)data.data());
|
|
}
|
|
|
|
virtual bool CheckExtension(const std::string& fn) {
|
|
return EndsWith(fn, ".jpg") ||
|
|
EndsWith(fn, ".jpeg") || EndsWith(fn, ".jpe");
|
|
}
|
|
} jpegReader;
|
|
|
|
}
|
|
|