// Ryzom - MMORPG Framework // Copyright (C) 2010 Winch Gate Property Limited // // This program is free software: you can redistribute it and/or modify // it under the terms of the GNU Affero General Public License as // published by the Free Software Foundation, either version 3 of the // License, or (at your option) any later version. // // This program is distributed in the hope that it will be useful, // but WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the // GNU Affero General Public License for more details. // // You should have received a copy of the GNU Affero General Public License // along with this program. If not, see . #include "stdpch.h" #include "filescopier.h" #include "utils.h" #include "operation.h" #include "nel/misc/path.h" #ifdef DEBUG_NEW #define new DEBUG_NEW #endif CFilesCopier::CFilesCopier(IOperationProgressListener *listener):m_listener(listener) { } CFilesCopier::~CFilesCopier() { } void CFilesCopier::setSourceDirectory(const QString &src) { m_sourceDirectory = src; } void CFilesCopier::setDesinationDirectory(const QString &dst) { m_destinationDirectory = dst; } void CFilesCopier::setIncludeFilter(const QStringList &filter) { m_includeFilter = filter; } void CFilesCopier::setExcludeFilter(const QStringList &filter) { m_excludeFilter = filter; } bool CFilesCopier::exec() { if (m_sourceDirectory.isEmpty() || m_destinationDirectory.isEmpty()) return false; if (m_listener) m_listener->operationPrepare(); QDir().mkpath(m_destinationDirectory); FilesToCopy files; CFilesCopier::getFilesList(files); return copyFiles(files); } void CFilesCopier::getFilesList(FilesToCopy &files) { QDir dir(m_sourceDirectory); QFileInfoList entries = dir.entryInfoList(m_includeFilter); foreach(const QFileInfo &entry, entries) { QString fullPath = entry.absoluteFilePath(); QString dstPath = m_destinationDirectory + "/" + dir.relativeFilePath(fullPath); if (entry.isDir()) { QDir().mkpath(dstPath); QDir subDir(fullPath); QDirIterator it(subDir, QDirIterator::Subdirectories); while (it.hasNext()) { fullPath = it.next(); if (it.fileName().startsWith('.')) continue; QFileInfo fileInfo = it.fileInfo(); dstPath = m_destinationDirectory + "/" + dir.relativeFilePath(fullPath); if (fileInfo.isDir()) { QDir().mkpath(dstPath); } else { FileToCopy file; file.filename = it.fileName(); file.src = it.filePath(); file.dst = dstPath; file.size = it.fileInfo().size(); file.date = it.fileInfo().lastModified().toTime_t(); files << file; } } } else { FileToCopy file; file.filename = entry.fileName(); file.src = entry.filePath(); file.dst = dstPath; file.size = entry.size(); file.date = entry.lastModified().toTime_t(); files << file; } } } bool CFilesCopier::copyFiles(const FilesToCopy &files) { qint64 totalSize = 0; foreach(const FileToCopy &file, files) { totalSize += file.size; } if (m_listener) { m_listener->operationInit(0, totalSize); m_listener->operationStart(); } qint64 processedSize = 0; foreach(const FileToCopy &file, files) { if (m_listener && m_listener->operationShouldStop()) { m_listener->operationStop(); return true; } if (m_listener) m_listener->operationProgress(processedSize, file.filename); QFileInfo dstFileInfo(file.dst); if (dstFileInfo.size() != file.size || dstFileInfo.lastModified().toTime_t() != file.date) { // force deleting previous file since it was incomplete QFile::remove(file.dst); if (!QFile::copy(file.src, file.dst)) { if (m_listener) m_listener->operationFail(QApplication::tr("Unable to copy file %1").arg(file.src)); return false; } if (!NLMISC::CFile::setFileModificationDate(qToUtf8(file.dst), file.date)) { qDebug() << "Unable to change date of " << file.dst; } } processedSize += file.size; } if (m_listener) { m_listener->operationSuccess(totalSize); } return true; }