363 lines
7.9 KiB
C++
363 lines
7.9 KiB
C++
#include "stdafx.h"
|
|
#include "ArchiveFS.h"
|
|
#include "../Common/Common.h"
|
|
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
// C7zFileEngineIterator
|
|
//
|
|
|
|
class C7zFileEngineIterator : public QAbstractFileEngineIterator
|
|
{
|
|
public:
|
|
#if QT_VERSION < QT_VERSION_CHECK(6, 8, 0)
|
|
C7zFileEngineIterator(QDir::Filters filters, const QStringList& filterNames,
|
|
const QStringList& allEntries)
|
|
: QAbstractFileEngineIterator(filters, filterNames), entries(allEntries), index(0) {}
|
|
~C7zFileEngineIterator() {}
|
|
|
|
QString next() override
|
|
{
|
|
if (!hasNext())
|
|
return QString();
|
|
++index;
|
|
return currentFilePath();
|
|
}
|
|
bool hasNext() const override
|
|
{
|
|
return index < entries.size();
|
|
}
|
|
#else
|
|
C7zFileEngineIterator(const QString &path, QDir::Filters filters, const QStringList& filterNames,
|
|
const QStringList& allEntries)
|
|
: QAbstractFileEngineIterator(path, filters, filterNames), entries(allEntries), index(0) {}
|
|
~C7zFileEngineIterator() {}
|
|
|
|
bool advance() override
|
|
{
|
|
if (index >= entries.size())
|
|
return false;
|
|
++index;
|
|
return true;
|
|
}
|
|
#endif
|
|
|
|
QString currentFileName() const override
|
|
{
|
|
if (index <= 0 || index > entries.size())
|
|
return QString();
|
|
return entries.at(index - 1);
|
|
}
|
|
|
|
private:
|
|
mutable QStringList entries;
|
|
mutable int index;
|
|
};
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
// C7zFileEngine
|
|
//
|
|
|
|
//QAtomicInt g_7zFileEngineCount;
|
|
|
|
C7zFileEngine::C7zFileEngine(const QString& filename, CArchive* pArchive, QMutex* pMutex)
|
|
: _flags(0), _size(0), _pos(0), _pArchive(pArchive), _pMutex(pMutex)
|
|
{
|
|
//g_7zFileEngineCount++;
|
|
setFileName(filename);
|
|
}
|
|
|
|
C7zFileEngine::~C7zFileEngine()
|
|
{
|
|
close();
|
|
//g_7zFileEngineCount--;
|
|
//qDebug() << "g_7zFileEngineCount" << g_7zFileEngineCount;
|
|
}
|
|
|
|
#if QT_VERSION < QT_VERSION_CHECK(6, 0, 0)
|
|
bool C7zFileEngine::open(QIODevice::OpenMode openMode)
|
|
#else
|
|
bool C7zFileEngine::open(QIODevice::OpenMode openMode, std::optional<QFile::Permissions> permissions)
|
|
#endif
|
|
{
|
|
close();
|
|
|
|
//if (openMode & QIODevice::WriteOnly) {
|
|
//
|
|
//} else
|
|
if (openMode & QIODevice::ReadOnly) {
|
|
|
|
QMutexLocker Lock(_pMutex);
|
|
|
|
QMap<int, QIODevice*> Files;
|
|
Files.insert(_index, new QBuffer(&_data));
|
|
_pArchive->Extract(&Files);
|
|
|
|
_size = _data.size();
|
|
_pos = 0;
|
|
|
|
} else
|
|
//if (openMode & QIODevice::Append) {
|
|
//
|
|
//} else
|
|
{
|
|
qWarning("Bad file open mode: %d", (int)openMode);
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
bool C7zFileEngine::close()
|
|
{
|
|
if (!isOpened())
|
|
return false;
|
|
|
|
_data.clear();
|
|
_size = 0;
|
|
_pos = 0;
|
|
|
|
return true;
|
|
}
|
|
|
|
bool C7zFileEngine::flush()
|
|
{
|
|
return true;
|
|
}
|
|
|
|
qint64 C7zFileEngine::size() const
|
|
{
|
|
return _size;
|
|
}
|
|
|
|
qint64 C7zFileEngine::pos() const
|
|
{
|
|
return _pos;
|
|
}
|
|
|
|
bool C7zFileEngine::seek(qint64 pos)
|
|
{
|
|
_pos = pos;
|
|
return true;
|
|
}
|
|
|
|
bool C7zFileEngine::isSequential() const
|
|
{
|
|
return false;
|
|
}
|
|
|
|
bool C7zFileEngine::remove()
|
|
{
|
|
return false;
|
|
}
|
|
|
|
bool C7zFileEngine::mkdir(const QString& dirName, bool createParentDirectories) const
|
|
{
|
|
return false;
|
|
}
|
|
|
|
bool C7zFileEngine::rmdir(const QString& dirName, bool recurseParentDirectories) const
|
|
{
|
|
return false;
|
|
}
|
|
|
|
bool C7zFileEngine::caseSensitive() const
|
|
{
|
|
return false;
|
|
}
|
|
|
|
bool C7zFileEngine::isRelativePath() const
|
|
{
|
|
return false;
|
|
}
|
|
|
|
QAbstractFileEngine::Iterator *C7zFileEngine::beginEntryList(QDir::Filters filters, const QStringList &filterNames)
|
|
{
|
|
QMutexLocker Lock(_pMutex);
|
|
|
|
QStringList allEntries;
|
|
for (int i = 0; i < _pArchive->FileCount(); i++) {
|
|
QString Path = _pArchive->FileProperty(i, "Path").toString();
|
|
if (!_filename.isEmpty()) {
|
|
if (!Path.startsWith(QString(_filename).replace("/", "\\")))
|
|
continue;
|
|
Path = Path.mid(_filename.length() + 1);
|
|
}
|
|
if (!Path.isEmpty() && !Path.contains("\\"))
|
|
allEntries.append(Path);
|
|
}
|
|
|
|
#if QT_VERSION < QT_VERSION_CHECK(6, 8, 0)
|
|
return new C7zFileEngineIterator(filters, filterNames, allEntries);
|
|
#else
|
|
return new C7zFileEngineIterator("", filters, filterNames, allEntries);
|
|
#endif
|
|
}
|
|
|
|
QAbstractFileEngine::FileFlags C7zFileEngine::fileFlags(FileFlags type) const
|
|
{
|
|
return type & _flags;
|
|
}
|
|
|
|
QString C7zFileEngine::fileName(FileName file) const
|
|
{
|
|
auto filename = Split2(_filename, "/", true);
|
|
if (filename.second.isEmpty()) {
|
|
filename.second = filename.first;
|
|
filename.first.clear();
|
|
}
|
|
switch (file)
|
|
{
|
|
case BaseName:
|
|
case AbsoluteName:
|
|
return filename.second;
|
|
case PathName:
|
|
case AbsolutePathName:
|
|
return filename.first;
|
|
//default:
|
|
//case DefaultName:
|
|
}
|
|
return _filename;
|
|
}
|
|
|
|
#if QT_VERSION < QT_VERSION_CHECK(6, 7, 0)
|
|
QDateTime C7zFileEngine::fileTime(FileTime time) const
|
|
#else
|
|
QDateTime C7zFileEngine::fileTime(QFile::FileTime time) const
|
|
#endif
|
|
{
|
|
switch (time)
|
|
{
|
|
#if QT_VERSION < QT_VERSION_CHECK(6, 7, 0)
|
|
case QAbstractFileEngine::ModificationTime:
|
|
#else
|
|
case QFile::FileModificationTime:
|
|
#endif
|
|
default:
|
|
return _datetime;
|
|
break;
|
|
};
|
|
}
|
|
|
|
void C7zFileEngine::setFileName(const QString& file)
|
|
{
|
|
int pos = file.indexOf(":") + 1;
|
|
_filename = file.mid(pos);
|
|
while (_filename.left(1) == "\\" || _filename.left(1) == "/")
|
|
_filename.remove(0, 1);
|
|
|
|
if (_filename.isEmpty()) { // root
|
|
_flags = ExistsFlag | DirectoryType | ReadOwnerPerm | ReadUserPerm | ReadGroupPerm | ReadOtherPerm;
|
|
return;
|
|
}
|
|
|
|
QMutexLocker Lock(_pMutex);
|
|
_index = _pArchive->FindByPath(_filename);
|
|
if (_index != -1) {
|
|
_flags = ExistsFlag | ReadOwnerPerm | ReadUserPerm | ReadGroupPerm | ReadOtherPerm;
|
|
if (_pArchive->FileProperty(_index, "IsDir").toBool())
|
|
_flags |= DirectoryType;
|
|
else
|
|
_flags |= FileType;
|
|
}
|
|
}
|
|
|
|
bool C7zFileEngine::atEnd() const
|
|
{
|
|
return _pos >= _data.size();
|
|
}
|
|
|
|
qint64 C7zFileEngine::read(char* data, qint64 maxlen)
|
|
{
|
|
qint64 togo = qMin(maxlen, _size - _pos);
|
|
if (togo > 0) {
|
|
memcpy(data, _data.constData() + _pos, togo);
|
|
_pos += togo;
|
|
}
|
|
return togo;
|
|
}
|
|
|
|
qint64 C7zFileEngine::readLine(char* data, qint64 maxlen)
|
|
{
|
|
return -1; // todo
|
|
}
|
|
|
|
qint64 C7zFileEngine::write(const char* data, qint64 len)
|
|
{
|
|
return -1;
|
|
}
|
|
|
|
bool C7zFileEngine::isOpened() const
|
|
{
|
|
return _size != 0;
|
|
}
|
|
|
|
QFile::FileError C7zFileEngine::error() const
|
|
{
|
|
return QFile::UnspecifiedError;
|
|
}
|
|
|
|
QString C7zFileEngine::errorString() const
|
|
{
|
|
return "";
|
|
}
|
|
|
|
bool C7zFileEngine::supportsExtension(Extension extension) const
|
|
{
|
|
return extension == QAbstractFileEngine::AtEndExtension;
|
|
}
|
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
// C7zFileEngineHandler
|
|
//
|
|
|
|
C7zFileEngineHandler::C7zFileEngineHandler(const QString& Scheme, QObject* parent)
|
|
: QObject(parent), m_pArchive(NULL)
|
|
{
|
|
m_Scheme = Scheme + ":";
|
|
}
|
|
|
|
C7zFileEngineHandler::~C7zFileEngineHandler()
|
|
{
|
|
Close();
|
|
}
|
|
|
|
bool C7zFileEngineHandler::Open(const QString& ArchivePath)
|
|
{
|
|
Close();
|
|
|
|
CArchive* pArchive = new CArchive(ArchivePath);
|
|
if (pArchive->Open() != ERR_7Z_OK) {
|
|
delete pArchive;
|
|
return false;
|
|
}
|
|
m_pArchive = pArchive;
|
|
return true;
|
|
}
|
|
|
|
void C7zFileEngineHandler::Close()
|
|
{
|
|
delete m_pArchive;
|
|
m_pArchive = NULL;
|
|
}
|
|
|
|
#if QT_VERSION < QT_VERSION_CHECK(6, 8, 0)
|
|
QAbstractFileEngine* C7zFileEngineHandler::create(const QString& filename) const
|
|
{
|
|
if (m_pArchive && filename.startsWith(m_Scheme))
|
|
return new C7zFileEngine(filename, m_pArchive, &m_Mutex);
|
|
|
|
return NULL;
|
|
}
|
|
#else
|
|
std::unique_ptr<QAbstractFileEngine> C7zFileEngineHandler::create(const QString& filename) const
|
|
{
|
|
if (m_pArchive && filename.startsWith(m_Scheme))
|
|
return std::unique_ptr<QAbstractFileEngine>(new C7zFileEngine(filename, m_pArchive, &m_Mutex));
|
|
|
|
return std::unique_ptr<QAbstractFileEngine>();
|
|
}
|
|
#endif
|