mirror of
https://bitbucket.org/smil3y/kdelibs.git
synced 2025-02-23 18:32:49 +00:00
358 lines
10 KiB
C++
358 lines
10 KiB
C++
/**
|
|
* QImageIO Routines to read/write EPS images.
|
|
* copyright (c) 1998 Dirk Schoenberger <dirk.schoenberger@freenet.de>
|
|
*
|
|
* This library is distributed under the conditions of the GNU LGPL.
|
|
*/
|
|
#include "eps.h"
|
|
#include <unistd.h>
|
|
#include <stdio.h>
|
|
#include <QtGui/QImage>
|
|
#include <QtCore/QFile>
|
|
#include <QtGui/QPainter>
|
|
#include <QtGui/QPrinter>
|
|
#include <QtCore/QTextStream>
|
|
#include <QtCore/QTemporaryFile>
|
|
#include <kapplication.h>
|
|
#include <kdebug.h>
|
|
#include <kdeversion.h>
|
|
|
|
#define BUFLEN 200
|
|
|
|
#define BBOX "%%BoundingBox:"
|
|
#define BBOX_LEN strlen(BBOX)
|
|
|
|
static bool seekToCodeStart( QIODevice * io, quint32 & ps_offset, quint32 & ps_size )
|
|
{
|
|
char buf[4]; // We at most need to read 4 bytes at a time
|
|
ps_offset=0L;
|
|
ps_size=0L;
|
|
|
|
if ( io->read(buf, 2)!=2 ) // Read first two bytes
|
|
{
|
|
kError(399) << "kimgio EPS: EPS file has less than 2 bytes." << endl;
|
|
return false;
|
|
}
|
|
|
|
if ( buf[0]=='%' && buf[1]=='!' ) // Check %! magic
|
|
{
|
|
kDebug(399) << "kimgio EPS: normal EPS file";
|
|
}
|
|
else if ( buf[0]==char(0xc5) && buf[1]==char(0xd0) ) // Check start of MS-DOS EPS magic
|
|
{ // May be a MS-DOS EPS file
|
|
if ( io->read(buf+2, 2)!=2 ) // Read further bytes of MS-DOS EPS magic
|
|
{
|
|
kError(399) << "kimgio EPS: potential MS-DOS EPS file has less than 4 bytes." << endl;
|
|
return false;
|
|
}
|
|
if ( buf[2]==char(0xd3) && buf[3]==char(0xc6) ) // Check last bytes of MS-DOS EPS magic
|
|
{
|
|
if (io->read(buf, 4)!=4) // Get offset of PostScript code in the MS-DOS EPS file.
|
|
{
|
|
kError(399) << "kimgio EPS: cannot read offset of MS-DOS EPS file" << endl;
|
|
return false;
|
|
}
|
|
ps_offset // Offset is in little endian
|
|
= ((unsigned char) buf[0])
|
|
+ ((unsigned char) buf[1] << 8)
|
|
+ ((unsigned char) buf[2] << 16)
|
|
+ ((unsigned char) buf[3] << 24);
|
|
if (io->read(buf, 4)!=4) // Get size of PostScript code in the MS-DOS EPS file.
|
|
{
|
|
kError(399) << "kimgio EPS: cannot read size of MS-DOS EPS file" << endl;
|
|
return false;
|
|
}
|
|
ps_size // Size is in little endian
|
|
= ((unsigned char) buf[0])
|
|
+ ((unsigned char) buf[1] << 8)
|
|
+ ((unsigned char) buf[2] << 16)
|
|
+ ((unsigned char) buf[3] << 24);
|
|
kDebug(399) << "kimgio EPS: Offset: " << ps_offset <<" Size: " << ps_size;
|
|
if ( !io->seek(ps_offset) ) // Get offset of PostScript code in the MS-DOS EPS file.
|
|
{
|
|
kError(399) << "kimgio EPS: cannot seek in MS-DOS EPS file" << endl;
|
|
return false;
|
|
}
|
|
if ( io->read(buf, 2)!=2 ) // Read first two bytes of what should be the Postscript code
|
|
{
|
|
kError(399) << "kimgio EPS: PostScript code has less than 2 bytes." << endl;
|
|
return false;
|
|
}
|
|
if ( buf[0]=='%' && buf[1]=='!' ) // Check %! magic
|
|
{
|
|
kDebug(399) << "kimgio EPS: MS-DOS EPS file";
|
|
}
|
|
else
|
|
{
|
|
kError(399) << "kimgio EPS: supposed Postscript code of a MS-DOS EPS file doe not start with %!." << endl;
|
|
return false;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
kError(399) << "kimgio EPS: wrong magic for potential MS-DOS EPS file!" << endl;
|
|
return false;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
kError(399) << "kimgio EPS: not an EPS file!" << endl;
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static bool bbox ( QIODevice *io, int *x1, int *y1, int *x2, int *y2)
|
|
{
|
|
char buf[BUFLEN+1];
|
|
|
|
bool ret = false;
|
|
|
|
while (io->readLine(buf, BUFLEN) > 0)
|
|
{
|
|
if (strncmp (buf, BBOX, BBOX_LEN) == 0)
|
|
{
|
|
// Some EPS files have non-integer values for the bbox
|
|
// We don't support that currently, but at least we parse it
|
|
float _x1, _y1, _x2, _y2;
|
|
if ( sscanf (buf, "%*s %f %f %f %f",
|
|
&_x1, &_y1, &_x2, &_y2) == 4) {
|
|
kDebug(399) << "kimgio EPS BBOX: " << _x1 << " " << _y1 << " " << _x2 << " " << _y2;
|
|
*x1=(int)_x1; *y1=(int)_y1; *x2=(int)_x2; *y2=(int)_y2;
|
|
ret = true;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
EPSHandler::EPSHandler()
|
|
{
|
|
}
|
|
|
|
bool EPSHandler::canRead() const
|
|
{
|
|
if (canRead(device())) {
|
|
setFormat("eps");
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
bool EPSHandler::read(QImage *image)
|
|
{
|
|
kDebug(399) << "kimgio EPS: starting...";
|
|
|
|
FILE * ghostfd;
|
|
int x1, y1, x2, y2;
|
|
//QTime dt;
|
|
//dt.start();
|
|
|
|
QString cmdBuf;
|
|
QString tmp;
|
|
|
|
QIODevice* io = device();
|
|
quint32 ps_offset, ps_size;
|
|
|
|
// find start of PostScript code
|
|
if ( !seekToCodeStart(io, ps_offset, ps_size) )
|
|
return false;
|
|
|
|
// find bounding box
|
|
if ( !bbox (io, &x1, &y1, &x2, &y2)) {
|
|
kError(399) << "kimgio EPS: no bounding box found!" << endl;
|
|
return false;
|
|
}
|
|
|
|
QTemporaryFile tmpFile;
|
|
if( !tmpFile.open() ) {
|
|
kError(399) << "kimgio EPS: no temp file!" << endl;
|
|
return false;
|
|
}
|
|
|
|
// x1, y1 -> translation
|
|
// x2, y2 -> new size
|
|
|
|
x2 -= x1;
|
|
y2 -= y1;
|
|
//kDebug(399) << "origin point: " << x1 << "," << y1 << " size:" << x2 << "," << y2;
|
|
double xScale = 1.0;
|
|
double yScale = 1.0;
|
|
int wantedWidth = x2;
|
|
int wantedHeight = y2;
|
|
|
|
// create GS command line
|
|
|
|
cmdBuf = "gs -sOutputFile=";
|
|
cmdBuf += tmpFile.fileName();
|
|
cmdBuf += " -q -g";
|
|
tmp.setNum( wantedWidth );
|
|
cmdBuf += tmp;
|
|
tmp.setNum( wantedHeight );
|
|
cmdBuf += 'x';
|
|
cmdBuf += tmp;
|
|
cmdBuf += " -dSAFER -dPARANOIDSAFER -dNOPAUSE -sDEVICE=ppm -c "
|
|
"0 0 moveto "
|
|
"1000 0 lineto "
|
|
"1000 1000 lineto "
|
|
"0 1000 lineto "
|
|
"1 1 254 255 div setrgbcolor fill "
|
|
"0 0 0 setrgbcolor - -c showpage quit";
|
|
|
|
// run ghostview
|
|
|
|
ghostfd = popen (QFile::encodeName(cmdBuf), "w");
|
|
|
|
if ( ghostfd == 0 ) {
|
|
kError(399) << "kimgio EPS: no GhostScript?" << endl;
|
|
return false;
|
|
}
|
|
|
|
fprintf (ghostfd, "\n%d %d translate\n", -qRound(x1*xScale), -qRound(y1*yScale));
|
|
|
|
// write image to gs
|
|
|
|
io->reset(); // Go back to start of file to give all the file to GhostScript
|
|
if (ps_offset>0L) // We have an offset
|
|
io->seek(ps_offset);
|
|
QByteArray buffer ( io->readAll() );
|
|
|
|
// If we have no MS-DOS EPS file or if the size seems wrong, then choose the buffer size
|
|
if (ps_size<=0 || ps_size>(unsigned int)buffer.size())
|
|
ps_size=buffer.size();
|
|
|
|
fwrite(buffer.data(), sizeof(char), ps_size, ghostfd);
|
|
buffer.resize(0);
|
|
|
|
pclose ( ghostfd );
|
|
|
|
// load image
|
|
if( image->load (tmpFile.fileName()) ) {
|
|
kDebug(399) << "kimgio EPS: success!";
|
|
//kDebug(399) << "Loading EPS took " << (float)(dt.elapsed()) / 1000 << " seconds";
|
|
return true;
|
|
}
|
|
|
|
kError(399) << "kimgio EPS: no image!" << endl;
|
|
return false;
|
|
}
|
|
|
|
|
|
// Sven Wiegand <SWiegand@tfh-berlin.de> -- eps output filter (from KSnapshot)
|
|
bool EPSHandler::write(const QImage &image)
|
|
{
|
|
QPrinter psOut(QPrinter::PrinterResolution);
|
|
QPainter p;
|
|
|
|
// making some definitions (papersize, output to file, filename):
|
|
psOut.setCreator( "KDE " KDE_VERSION_STRING );
|
|
if ( psOut.outputFileName().isEmpty() )
|
|
psOut.setOutputFileName( "untitled_printer_document" );
|
|
|
|
// Extension must be .eps so that Qt generates EPS file
|
|
QTemporaryFile tmpFile("XXXXXXXX.eps");
|
|
if ( !tmpFile.open() )
|
|
return false;
|
|
|
|
psOut.setOutputFileName(tmpFile.fileName());
|
|
psOut.setOutputFormat(QPrinter::PostScriptFormat);
|
|
psOut.setFullPage(true);
|
|
psOut.setPaperSize(image.size(), QPrinter::DevicePixel);
|
|
|
|
// painting the pixmap to the "printer" which is a file
|
|
p.begin( &psOut );
|
|
p.drawImage( QPoint( 0, 0 ), image );
|
|
p.end();
|
|
|
|
// Copy file to imageio struct
|
|
QFile inFile(tmpFile.fileName());
|
|
if ( !inFile.open( QIODevice::ReadOnly ) )
|
|
return false;
|
|
|
|
QTextStream in( &inFile );
|
|
in.setCodec( "ISO-8859-1" );
|
|
QTextStream out( device() );
|
|
out.setCodec( "ISO-8859-1" );
|
|
|
|
QString szInLine = in.readLine();
|
|
out << szInLine << '\n';
|
|
|
|
while( !in.atEnd() ){
|
|
szInLine = in.readLine();
|
|
out << szInLine << '\n';
|
|
}
|
|
|
|
inFile.close();
|
|
|
|
return true;
|
|
}
|
|
|
|
QByteArray EPSHandler::name() const
|
|
{
|
|
return "eps";
|
|
}
|
|
|
|
bool EPSHandler::canRead(QIODevice *device)
|
|
{
|
|
if (!device) {
|
|
qWarning("EPSHandler::canRead() called with no device");
|
|
return false;
|
|
}
|
|
|
|
qint64 oldPos = device->pos();
|
|
|
|
QByteArray head = device->readLine(64);
|
|
int readBytes = head.size();
|
|
if (device->isSequential()) {
|
|
while (readBytes > 0)
|
|
device->ungetChar(head[readBytes-- - 1]);
|
|
} else {
|
|
device->seek(oldPos);
|
|
}
|
|
|
|
return head.contains("%!PS-Adobe");
|
|
}
|
|
|
|
class EPSPlugin : public QImageIOPlugin
|
|
{
|
|
public:
|
|
QStringList keys() const;
|
|
Capabilities capabilities(QIODevice *device, const QByteArray &format) const;
|
|
QImageIOHandler *create(QIODevice *device, const QByteArray &format = QByteArray()) const;
|
|
};
|
|
|
|
QStringList EPSPlugin::keys() const
|
|
{
|
|
return QStringList() << "eps" << "EPS" << "epsi" << "EPSI" << "epsf" << "EPSF";
|
|
}
|
|
|
|
QImageIOPlugin::Capabilities EPSPlugin::capabilities(QIODevice *device, const QByteArray &format) const
|
|
{
|
|
if (format == "eps" || format == "epsi" || format == "EPS" || format == "EPSI" ||
|
|
format == "epsf" || format == "EPSF")
|
|
return Capabilities(CanRead | CanWrite);
|
|
if (!format.isEmpty())
|
|
return 0;
|
|
if (!device->isOpen())
|
|
return 0;
|
|
|
|
Capabilities cap;
|
|
if (device->isReadable() && EPSHandler::canRead(device))
|
|
cap |= CanRead;
|
|
if (device->isWritable())
|
|
cap |= CanWrite;
|
|
return cap;
|
|
}
|
|
|
|
QImageIOHandler *EPSPlugin::create(QIODevice *device, const QByteArray &format) const
|
|
{
|
|
QImageIOHandler *handler = new EPSHandler;
|
|
handler->setDevice(device);
|
|
handler->setFormat(format);
|
|
return handler;
|
|
}
|
|
|
|
Q_EXPORT_STATIC_PLUGIN(EPSPlugin)
|
|
Q_EXPORT_PLUGIN2(eps, EPSPlugin)
|