- Reestructuración de ficheros y directorios general
- merge v0.01 --> Añadido fileselector - Añadidas fuentes de Gem y Pure Data - pix2jpg incluído en Gem. Archivos de construcción de Gem modificados. - Añadido fichero ompiling.txt con instrucciones de compilación
This commit is contained in:
parent
c9adfd020b
commit
e85d191b46
3100 changed files with 775434 additions and 3073 deletions
551
Gem/plugins/imageQT/imageQT.cpp
Normal file
551
Gem/plugins/imageQT/imageQT.cpp
Normal file
|
@ -0,0 +1,551 @@
|
|||
////////////////////////////////////////////////////////
|
||||
//
|
||||
// GEM - Graphics Environment for Multimedia
|
||||
//
|
||||
// zmoelnig@iem.kug.ac.at
|
||||
//
|
||||
// Implementation file
|
||||
//
|
||||
// Copyright (c) 1997-1999 Mark Danks.
|
||||
// Copyright (c) Günther Geiger.
|
||||
// Copyright (c) 2001-2011 IOhannes m zmölnig. forum::für::umläute. IEM. zmoelnig@iem.at
|
||||
// For information on usage and redistribution, and for a DISCLAIMER OF ALL
|
||||
// WARRANTIES, see the file, "GEM.LICENSE.TERMS" in this distribution.
|
||||
//
|
||||
/////////////////////////////////////////////////////////
|
||||
#ifdef HAVE_CONFIG_H
|
||||
# include "config.h"
|
||||
#endif
|
||||
|
||||
|
||||
#if defined __APPLE__
|
||||
# if !defined __x86_64__
|
||||
// with OSX10.6, apple has removed loads of Carbon functionality (in 64bit mode)
|
||||
// LATER make this a real check in configure
|
||||
# define HAVE_CARBONQUICKTIME
|
||||
# elif defined HAVE_QUICKTIME
|
||||
# undef HAVE_QUICKTIME
|
||||
# endif
|
||||
#endif
|
||||
|
||||
#ifdef HAVE_QUICKTIME
|
||||
# ifdef HAVE_CARBONQUICKTIME
|
||||
//
|
||||
# elif defined _WIN32
|
||||
//
|
||||
# else
|
||||
# undef HAVE_QUICKTIME
|
||||
# endif
|
||||
#endif
|
||||
|
||||
#ifdef HAVE_QUICKTIME
|
||||
|
||||
# include "Gem/RTE.h"
|
||||
# include "imageQT.h"
|
||||
#include "plugins/PluginFactory.h"
|
||||
|
||||
# ifdef HAVE_CARBONQUICKTIME
|
||||
# include <Carbon/Carbon.h>
|
||||
# include <QuickTime/QuickTime.h>
|
||||
# include <QuickTime/ImageCompression.h>
|
||||
# define IMAGEQT_RGBA_PIXELFORMAT k32ARGBPixelFormat
|
||||
# elif defined _WIN32
|
||||
# include <QTML.h>
|
||||
# include <Movies.h>
|
||||
# include <QuickTimeComponents.h>
|
||||
# include "Gem/Exception.h"
|
||||
# define OffsetRect MacOffsetRect
|
||||
# define IMAGEQT_RGBA_PIXELFORMAT k32RGBAPixelFormat
|
||||
# endif
|
||||
|
||||
#ifdef _MSC_VER /* This is only for Microsoft's compiler, not cygwin, e.g. */
|
||||
# define snprintf _snprintf
|
||||
# define vsnprintf _vsnprintf
|
||||
#endif
|
||||
|
||||
#define QT_MAX_FILENAMELENGTH 256
|
||||
#include <stdio.h>
|
||||
# include <map>
|
||||
|
||||
//# include <string.h>
|
||||
//# include <fcntl.h>
|
||||
|
||||
using namespace gem::plugins;
|
||||
|
||||
|
||||
REGISTER_IMAGELOADERFACTORY("QT", imageQT);
|
||||
REGISTER_IMAGESAVERFACTORY("QT", imageQT);
|
||||
|
||||
|
||||
#if defined __APPLE__
|
||||
static OSStatus
|
||||
FSPathMakeFSSpec(
|
||||
const UInt8 *path,
|
||||
FSSpec *spec)
|
||||
{
|
||||
OSStatus result;
|
||||
FSRef ref;
|
||||
Boolean *isDirectory=NULL;
|
||||
|
||||
/* check parameters */
|
||||
require_action(NULL != spec, BadParameter, result = paramErr);
|
||||
|
||||
/* convert the POSIX path to an FSRef */
|
||||
result = FSPathMakeRef(path, &ref, isDirectory);
|
||||
require_noerr(result, FSPathMakeRef);
|
||||
|
||||
/* and then convert the FSRef to an FSSpec */
|
||||
result = FSGetCatalogInfo(&ref, kFSCatInfoNone, NULL, NULL, spec, NULL);
|
||||
require_noerr(result, FSGetCatalogInfo);
|
||||
|
||||
FSGetCatalogInfo:
|
||||
FSPathMakeRef:
|
||||
BadParameter:
|
||||
|
||||
return ( result );
|
||||
}
|
||||
|
||||
#elif defined _WIN32
|
||||
static OSStatus
|
||||
FSPathMakeFSSpec(
|
||||
const UInt8 *path,
|
||||
FSSpec *spec)
|
||||
{
|
||||
char filename[QT_MAX_FILENAMELENGTH];
|
||||
UInt8*filename8=reinterpret_cast<UInt8*>(filename);
|
||||
FILE*fil=NULL;
|
||||
|
||||
snprintf(filename, QT_MAX_FILENAMELENGTH, "%s", path);
|
||||
c2pstr(filename);
|
||||
OSErr err=FSMakeFSSpec (0, 0L, filename8, spec);
|
||||
if (err != noErr && err != -37){
|
||||
error("GEM: recordQT: error %d in FSMakeFSSpec()", err);
|
||||
} else {
|
||||
err = noErr;
|
||||
}
|
||||
return err;
|
||||
}
|
||||
#endif
|
||||
|
||||
// MemAlloc
|
||||
// Simple memory allocation wrapper
|
||||
static void* MemAlloc(unsigned long memsize)
|
||||
{
|
||||
if(memsize <= 0)
|
||||
return NULL;
|
||||
else
|
||||
return (malloc(memsize));
|
||||
}
|
||||
|
||||
static unsigned char* CStringToPString(char *string)
|
||||
{
|
||||
unsigned char *newString = (unsigned char*)MemAlloc(strlen(string) + 1);
|
||||
size_t i = 0;
|
||||
|
||||
for(i = 0; i < strlen(string); i++)
|
||||
newString[i+1] = string[i];
|
||||
|
||||
newString[0] = i;
|
||||
|
||||
return newString;
|
||||
}
|
||||
|
||||
|
||||
static void InvertGLImage( unsigned char *imageData, unsigned char * outData, long imageSize, long rowBytes )
|
||||
{
|
||||
long i, j;
|
||||
// This is not an optimized routine!
|
||||
|
||||
// FIXXME use a flip function in GemPixUtils for this
|
||||
|
||||
// Copy rows into tmp buffer one at a time, reversing their order
|
||||
for (i = 0, j = imageSize - rowBytes; i < imageSize; i += rowBytes, j -= rowBytes) {
|
||||
memcpy( &outData[j], &imageData[i], static_cast<size_t>(rowBytes) );
|
||||
}
|
||||
}
|
||||
|
||||
static std::map<std::string, OSType> s_mime2type;
|
||||
|
||||
static bool mime2type(const std::string&mimetype, OSType&filetype) {
|
||||
static bool done=false;
|
||||
if(!done) {
|
||||
done=true;
|
||||
//s_mime2type["image/"]=kQTFileTypeAIFF;
|
||||
//s_mime2type["image/"]=kQTFileTypeAIFC;
|
||||
//s_mime2type["image/"]=kQTFileTypeDVC;
|
||||
//s_mime2type["image/"]=kQTFileTypeMIDI;
|
||||
s_mime2type["image/pict"]=kQTFileTypePicture;//-
|
||||
//s_mime2type["image/"]=kQTFileTypeMovie;
|
||||
//s_mime2type["image/"]=kQTFileTypeText;
|
||||
//s_mime2type["image/"]=kQTFileTypeWave;
|
||||
//s_mime2type["image/"]=kQTFileTypeSystemSevenSound;
|
||||
//s_mime2type["image/"]=kQTFileTypeMuLaw;
|
||||
//s_mime2type["image/"]=kQTFileTypeAVI;
|
||||
//s_mime2type["image/"]=kQTFileTypeSoundDesignerII;
|
||||
//s_mime2type["image/"]=kQTFileTypeAudioCDTrack;
|
||||
//s_mime2type["image/pict"]=kQTFileTypePICS;
|
||||
s_mime2type["image/gif"]=kQTFileTypeGIF;//-
|
||||
s_mime2type["image/png"]=kQTFileTypePNG;//-
|
||||
s_mime2type["image/tiff"]=kQTFileTypeTIFF;//-
|
||||
s_mime2type["image/psd"]=kQTFileTypePhotoShop;//-
|
||||
s_mime2type["image/sgi"]=kQTFileTypeSGIImage;//-
|
||||
s_mime2type["image/bmp"]=kQTFileTypeBMP;//-
|
||||
s_mime2type["image/jpeg"]=kQTFileTypeJPEG;//-
|
||||
// s_mime2type["image/"]=kQTFileTypeJFIF;
|
||||
s_mime2type["image/mac"]=kQTFileTypeMacPaint;//-
|
||||
s_mime2type["image/targa"]=kQTFileTypeTargaImage;//-
|
||||
// s_mime2type["image/"]=kQTFileTypeQuickDrawGXPicture;
|
||||
s_mime2type["image/x-quicktime"]=kQTFileTypeQuickTimeImage;//-
|
||||
//s_mime2type["image/"]=kQTFileType3DMF;
|
||||
//s_mime2type["image/"]=kQTFileTypeFLC;
|
||||
//s_mime2type["image/"]=kQTFileTypeFlash;
|
||||
//s_mime2type["image/"]=kQTFileTypeFlashPix;
|
||||
//s_mime2type["image/"]=kQTFileTypeMP4;
|
||||
//s_mime2type["image/"]=kQTFileTypePDF;
|
||||
//s_mime2type["image/"]=kQTFileType3GPP;
|
||||
//s_mime2type["image/"]=kQTFileTypeAMR;
|
||||
//s_mime2type["image/"]=kQTFileTypeSDV;
|
||||
//s_mime2type["image/"]=kQTFileType3GP2;
|
||||
//s_mime2type["image/"]=kQTFileTypeAMC;
|
||||
s_mime2type["image/jp2"]=kQTFileTypeJPEG2000;//-
|
||||
}
|
||||
|
||||
std::map<std::string, OSType>::iterator it = s_mime2type.find(mimetype);
|
||||
if(s_mime2type.end() != it) {
|
||||
filetype=it->second;
|
||||
return true;
|
||||
}
|
||||
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
|
||||
/////////////////////////////////////////////////////////
|
||||
//
|
||||
// imageQT
|
||||
//
|
||||
/////////////////////////////////////////////////////////
|
||||
// Constructor
|
||||
//
|
||||
/////////////////////////////////////////////////////////
|
||||
|
||||
imageQT :: imageQT(void)
|
||||
{
|
||||
static bool firsttime=true;
|
||||
if(firsttime) {
|
||||
#ifdef _WIN32
|
||||
// Initialize QuickTime Media Layer
|
||||
OSErr err = noErr;
|
||||
if ((err = InitializeQTML(0))) {
|
||||
throw(GemException("unable to initialize QuickTime"));
|
||||
}
|
||||
// start QuickTime
|
||||
if (err = EnterMovies()) {
|
||||
throw(GemException("unable to initialize QuickTime/Movies"));
|
||||
}
|
||||
verbose(1, "imageQT: QT init done");
|
||||
#endif // WINDOWS
|
||||
firsttime=false;
|
||||
}
|
||||
//post("imageQT");
|
||||
}
|
||||
imageQT :: ~imageQT(void)
|
||||
{
|
||||
//post("~imageQT");
|
||||
}
|
||||
|
||||
/////////////////////////////////////////////////////////
|
||||
// really open the file ! (OS dependent)
|
||||
//
|
||||
/////////////////////////////////////////////////////////
|
||||
|
||||
static bool QuickTimeImage2mem(GraphicsImportComponent inImporter, imageStruct&result)
|
||||
{
|
||||
Rect r;
|
||||
if (::GraphicsImportGetNaturalBounds(inImporter, &r)) return NULL; //get an image size
|
||||
::OffsetRect(&r, -r.left, -r.top);
|
||||
if (::GraphicsImportSetBoundsRect(inImporter, &r)) return NULL;
|
||||
ImageDescriptionHandle imageDescH = NULL;
|
||||
if (::GraphicsImportGetImageDescription(inImporter, &imageDescH)) return NULL;
|
||||
|
||||
result.xsize = (*imageDescH)->width;
|
||||
result.ysize = (*imageDescH)->height;
|
||||
result.upsidedown = true;
|
||||
|
||||
OSType pixelformat = 0;
|
||||
|
||||
/* afaik, there is no 8bit grayscale format....
|
||||
* and even if it was, k8GrayPixelFormat would not be a define...
|
||||
*/
|
||||
#ifdef k8GrayPixelFormat
|
||||
/* from the docs on "depth": what depth is this data (1-32) or ( 33-40 grayscale ) */
|
||||
if ((*imageDescH)->depth <= 32) {
|
||||
result.setCsizeByFormat(GL_RGBA_GEM);
|
||||
pixelformat = IMAGEQT_RGBA_PIXELFORMAT;
|
||||
} else {
|
||||
result.setCsizeByFormat(GL_LUMINANCE);
|
||||
pixelformat = k8GrayPixelFormat;
|
||||
}
|
||||
#else
|
||||
result.setCsizeByFormat(GL_RGBA_GEM);
|
||||
pixelformat = IMAGEQT_RGBA_PIXELFORMAT;
|
||||
#endif
|
||||
|
||||
::DisposeHandle(reinterpret_cast<Handle>(imageDescH));
|
||||
imageDescH = NULL;
|
||||
result.reallocate();
|
||||
|
||||
#ifdef __DEBUG__
|
||||
post("QuickTimeImage2mem() : allocate %d bytes", result.xsize*result.ysize*result.csize);
|
||||
#endif
|
||||
if (result.data == NULL) {
|
||||
error("Can't allocate memory for an image.");
|
||||
return false;
|
||||
}
|
||||
|
||||
GWorldPtr gw = NULL;
|
||||
|
||||
OSErr err = QTNewGWorldFromPtr(&gw,
|
||||
/* taken from pix_filmDarwin */
|
||||
pixelformat, // gives noErr
|
||||
&r, NULL, NULL, 0,
|
||||
// keepLocal,
|
||||
//useDistantHdwrMem,
|
||||
result.data,
|
||||
static_cast<long>(result.xsize * result.csize));
|
||||
if (err) {
|
||||
error("Can't create QTNewWorld");
|
||||
}
|
||||
|
||||
::GraphicsImportSetGWorld(inImporter, gw, NULL);
|
||||
::GraphicsImportDraw(inImporter);
|
||||
::DisposeGWorld(gw); //dispose the offscreen
|
||||
gw = NULL;
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
bool imageQT :: load(std::string filename, imageStruct&result, gem::Properties&props)
|
||||
{
|
||||
OSErr err;
|
||||
GraphicsImportComponent importer = NULL;
|
||||
|
||||
::verbose(2, "reading '%s' with QuickTime", filename.c_str());
|
||||
std::string myfilename=filename;
|
||||
// does the file even exist?
|
||||
if (!filename.empty()) {
|
||||
FSSpec spec;
|
||||
err = ::FSPathMakeFSSpec( reinterpret_cast<const UInt8*>(myfilename.c_str()), &spec);
|
||||
if (err) {
|
||||
error("GemImageLoad: Unable to find file: %s", filename.c_str());
|
||||
error("parID : %d", spec.parID);
|
||||
return false;
|
||||
}
|
||||
err = ::GetGraphicsImporterForFile(&spec, &importer);
|
||||
if (err) {
|
||||
error("GemImageLoad: Unable to import image '%s'", filename.c_str());
|
||||
return false;
|
||||
}
|
||||
}
|
||||
bool ret = QuickTimeImage2mem(importer, result);
|
||||
|
||||
::CloseComponent(importer);
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
static bool touch(std::string filename) {
|
||||
#ifdef __APPLE__
|
||||
int fd;
|
||||
fd = open(filename.c_str(), O_CREAT | O_RDWR, 0600);
|
||||
if (fd < 0)
|
||||
return false;
|
||||
write(fd, " ", 1);
|
||||
close(fd);
|
||||
|
||||
return true;
|
||||
#elif defined _WIN32
|
||||
FILE*outfile=fopen(filename.c_str(), "");
|
||||
if(NULL==outfile)
|
||||
return false;
|
||||
fclose(outfile);
|
||||
return true;
|
||||
#endif
|
||||
return false;
|
||||
}
|
||||
|
||||
bool imageQT::save(const imageStruct&constimage, const std::string&filename, const std::string&mimetype, const gem::Properties&props) {
|
||||
OSErr err=noErr;
|
||||
ComponentResult cErr = 0;
|
||||
|
||||
GWorldPtr img = NULL;
|
||||
GraphicsExportComponent geComp = NULL;
|
||||
Rect r;
|
||||
|
||||
FSSpec spec;
|
||||
|
||||
OSType osFileType=kQTFileTypeTIFF;
|
||||
mime2type(mimetype, osFileType);
|
||||
std::string myfilename=filename.c_str();
|
||||
|
||||
const UInt8*filename8=reinterpret_cast<const UInt8*>(myfilename.c_str());
|
||||
#if defined __APPLE__
|
||||
FSRef ref;
|
||||
err = ::FSPathMakeRef(filename8, &ref, NULL );
|
||||
|
||||
if (err == fnfErr) {
|
||||
// if the file does not yet exist, then let's create the file
|
||||
touch(myfilename);
|
||||
err = FSPathMakeRef(filename8, &ref, NULL);
|
||||
}
|
||||
|
||||
if (err != noErr) {
|
||||
verbose(1, "ERROR: %d in FSPathMakeRef()", err);
|
||||
}
|
||||
|
||||
err = ::FSGetCatalogInfo(&ref, kFSCatInfoNodeFlags, NULL, NULL, &spec, NULL);
|
||||
|
||||
if (err != noErr) {
|
||||
verbose(1, "ERROR: %d in FSGetCatalogInfo()", err);
|
||||
}
|
||||
|
||||
err = FSMakeFSSpec(spec.vRefNum, spec.parID, filename8, &spec); //this always gives an error -37 ???
|
||||
|
||||
#elif defined _WIN32
|
||||
touch(myfilename);
|
||||
err = FSMakeFSSpec (0, 0L, filename8, &spec);
|
||||
#endif
|
||||
if (err != noErr && err != -37){
|
||||
verbose(1, "GEM: imageQT: error %d in FSMakeFSSpec()", err);
|
||||
}
|
||||
|
||||
err = OpenADefaultComponent(GraphicsExporterComponentType, osFileType, &geComp);
|
||||
if (err != noErr) {
|
||||
error("ERROR: %d in OpenADefaultComponent()", err);
|
||||
return false; // FIXME:
|
||||
}
|
||||
|
||||
r.top = 0;
|
||||
r.left = 0;
|
||||
r.bottom = constimage.ysize;
|
||||
r.right = constimage.xsize;
|
||||
|
||||
imageStruct rgbaimg;
|
||||
rgbaimg.convertFrom(&constimage, GL_RGBA_GEM);
|
||||
|
||||
unsigned char *data = NULL;
|
||||
if(!rgbaimg.upsidedown) { // the image is openGL-oriented, not quicktime-oriented! flip it!
|
||||
int rowBytes = rgbaimg.xsize * rgbaimg.csize;
|
||||
int imageSize = rgbaimg.ysize * rowBytes;
|
||||
|
||||
data = new unsigned char[imageSize];
|
||||
|
||||
InvertGLImage(rgbaimg.data, data, imageSize, rowBytes);
|
||||
}
|
||||
|
||||
err = QTNewGWorldFromPtr(&img,
|
||||
IMAGEQT_RGBA_PIXELFORMAT, //k32RGBAPixelFormat,
|
||||
&r, NULL, NULL, 0,
|
||||
(data?data:rgbaimg.data),
|
||||
static_cast<long>(rgbaimg.xsize * rgbaimg.csize));
|
||||
|
||||
// is this the right place to free the "data" buffer (if used)?
|
||||
// i don't know, whether quicktime still needs the buffer...
|
||||
|
||||
if (err != noErr) {
|
||||
error("ERROR: %d in QTNewGWorldFromPtr()", err);
|
||||
if(data)delete[]data;
|
||||
return false; // FIXME:
|
||||
}
|
||||
|
||||
// Set the input GWorld for the exporter
|
||||
cErr = GraphicsExportSetInputGWorld(geComp, img);
|
||||
if (cErr != noErr) {
|
||||
error("ERROR: %d in GraphicsExportSetInputGWorld()", cErr);
|
||||
if(data)delete[]data;
|
||||
return false; // FIXME:
|
||||
}
|
||||
|
||||
// Set the output file to our FSSpec
|
||||
cErr = GraphicsExportSetOutputFile(geComp, &spec);
|
||||
if (cErr != noErr) {
|
||||
error("ERROR: %i in GraphicsExportSetOutputFile()", cErr);
|
||||
if(data)delete[]data;
|
||||
return false; // FIXME:
|
||||
}
|
||||
|
||||
// Set the compression quality (needed for JPEG, not necessarily for other formats)
|
||||
/*
|
||||
codecMinQuality
|
||||
codecLowQuality
|
||||
codecNormalQuality
|
||||
codecHighQuality
|
||||
codecMaxQuality
|
||||
codecLosslessQuality
|
||||
*/
|
||||
|
||||
CodecQ quality=codecHighQuality;
|
||||
|
||||
double d=0.;
|
||||
if(props.get("quality", d)) {
|
||||
// <0 = minqality
|
||||
// >=100 = lossless
|
||||
if(d<0.)d=0.;
|
||||
else if(d>100.)d=100.;
|
||||
|
||||
CodecQ maxQ=codecLosslessQuality;
|
||||
double maxQ_d=(double)maxQ;
|
||||
double quality_d=maxQ_d * d / 100.; // 0..maxQ
|
||||
|
||||
quality=(CodecQ)quality_d;
|
||||
}
|
||||
|
||||
cErr = GraphicsExportSetCompressionQuality(geComp, quality);
|
||||
|
||||
// Export it
|
||||
cErr = GraphicsExportDoExport(geComp, NULL);
|
||||
if (cErr != noErr) {
|
||||
error("ERROR: %i in GraphicsExportDoExport()", cErr);
|
||||
if(data)delete[]data;
|
||||
return false; // FIXME:
|
||||
}
|
||||
|
||||
// finally, close the component
|
||||
if (geComp != NULL)
|
||||
CloseComponent(geComp);
|
||||
|
||||
if(data)delete[]data;
|
||||
return true;
|
||||
}
|
||||
|
||||
|
||||
float imageQT::estimateSave(const imageStruct&img, const std::string&filename, const std::string&mimetype, const gem::Properties&props) {
|
||||
float result=0.;
|
||||
|
||||
OSType filetype; // just a dummy
|
||||
if(mime2type(mimetype, filetype))
|
||||
result+=100.;
|
||||
|
||||
// LATER check some properties....
|
||||
if(gem::Properties::UNSET != props.type("quality"))
|
||||
result += 1.;
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
void imageQT::getWriteCapabilities(std::vector<std::string>&mimetypes, gem::Properties&props) {
|
||||
mimetypes.clear();
|
||||
props.clear();
|
||||
|
||||
std::map<std::string, OSType>::iterator it;
|
||||
for(it = s_mime2type.begin(); it!=s_mime2type.end(); ++it) {
|
||||
mimetypes.push_back(it->first);
|
||||
}
|
||||
|
||||
gem::any value;
|
||||
|
||||
value=100.f;
|
||||
props.set("quality", value);
|
||||
}
|
||||
#endif /* have_quicktime */
|
Loading…
Add table
Add a link
Reference in a new issue