2006-07-07 14:58:12 +00:00
|
|
|
/***************************************************************************
|
|
|
|
* Copyright (C) 2006 by Pino Toscano <toscano.pino@tiscali.it> *
|
|
|
|
* *
|
|
|
|
* This program 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 2 of the License, or *
|
|
|
|
* (at your option) any later version. *
|
|
|
|
***************************************************************************/
|
|
|
|
|
2007-04-19 18:30:20 +00:00
|
|
|
#include "generator_tiff.h"
|
|
|
|
|
2006-07-12 13:33:29 +00:00
|
|
|
#include <qdatetime.h>
|
2006-07-07 14:58:12 +00:00
|
|
|
#include <qfile.h>
|
|
|
|
#include <qimage.h>
|
|
|
|
#include <qlist.h>
|
2006-11-17 15:21:40 +00:00
|
|
|
#include <qpainter.h>
|
2007-10-15 23:01:27 +00:00
|
|
|
#include <QtGui/QPrinter>
|
|
|
|
|
2007-07-14 14:00:19 +00:00
|
|
|
#include <kaboutdata.h>
|
2008-01-25 00:05:57 +00:00
|
|
|
#include <kdebug.h>
|
2006-07-12 13:33:29 +00:00
|
|
|
#include <kglobal.h>
|
2006-07-10 22:52:50 +00:00
|
|
|
#include <klocale.h>
|
2006-07-07 14:58:12 +00:00
|
|
|
|
2007-01-03 14:30:48 +00:00
|
|
|
#include <okular/core/document.h>
|
|
|
|
#include <okular/core/page.h>
|
2007-11-26 21:43:54 +00:00
|
|
|
#include <okular/core/fileprinter.h>
|
2008-05-31 11:40:49 +00:00
|
|
|
#include <okular/core/utils.h>
|
2007-01-03 14:30:48 +00:00
|
|
|
|
2006-07-07 14:58:12 +00:00
|
|
|
#include <tiff.h>
|
|
|
|
#include <tiffio.h>
|
|
|
|
|
2008-01-25 00:05:57 +00:00
|
|
|
#define TiffDebug 4714
|
|
|
|
|
2006-07-07 14:58:12 +00:00
|
|
|
class TIFFGenerator::Private
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
Private()
|
|
|
|
: tiff( 0 ) {}
|
|
|
|
|
|
|
|
TIFF* tiff;
|
|
|
|
};
|
|
|
|
|
2006-07-12 13:33:29 +00:00
|
|
|
static QDateTime convertTIFFDateTime( const char* tiffdate )
|
|
|
|
{
|
|
|
|
if ( !tiffdate )
|
|
|
|
return QDateTime();
|
|
|
|
|
|
|
|
return QDateTime::fromString( QString::fromLatin1( tiffdate ), "yyyy:MM:dd HH:mm:ss" );
|
|
|
|
}
|
2006-07-07 14:58:12 +00:00
|
|
|
|
2008-05-31 11:40:49 +00:00
|
|
|
static void adaptSizeToResolution( TIFF *tiff, ttag_t whichres, double dpi, uint32 *size )
|
|
|
|
{
|
|
|
|
float resvalue = 1.0;
|
|
|
|
uint16 resunit = 0;
|
|
|
|
if ( !TIFFGetField( tiff, whichres, &resvalue )
|
|
|
|
|| !TIFFGetField( tiff, TIFFTAG_RESOLUTIONUNIT, &resunit ) )
|
|
|
|
return;
|
|
|
|
|
|
|
|
float newsize = *size / resvalue;
|
|
|
|
switch ( resunit )
|
|
|
|
{
|
|
|
|
case RESUNIT_INCH:
|
|
|
|
*size = (uint32)( newsize * dpi );
|
|
|
|
break;
|
|
|
|
case RESUNIT_CENTIMETER:
|
|
|
|
*size = (uint32)( newsize * 10.0 / 25.4 * dpi );
|
|
|
|
break;
|
|
|
|
case RESUNIT_NONE:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-12-02 20:57:24 +00:00
|
|
|
static KAboutData createAboutData()
|
2006-07-07 14:58:12 +00:00
|
|
|
{
|
2007-12-02 20:57:24 +00:00
|
|
|
KAboutData aboutData(
|
2007-08-28 23:27:37 +00:00
|
|
|
"okular_tiff",
|
|
|
|
"okular_tiff",
|
2007-07-14 14:00:19 +00:00
|
|
|
ki18n( "TIFF Backend" ),
|
2008-01-27 16:43:16 +00:00
|
|
|
"0.1.1",
|
2007-07-14 14:00:19 +00:00
|
|
|
ki18n( "A TIFF backend" ),
|
|
|
|
KAboutData::License_GPL,
|
2008-01-27 15:47:25 +00:00
|
|
|
ki18n( "© 2006-2008 Pino Toscano" ),
|
|
|
|
ki18nc( "This represents the libtiff version, as string with copyrights as well; can be left as-is.", "%1" ).subs( TIFFGetVersion() )
|
2007-07-14 14:00:19 +00:00
|
|
|
);
|
2007-12-02 20:57:24 +00:00
|
|
|
aboutData.addAuthor( ki18n( "Pino Toscano" ), KLocalizedString(), "pino@kde.org" );
|
|
|
|
return aboutData;
|
|
|
|
}
|
|
|
|
|
|
|
|
OKULAR_EXPORT_PLUGIN( TIFFGenerator, createAboutData() )
|
|
|
|
|
|
|
|
TIFFGenerator::TIFFGenerator( QObject *parent, const QVariantList &args )
|
|
|
|
: Okular::Generator( parent, args ),
|
|
|
|
d( new Private ), m_docInfo( 0 )
|
|
|
|
{
|
|
|
|
setFeature( Threaded );
|
|
|
|
setFeature( PrintNative );
|
|
|
|
setFeature( PrintToFile );
|
2006-07-07 14:58:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
TIFFGenerator::~TIFFGenerator()
|
|
|
|
{
|
|
|
|
if ( d->tiff )
|
|
|
|
{
|
|
|
|
TIFFClose( d->tiff );
|
|
|
|
d->tiff = 0;
|
|
|
|
}
|
|
|
|
|
2007-01-24 15:06:45 +00:00
|
|
|
delete m_docInfo;
|
2006-07-07 14:58:12 +00:00
|
|
|
delete d;
|
|
|
|
}
|
|
|
|
|
2006-09-21 08:45:36 +00:00
|
|
|
bool TIFFGenerator::loadDocument( const QString & fileName, QVector<Okular::Page*> & pagesVector )
|
2006-07-14 10:54:29 +00:00
|
|
|
{
|
|
|
|
d->tiff = TIFFOpen( QFile::encodeName( fileName ), "r" );
|
|
|
|
if ( !d->tiff )
|
|
|
|
return false;
|
|
|
|
|
2007-01-05 17:17:30 +00:00
|
|
|
loadPages( pagesVector );
|
2006-07-14 10:54:29 +00:00
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2007-10-08 16:46:51 +00:00
|
|
|
bool TIFFGenerator::doCloseDocument()
|
2006-07-07 14:58:12 +00:00
|
|
|
{
|
|
|
|
// closing the old document
|
|
|
|
if ( d->tiff )
|
|
|
|
{
|
|
|
|
TIFFClose( d->tiff );
|
|
|
|
d->tiff = 0;
|
2006-07-10 22:52:50 +00:00
|
|
|
delete m_docInfo;
|
|
|
|
m_docInfo = 0;
|
2008-01-25 00:05:57 +00:00
|
|
|
m_pageMapping.clear();
|
2006-07-07 14:58:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2007-01-24 15:06:45 +00:00
|
|
|
QImage TIFFGenerator::image( Okular::PixmapRequest * request )
|
2006-07-07 14:58:12 +00:00
|
|
|
{
|
|
|
|
bool generated = false;
|
2006-10-28 15:57:58 +00:00
|
|
|
QImage img;
|
2006-07-07 14:58:12 +00:00
|
|
|
|
2008-01-25 00:05:57 +00:00
|
|
|
if ( TIFFSetDirectory( d->tiff, mapPage( request->page()->number() ) ) )
|
2006-07-07 14:58:12 +00:00
|
|
|
{
|
2006-10-28 15:57:58 +00:00
|
|
|
int rotation = request->page()->rotation();
|
2008-05-31 11:40:49 +00:00
|
|
|
uint32 width = 1;
|
|
|
|
uint32 height = 1;
|
|
|
|
TIFFGetField( d->tiff, TIFFTAG_IMAGEWIDTH, &width );
|
|
|
|
TIFFGetField( d->tiff, TIFFTAG_IMAGELENGTH, &height );
|
2006-07-07 14:58:12 +00:00
|
|
|
if ( rotation % 2 == 1 )
|
|
|
|
qSwap( width, height );
|
|
|
|
|
|
|
|
QImage image( width, height, QImage::Format_RGB32 );
|
|
|
|
uint32 * data = (uint32 *)image.bits();
|
|
|
|
|
|
|
|
// read data
|
|
|
|
if ( TIFFReadRGBAImageOriented( d->tiff, width, height, data, ORIENTATION_TOPLEFT ) != 0 )
|
|
|
|
{
|
|
|
|
// an image read by ReadRGBAImage is ABGR, we need ARGB, so swap red and blue
|
|
|
|
uint32 size = width * height;
|
|
|
|
for ( uint32 i = 0; i < size; ++i )
|
|
|
|
{
|
|
|
|
uint32 red = ( data[i] & 0x00FF0000 ) >> 16;
|
|
|
|
uint32 blue = ( data[i] & 0x000000FF ) << 16;
|
|
|
|
data[i] = ( data[i] & 0xFF00FF00 ) + red + blue;
|
|
|
|
}
|
|
|
|
|
2006-10-25 15:35:53 +00:00
|
|
|
int reqwidth = request->width();
|
|
|
|
int reqheight = request->height();
|
2006-07-07 14:58:12 +00:00
|
|
|
if ( rotation % 2 == 1 )
|
|
|
|
qSwap( reqwidth, reqheight );
|
2006-10-28 15:57:58 +00:00
|
|
|
img = image.scaled( reqwidth, reqheight, Qt::IgnoreAspectRatio, Qt::SmoothTransformation );
|
2006-07-07 14:58:12 +00:00
|
|
|
|
|
|
|
generated = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( !generated )
|
|
|
|
{
|
2006-10-28 15:57:58 +00:00
|
|
|
img = QImage( request->width(), request->height(), QImage::Format_RGB32 );
|
|
|
|
img.fill( qRgb( 255, 255, 255 ) );
|
2006-07-07 14:58:12 +00:00
|
|
|
}
|
|
|
|
|
2007-01-24 15:06:45 +00:00
|
|
|
return img;
|
2006-07-07 14:58:12 +00:00
|
|
|
}
|
|
|
|
|
2006-09-21 08:45:36 +00:00
|
|
|
const Okular::DocumentInfo * TIFFGenerator::generateDocumentInfo()
|
2006-07-10 22:52:50 +00:00
|
|
|
{
|
|
|
|
if ( !d->tiff )
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
if ( m_docInfo )
|
|
|
|
return m_docInfo;
|
|
|
|
|
2006-09-21 08:45:36 +00:00
|
|
|
m_docInfo = new Okular::DocumentInfo();
|
2006-07-10 22:52:50 +00:00
|
|
|
|
2007-05-13 12:54:46 +00:00
|
|
|
m_docInfo->set( Okular::DocumentInfo::MimeType, "image/tiff" );
|
2006-07-10 22:52:50 +00:00
|
|
|
|
|
|
|
char* buffer = 0;
|
|
|
|
TIFFGetField( d->tiff, TIFFTAG_IMAGEDESCRIPTION, &buffer );
|
2007-05-13 12:54:46 +00:00
|
|
|
m_docInfo->set( Okular::DocumentInfo::Description, buffer ? QString::fromLatin1( buffer ) : i18n( "Unknown" ) );
|
2006-07-10 22:52:50 +00:00
|
|
|
|
|
|
|
buffer = 0;
|
|
|
|
TIFFGetField( d->tiff, TIFFTAG_SOFTWARE, &buffer );
|
2007-05-13 12:54:46 +00:00
|
|
|
m_docInfo->set( Okular::DocumentInfo::Producer, buffer ? QString::fromLatin1( buffer ) : i18n( "Unknown" ) );
|
2006-07-10 22:52:50 +00:00
|
|
|
|
|
|
|
buffer = 0;
|
|
|
|
TIFFGetField( d->tiff, TIFFTAG_COPYRIGHT, &buffer );
|
2007-05-13 12:54:46 +00:00
|
|
|
m_docInfo->set( Okular::DocumentInfo::Copyright, buffer ? QString::fromLatin1( buffer ) : i18n( "Unknown" ) );
|
2006-07-10 22:52:50 +00:00
|
|
|
|
|
|
|
buffer = 0;
|
|
|
|
TIFFGetField( d->tiff, TIFFTAG_ARTIST, &buffer );
|
2007-05-13 12:54:46 +00:00
|
|
|
m_docInfo->set( Okular::DocumentInfo::Author, buffer ? QString::fromLatin1( buffer ) : i18n( "Unknown" ) );
|
2006-07-10 22:52:50 +00:00
|
|
|
|
2006-07-12 13:33:29 +00:00
|
|
|
buffer = 0;
|
|
|
|
TIFFGetField( d->tiff, TIFFTAG_DATETIME, &buffer );
|
|
|
|
QDateTime date = convertTIFFDateTime( buffer );
|
2007-05-13 12:54:46 +00:00
|
|
|
m_docInfo->set( Okular::DocumentInfo::CreationDate, date.isValid() ? KGlobal::locale()->formatDateTime( date, KLocale::LongDate, true ) : i18n( "Unknown" ) );
|
2006-07-12 13:33:29 +00:00
|
|
|
|
2006-07-10 22:52:50 +00:00
|
|
|
return m_docInfo;
|
|
|
|
}
|
|
|
|
|
2007-01-05 17:17:30 +00:00
|
|
|
void TIFFGenerator::loadPages( QVector<Okular::Page*> & pagesVector )
|
2006-07-07 14:58:12 +00:00
|
|
|
{
|
|
|
|
if ( !d->tiff )
|
|
|
|
return;
|
|
|
|
|
|
|
|
tdir_t dirs = TIFFNumberOfDirectories( d->tiff );
|
|
|
|
pagesVector.resize( dirs );
|
2008-01-25 00:05:57 +00:00
|
|
|
tdir_t realdirs = 0;
|
2006-07-07 14:58:12 +00:00
|
|
|
|
|
|
|
uint32 width = 0;
|
|
|
|
uint32 height = 0;
|
|
|
|
|
|
|
|
for ( tdir_t i = 0; i < dirs; ++i )
|
|
|
|
{
|
|
|
|
if ( !TIFFSetDirectory( d->tiff, i ) )
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if ( TIFFGetField( d->tiff, TIFFTAG_IMAGEWIDTH, &width ) != 1 ||
|
|
|
|
TIFFGetField( d->tiff, TIFFTAG_IMAGELENGTH, &height ) != 1 )
|
|
|
|
continue;
|
|
|
|
|
2008-05-31 11:40:49 +00:00
|
|
|
adaptSizeToResolution( d->tiff, TIFFTAG_XRESOLUTION, Okular::Utils::dpiX(), &width );
|
|
|
|
adaptSizeToResolution( d->tiff, TIFFTAG_YRESOLUTION, Okular::Utils::dpiY(), &height );
|
|
|
|
|
2008-01-25 00:05:57 +00:00
|
|
|
Okular::Page * page = new Okular::Page( realdirs, width, height, Okular::Rotation0 );
|
|
|
|
pagesVector[ realdirs ] = page;
|
|
|
|
|
|
|
|
m_pageMapping[ realdirs ] = i;
|
2006-07-07 14:58:12 +00:00
|
|
|
|
2008-01-25 00:05:57 +00:00
|
|
|
++realdirs;
|
2006-07-07 14:58:12 +00:00
|
|
|
}
|
2008-01-25 00:05:57 +00:00
|
|
|
|
|
|
|
pagesVector.resize( realdirs );
|
2006-07-07 14:58:12 +00:00
|
|
|
}
|
|
|
|
|
2007-10-15 23:01:27 +00:00
|
|
|
bool TIFFGenerator::print( QPrinter& printer )
|
2006-11-17 15:21:40 +00:00
|
|
|
{
|
|
|
|
uint32 width = 0;
|
|
|
|
uint32 height = 0;
|
|
|
|
|
|
|
|
tdir_t dirs = TIFFNumberOfDirectories( d->tiff );
|
|
|
|
|
|
|
|
QPainter p( &printer );
|
|
|
|
|
2007-11-26 21:43:54 +00:00
|
|
|
QList<int> pageList = Okular::FilePrinter::pageList( printer, document()->pages(),
|
|
|
|
document()->bookmarkedPageList() );
|
|
|
|
|
|
|
|
for ( tdir_t i = 0; i < pageList.count(); ++i )
|
2006-11-17 15:21:40 +00:00
|
|
|
{
|
2008-01-25 00:05:57 +00:00
|
|
|
if ( !TIFFSetDirectory( d->tiff, mapPage( pageList[i] - 1 ) ) )
|
2006-11-17 15:21:40 +00:00
|
|
|
continue;
|
|
|
|
|
|
|
|
if ( TIFFGetField( d->tiff, TIFFTAG_IMAGEWIDTH, &width ) != 1 ||
|
|
|
|
TIFFGetField( d->tiff, TIFFTAG_IMAGELENGTH, &height ) != 1 )
|
|
|
|
continue;
|
|
|
|
|
|
|
|
QImage image( width, height, QImage::Format_RGB32 );
|
|
|
|
uint32 * data = (uint32 *)image.bits();
|
|
|
|
|
|
|
|
// read data
|
|
|
|
if ( TIFFReadRGBAImageOriented( d->tiff, width, height, data, ORIENTATION_TOPLEFT ) != 0 )
|
|
|
|
{
|
|
|
|
// an image read by ReadRGBAImage is ABGR, we need ARGB, so swap red and blue
|
|
|
|
uint32 size = width * height;
|
|
|
|
for ( uint32 i = 0; i < size; ++i )
|
|
|
|
{
|
|
|
|
uint32 red = ( data[i] & 0x00FF0000 ) >> 16;
|
|
|
|
uint32 blue = ( data[i] & 0x000000FF ) << 16;
|
|
|
|
data[i] = ( data[i] & 0xFF00FF00 ) + red + blue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( i != 0 )
|
|
|
|
printer.newPage();
|
|
|
|
|
|
|
|
p.drawImage( 0, 0, image );
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2008-01-25 00:05:57 +00:00
|
|
|
int TIFFGenerator::mapPage( int page ) const
|
|
|
|
{
|
|
|
|
QHash< int, int >::const_iterator it = m_pageMapping.find( page );
|
|
|
|
if ( it == m_pageMapping.end() )
|
|
|
|
{
|
|
|
|
kWarning(TiffDebug) << "Requesting unmapped page" << page << ":" << m_pageMapping;
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
return it.value();
|
|
|
|
}
|
|
|
|
|
2006-07-07 14:58:12 +00:00
|
|
|
#include "generator_tiff.moc"
|
|
|
|
|