/* * 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. * * 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 General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #include "UBDownloadManager.h" #include "core/UBApplication.h" #include "core/UBPersistenceManager.h" #include "gui/UBMainWindow.h" #include "board/UBBoardController.h" #include "board/UBBoardPaletteManager.h" #include "frameworks/UBFileSystemUtils.h" #include "core/memcheck.h" UBAsyncLocalFileDownloader::UBAsyncLocalFileDownloader(sDownloadFileDesc desc, QObject *parent) : QThread(parent) , mDesc(desc) , m_bAborting(false) { } UBAsyncLocalFileDownloader *UBAsyncLocalFileDownloader::download() { if (!QFile::exists(QUrl(mDesc.srcUrl).toLocalFile())) { qDebug() << "file" << mDesc.srcUrl << "does not present in fs"; return this; } start(); return this; } void UBAsyncLocalFileDownloader::run() { QString mimeType = UBFileSystemUtils::mimeTypeFromFileName(mDesc.srcUrl); int position=mimeType.indexOf(";"); if(position != -1) mimeType=mimeType.left(position); UBMimeType::Enum itemMimeType = UBFileSystemUtils::mimeTypeFromString(mimeType); QString destDirectory; if (UBMimeType::Video == itemMimeType) destDirectory = UBPersistenceManager::videoDirectory; else if (UBMimeType::Audio == itemMimeType) destDirectory = UBPersistenceManager::audioDirectory; if (mDesc.originalSrcUrl.isEmpty()) mDesc.originalSrcUrl = mDesc.srcUrl; QString uuid = QUuid::createUuid(); UBPersistenceManager::persistenceManager()->addFileToDocument(UBApplication::boardController->selectedDocument(), QUrl(mDesc.srcUrl).toLocalFile(), destDirectory, uuid, mTo, NULL); if (m_bAborting) { if (QFile::exists(mTo)) QFile::remove(mTo); } else emit signal_asyncCopyFinished(mDesc.id, !mTo.isEmpty(), QUrl::fromLocalFile(mTo), QUrl::fromLocalFile(mDesc.originalSrcUrl), "", NULL, mDesc.pos, mDesc.size, mDesc.isBackground); } void UBAsyncLocalFileDownloader::abort() { m_bAborting = true; } /** The unique instance of the download manager */ static UBDownloadManager* pInstance = NULL; /** * \brief Constructor * @param parent as the parent widget * @param name as the object name */ UBDownloadManager::UBDownloadManager(QObject *parent, const char *name):QObject(parent) { setObjectName(name); init(); connect(this, SIGNAL(fileAddedToDownload()), this, SLOT(onUpdateDownloadLists())); } /** * \brief Destructor */ UBDownloadManager::~UBDownloadManager() { } /** * \brief Get the download manager * @return a pointer on the download manager */ UBDownloadManager* UBDownloadManager::downloadManager() { if(NULL == pInstance) { pInstance = new UBDownloadManager(); } return pInstance; } void UBDownloadManager::destroy() { if(pInstance) { delete pInstance; } pInstance = NULL; } /** * \brief Add a file to the download list * @param desc as the given file description */ int UBDownloadManager::addFileToDownload(sDownloadFileDesc desc) { // Set the ID for this download desc.id = mLastID; mLastID++; // Add the file to the pending download list mPendingDL.append(desc); // If the download is modal, show the download dialog if(desc.modal) { // Update the download order (priority to modal files) updateDownloadOrder(); UBApplication::mainWindow->showDownloadWidget(); } UBApplication::boardController->paletteManager()->startDownloads(); emit fileAddedToDownload(); return desc.id; } /** * \brief Initialize the download manager */ void UBDownloadManager::init() { mCrntDL.clear(); mPendingDL.clear(); mDownloads.clear(); mLastID = 1; mDLAvailability.clear(); for(int i=0; i modalFiles; QVector nonModalfiles; for(int i=0; i UBDownloadManager::currentDownloads() { return mCrntDL; } /** * \brief Get the list of the pending downloads * @return a QVector of pending downloads */ QVector UBDownloadManager::pendingDownloads() { return mPendingDL; } /** * \brief Update the file transfer information * @param desc as the current downloaded file description */ void UBDownloadManager::onDownloadProgress(int id, qint64 received, qint64 total) { updateFileCurrentSize(id, received, total); } /** * \brief Called when the download of the given file is finished * @param desc as the current downloaded file description */ void UBDownloadManager::onDownloadFinished(int id, bool pSuccess, QUrl sourceUrl, QUrl contentUrl, QString pContentTypeHeader, QByteArray pData, QPointF pPos, QSize pSize, bool isBackground) { // Temporary data for dnd do not delete it please Q_UNUSED(pPos) Q_UNUSED(pSize) Q_UNUSED(isBackground) for(int i=0; i= 0 && total >= 0) { // ------------------------------------- // [=============== x% ==== ] // ------------------------------------- desc.currentSize = received; desc.totalSize = total; emit downloadUpdated(id, received, total); } else { // ------------------------------------- // [=============== 100% ==============] // ------------------------------------- // received and total are negative. That means that the download is finished desc.currentSize = mCrntDL.at(i).totalSize; // Remove the finished file from the current download list mCrntDL.remove(i); // Here we don't forget to remove the reply related to the finished download mDownloads.remove(id); // Free the download slot used by the finished file for(int j=0; j(cpHelper->download()); if (!res) delete res; else mDownloads[desc.id] = res; } else { UBDownloadHttpFile* http = new UBDownloadHttpFile(desc.id, this); connect(http, SIGNAL(downloadProgress(int, qint64,qint64)), this, SLOT(onDownloadProgress(int,qint64,qint64))); connect(http, SIGNAL(downloadFinished(int, bool, QUrl, QUrl, QString, QByteArray, QPointF, QSize, bool)), this, SLOT(onDownloadFinished(int, bool, QUrl, QUrl, QString, QByteArray, QPointF, QSize, bool))); //the desc.srcUrl is encoded. So we have to decode it before. QUrl url; url.setEncodedUrl(desc.srcUrl.toUtf8()); // We send here the request and store its reply in order to be able to cancel it if needed mDownloads[desc.id] = dynamic_cast(http->get(url, desc.pos, desc.size, desc.isBackground)); } } /** * \brief Verify if modal downloads remains and notify everyone if it is not the case. */ void UBDownloadManager::checkIfModalRemains() { bool bModal = false; for(int i=0; ihideDownloadWidget(); // Notify that no modal downloads are pending emit downloadModalFinished(); } } /** * \brief Cancel all downloads */ void UBDownloadManager::cancelDownloads() { // Stop the current downloads QMap::iterator it = mDownloads.begin(); for(; it!=mDownloads.end();it++) { QNetworkReply *netReply = dynamic_cast(it.value()); if (netReply) netReply->abort(); else { UBAsyncLocalFileDownloader *localDownload = dynamic_cast(it.value()); if (localDownload) localDownload->abort(); } } // Clear all the lists init(); checkIfModalRemains(); finishDownloads(true); } void UBDownloadManager::onDownloadError(int id) { QNetworkReply *pReply = dynamic_cast(mDownloads.value(id)); if(NULL != pReply) { // Check which error occured: switch(pReply->error()) { case QNetworkReply::OperationCanceledError: // For futur developments: do something in case of download aborting (message? remove the download?) break; default: // Check the documentation of QNetworkReply in Qt Assistant for the different error cases break; } } } void UBDownloadManager::finishDownloads(bool cancel) { UBApplication::boardController->paletteManager()->stopDownloads(); if(cancel){ emit cancelAllDownloads(); } else{ emit allDownloadsFinished(); } } void UBDownloadManager::cancelDownload(int id) { if (!mDownloads.size()) return; // Stop the download QNetworkReply *pNetworkDownload = dynamic_cast(mDownloads[id]); if (pNetworkDownload) pNetworkDownload->abort(); else { UBAsyncLocalFileDownloader *pLocalDownload = dynamic_cast(mDownloads[id]); if (pLocalDownload) { if (pLocalDownload->isRunning()) pLocalDownload->abort(); } } mDownloads.remove(id); // Remove the canceled download from the download lists bool bFound = false; for(int i=0; i