You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
237 lines
6.0 KiB
237 lines
6.0 KiB
/*************************************************************************** |
|
* Copyright (C) 2007 by Tobias Koenig <tokoe@kde.org> * |
|
* * |
|
* 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. * |
|
***************************************************************************/ |
|
|
|
#include "document.h" |
|
|
|
#include <QtGui/QImage> |
|
#include <QtGui/QImageReader> |
|
|
|
#include <klocale.h> |
|
#include <kmimetype.h> |
|
#include <kzip.h> |
|
#include <ktar.h> |
|
|
|
#include <memory> |
|
|
|
#include "unrar.h" |
|
#include "directory.h" |
|
#include "qnatsort.h" |
|
|
|
using namespace ComicBook; |
|
|
|
static void imagesInArchive( const QString &prefix, const KArchiveDirectory* dir, QStringList *entries ) |
|
{ |
|
Q_FOREACH ( const QString &entry, dir->entries() ) { |
|
const KArchiveEntry *e = dir->entry( entry ); |
|
if ( e->isDirectory() ) { |
|
imagesInArchive( prefix + entry + '/', static_cast<const KArchiveDirectory*>( e ), entries ); |
|
} else if ( e->isFile() ) { |
|
entries->append( prefix + entry ); |
|
} |
|
} |
|
} |
|
|
|
|
|
Document::Document() |
|
: mDirectory( 0 ), mUnrar( 0 ), mArchive( 0 ) |
|
{ |
|
} |
|
|
|
Document::~Document() |
|
{ |
|
} |
|
|
|
bool Document::open( const QString &fileName ) |
|
{ |
|
close(); |
|
|
|
const KMimeType::Ptr mime = KMimeType::findByFileContent( fileName ); |
|
|
|
/** |
|
* We have a zip archive |
|
*/ |
|
if ( mime->is( "application/x-cbz" ) || mime->name() == "application/zip" ) { |
|
mArchive = new KZip( fileName ); |
|
|
|
if ( !processArchive() ) { |
|
return false; |
|
} |
|
/** |
|
* We have a TAR archive |
|
*/ |
|
} else if ( mime->is( "application/x-cbt" ) || mime->name() == "application/x-gzip" || |
|
mime->name() == "application/x-tar" || mime->name() == "application/x-bzip" ) { |
|
mArchive = new KTar( fileName ); |
|
|
|
if ( !processArchive() ) { |
|
return false; |
|
} |
|
} else if ( mime->is( "application/x-cbr" ) || mime->name() == "application/x-rar" ) { |
|
if ( !Unrar::isAvailable() ) { |
|
mLastErrorString = i18n( "Cannot open document, unrar was not found." ); |
|
return false; |
|
} |
|
|
|
if ( !Unrar::isSuitableVersionAvailable() ) { |
|
mLastErrorString = i18n( "The version of unrar on your system is not suitable for opening comicbooks." ); |
|
return false; |
|
} |
|
|
|
/** |
|
* We have a rar archive |
|
*/ |
|
mUnrar = new Unrar(); |
|
|
|
if ( !mUnrar->open( fileName ) ) { |
|
delete mUnrar; |
|
mUnrar = 0; |
|
|
|
return false; |
|
} |
|
|
|
extractImageFiles( mUnrar->list() ); |
|
} else if ( mime->is( "inode/directory" ) ) { |
|
mDirectory = new Directory(); |
|
|
|
if ( !mDirectory->open( fileName ) ) { |
|
delete mDirectory; |
|
mDirectory = 0; |
|
|
|
return false; |
|
} |
|
|
|
extractImageFiles( mDirectory->list() ); |
|
} else { |
|
mLastErrorString = i18n( "Unknown ComicBook format." ); |
|
return false; |
|
} |
|
|
|
return true; |
|
} |
|
|
|
void Document::close() |
|
{ |
|
mLastErrorString.clear(); |
|
|
|
if ( !( mArchive || mUnrar || mDirectory ) ) |
|
return; |
|
|
|
delete mArchive; |
|
mArchive = 0; |
|
delete mDirectory; |
|
mDirectory = 0; |
|
delete mUnrar; |
|
mUnrar = 0; |
|
mPageMap.clear(); |
|
} |
|
|
|
bool Document::processArchive() { |
|
if ( !mArchive->open( QIODevice::ReadOnly ) ) { |
|
delete mArchive; |
|
mArchive = 0; |
|
|
|
return false; |
|
} |
|
|
|
const KArchiveDirectory *directory = mArchive->directory(); |
|
if ( !directory ) { |
|
delete mArchive; |
|
mArchive = 0; |
|
|
|
return false; |
|
} |
|
|
|
mArchiveDir = const_cast<KArchiveDirectory*>( directory ); |
|
|
|
QStringList entries; |
|
imagesInArchive( QString(), mArchiveDir, &entries ); |
|
|
|
extractImageFiles( entries ); |
|
return true; |
|
} |
|
|
|
void Document::extractImageFiles( const QStringList &list ) |
|
{ |
|
QStringList files( list ); |
|
|
|
qSort( files.begin(), files.end(), caseSensitiveNaturalOrderLessThen ); |
|
|
|
for ( int i = 0; i < files.count(); ++i ) { |
|
const QString lowerFile = files[ i ].toLower(); |
|
|
|
if ( lowerFile.endsWith( ".gif" ) || |
|
lowerFile.endsWith( ".jpg" ) || |
|
lowerFile.endsWith( ".jpeg" ) || |
|
lowerFile.endsWith( ".png" ) || |
|
lowerFile.endsWith( ".tif" ) || |
|
lowerFile.endsWith( ".tiff" ) ) |
|
mPageMap.append( files[ i ] ); |
|
} |
|
} |
|
|
|
int Document::pages() const |
|
{ |
|
return mPageMap.count(); |
|
} |
|
|
|
QStringList Document::pageTitles() const |
|
{ |
|
return QStringList(); |
|
} |
|
|
|
QImage Document::pageImage( int page ) const |
|
{ |
|
if ( mArchive ) { |
|
const KArchiveFile *entry = static_cast<const KArchiveFile*>( mArchiveDir->entry( mPageMap[ page ] ) ); |
|
if ( entry ) |
|
return QImage::fromData( entry->data() ); |
|
} else if ( mDirectory ) { |
|
return QImage( mPageMap[ page ] ); |
|
} else { |
|
return QImage::fromData( mUnrar->contentOf( mPageMap[ page ] ) ); |
|
} |
|
|
|
return QImage(); |
|
} |
|
|
|
QSize Document::pageSize( int page ) const |
|
{ |
|
std::auto_ptr< QIODevice > dev; |
|
|
|
if ( mArchive ) { |
|
const KArchiveFile *entry = static_cast<const KArchiveFile*>( mArchiveDir->entry( mPageMap[ page ] ) ); |
|
if ( entry ) { |
|
dev.reset( entry->createDevice() ); |
|
} |
|
|
|
} else if ( mDirectory ) { |
|
dev.reset( mDirectory->createDevice( mPageMap[ page ] ) ); |
|
} else { |
|
dev.reset( mUnrar->createDevice( mPageMap[ page ] ) ); |
|
} |
|
|
|
if ( dev.get() ) { |
|
QImageReader reader( dev.get() ); |
|
if ( reader.canRead() ) { |
|
QSize s = reader.size(); |
|
if ( !s.isValid() ) { |
|
s = reader.read().size(); |
|
} |
|
return s; |
|
} |
|
} |
|
|
|
return QSize(); |
|
} |
|
|
|
QString Document::lastErrorString() const |
|
{ |
|
return mLastErrorString; |
|
} |
|
|
|
|