298 lines
7.2 KiB
C++
Raw Normal View History

2011-02-21 16:30:31 +01:00
/**************************************************************************
**
** This file is part of Qt SDK**
**
** Copyright (c) 2011 Nokia Corporation and/or its subsidiary(-ies).*
**
** Contact: Nokia Corporation qt-info@nokia.com**
**
** No Commercial Usage
**
** This file contains pre-release code and may not be distributed.
** You may use this file in accordance with the terms and conditions
** contained in the Technology Preview License Agreement accompanying
** this package.
**
** GNU Lesser General Public License Usage
**
** This file may be used under the terms of the GNU Lesser General Public
** License version 2.1 as published by the Free Software Foundation and
** appearing in the file LICENSE.LGPL included in the packaging of this file.
** Please review the following information to ensure the GNU Lesser General
** Public License version 2.1 requirements will be met:
** http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.
**
** In addition, as a special exception, Nokia gives you certain additional
** rights. These rights are described in the Nokia Qt LGPL Exception version
** 1.1, included in the file LGPL_EXCEPTION.txt in this package.
**
** If you are unsure which license is appropriate for your use, please contact
** (qt-info@nokia.com).
**
**************************************************************************/
#include "binaryformatengine.h"
using namespace QInstallerCreator;
namespace {
2011-02-21 16:30:31 +01:00
class StringListIterator : public QAbstractFileEngineIterator
{
public:
StringListIterator( const QStringList &list, QDir::Filters filters, const QStringList &nameFilters)
: QAbstractFileEngineIterator(filters, nameFilters),
list(list),
index(-1)
2011-02-21 16:30:31 +01:00
{
}
bool hasNext() const
{
return index < list.size() - 1;
}
QString next()
{
if(!hasNext())
2011-02-21 16:30:31 +01:00
return QString();
++index;
return currentFilePath();
}
QString currentFileName() const
{
return index < 0 ? QString() : list[index];
2011-02-21 16:30:31 +01:00
}
private:
const QStringList list;
int index;
};
} // anon namespace
BinaryFormatEngine::BinaryFormatEngine(const ComponentIndex &index, const QString &fileName)
: m_index(index)
, m_hasComponent(false)
, m_hasArchive(false)
, m_archive(0)
2011-02-21 16:30:31 +01:00
{
setArchive(fileName);
2011-02-21 16:30:31 +01:00
}
BinaryFormatEngine::~BinaryFormatEngine()
{
}
void BinaryFormatEngine::setArchive(const QString &file)
2011-02-21 16:30:31 +01:00
{
m_fileNamePath = file;
static const QChar sep = QChar::fromLatin1('/');
static const QString prefix = QString::fromLatin1("installer://");
Q_ASSERT(file.toLower().startsWith(prefix));
2011-02-21 16:30:31 +01:00
// cut the prefix
QString path = file.mid(prefix.length());
while (path.endsWith(sep))
path.chop(1);
2011-02-21 16:30:31 +01:00
QString arch;
const QString comp = path.section(sep, 0, 0);
2011-02-21 16:30:31 +01:00
m_hasComponent = !comp.isEmpty();
m_hasArchive = path.contains(sep);
if (m_hasArchive)
arch = path.section(sep, 1, 1);
2011-02-21 16:30:31 +01:00
m_component = m_index.componentByName(comp.toUtf8());
m_archive = m_component.archiveByName(arch.toUtf8());
2011-02-21 16:30:31 +01:00
}
/**
* \reimp
*/
void BinaryFormatEngine::setFileName(const QString &file)
2011-02-21 16:30:31 +01:00
{
setArchive(file);
2011-02-21 16:30:31 +01:00
}
/**
* \reimp
*/
bool BinaryFormatEngine::close()
{
if (m_archive == 0)
2011-02-21 16:30:31 +01:00
return false;
const bool result = m_archive->isOpen();
m_archive->close();
return result;
}
/**
* \reimp
*/
bool BinaryFormatEngine::open(QIODevice::OpenMode mode)
2011-02-21 16:30:31 +01:00
{
return m_archive == 0 ? false : m_archive->open(mode);
2011-02-21 16:30:31 +01:00
}
/**
* \reimp
*/
qint64 BinaryFormatEngine::pos() const
{
return m_archive == 0 ? 0 : m_archive->pos();
}
/**
* \reimp
*/
qint64 BinaryFormatEngine::read(char *data, qint64 maxlen)
2011-02-21 16:30:31 +01:00
{
return m_archive == 0 ? -1 : m_archive->read(data, maxlen);
2011-02-21 16:30:31 +01:00
}
/**
* \reimp
*/
bool BinaryFormatEngine::seek(qint64 offset)
2011-02-21 16:30:31 +01:00
{
return m_archive == 0 ? false : m_archive->seek(offset);
2011-02-21 16:30:31 +01:00
}
/**
* \reimp
*/
QString BinaryFormatEngine::fileName(FileName file) const
2011-02-21 16:30:31 +01:00
{
switch(file) {
2011-02-21 16:30:31 +01:00
case BaseName:
return m_fileNamePath.section(QChar::fromLatin1('/'), -1, -1, QString::SectionSkipEmpty);
2011-02-21 16:30:31 +01:00
case PathName:
case AbsolutePathName:
case CanonicalPathName:
return m_fileNamePath.section(QChar::fromLatin1('/'), 0, -2, QString::SectionSkipEmpty);
2011-02-21 16:30:31 +01:00
case DefaultName:
case AbsoluteName:
case CanonicalName:
return m_fileNamePath;
default:
return QString();
}
}
/**
* \reimp
*/
bool BinaryFormatEngine::copy(const QString &newName)
2011-02-21 16:30:31 +01:00
{
if (QFile::exists(newName))
2011-02-21 16:30:31 +01:00
return false;
QFile target(newName);
if (!target.open(QIODevice::WriteOnly))
2011-02-21 16:30:31 +01:00
return false;
qint64 bytesLeft = size();
if (!open(QIODevice::ReadOnly))
2011-02-21 16:30:31 +01:00
return false;
char data[4096];
while(bytesLeft > 0) {
const qint64 len = qMin<qint64>(bytesLeft, 4096);
const qint64 bytesRead = read(data, len);
if (bytesRead != len) {
2011-02-21 16:30:31 +01:00
close();
return false;
}
const qint64 bytesWritten = target.write(data, len);
if (bytesWritten != len) {
2011-02-21 16:30:31 +01:00
close();
return false;
}
bytesLeft -= len;
}
close();
return true;
}
/**
* \reimp
*/
QAbstractFileEngine::FileFlags BinaryFormatEngine::fileFlags(FileFlags type) const
2011-02-21 16:30:31 +01:00
{
FileFlags result;
if ((type & FileType) && m_archive != 0)
2011-02-21 16:30:31 +01:00
result |= FileType;
if ((type & DirectoryType) && !m_hasArchive)
2011-02-21 16:30:31 +01:00
result |= DirectoryType;
if ((type & ExistsFlag) && m_hasArchive && m_archive != 0)
2011-02-21 16:30:31 +01:00
result |= ExistsFlag;
if ((type & ExistsFlag) && !m_hasArchive && !m_component.name().isEmpty())
2011-02-21 16:30:31 +01:00
result |= ExistsFlag;
return result;
}
/**
* \reimp
*/
QAbstractFileEngineIterator *BinaryFormatEngine::beginEntryList(QDir::Filters filters, const QStringList &filterNames)
2011-02-21 16:30:31 +01:00
{
const QStringList entries = entryList(filters, filterNames);
return new StringListIterator(entries, filters, filterNames);
2011-02-21 16:30:31 +01:00
}
/**
* \reimp
*/
QStringList BinaryFormatEngine::entryList(QDir::Filters filters, const QStringList &filterNames) const
2011-02-21 16:30:31 +01:00
{
if (m_hasArchive)
2011-02-21 16:30:31 +01:00
return QStringList();
QStringList result;
if (m_hasComponent && (filters & QDir::Files)) {
2011-02-21 16:30:31 +01:00
const QVector< QSharedPointer<Archive> > archives = m_component.archives();
foreach (const QSharedPointer<Archive> &i, archives)
result.push_back(QString::fromUtf8(i->name()));
2011-02-21 16:30:31 +01:00
}
else if (!m_hasComponent && (filters & QDir::Dirs)) {
const QVector<Component> components = m_index.components();
foreach (const Component &i, components)
result.push_back(QString::fromUtf8(i.name()));
2011-02-21 16:30:31 +01:00
}
if (filterNames.isEmpty())
2011-02-21 16:30:31 +01:00
return result;
QList<QRegExp> regexps;
foreach (const QString &i, filterNames)
regexps.push_back(QRegExp(i, Qt::CaseInsensitive, QRegExp::Wildcard));
2011-02-21 16:30:31 +01:00
QStringList entries;
foreach (const QString &i, result) {
2011-02-21 16:30:31 +01:00
bool matched = false;
foreach (const QRegExp &reg, regexps) {
matched = reg.exactMatch(i);
if (matched)
2011-02-21 16:30:31 +01:00
break;
}
if (matched)
entries.push_back(i);
2011-02-21 16:30:31 +01:00
}
return entries;
}
/**
* \reimp
*/
qint64 BinaryFormatEngine::size() const
{
return m_archive == 0 ? 0 : m_archive->size();
}