/* * Bittorrent Client using Qt4 and libtorrent. * Copyright (C) 2006-2007 Christophe Dumez * * 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 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 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 "FinishedTorrents.h" #include "misc.h" #include "GUI.h" #include "properties_imp.h" #include "allocationDlg.h" #include #include FinishedTorrents::FinishedTorrents(QObject *parent, bittorrent *BTSession){ setupUi(this); nbFinished = 0; this->parent = parent; this->BTSession = BTSession; finishedListModel = new QStandardItemModel(0,9); finishedListModel->setHeaderData(NAME, Qt::Horizontal, tr("Name", "i.e: file name")); finishedListModel->setHeaderData(SIZE, Qt::Horizontal, tr("Size", "i.e: file size")); finishedListModel->setHeaderData(PROGRESS, Qt::Horizontal, tr("Progress", "i.e: % downloaded")); finishedListModel->setHeaderData(DLSPEED, Qt::Horizontal, tr("DL Speed", "i.e: Download speed")); finishedListModel->setHeaderData(UPSPEED, Qt::Horizontal, tr("UP Speed", "i.e: Upload speed")); finishedListModel->setHeaderData(SEEDSLEECH, Qt::Horizontal, tr("Seeds/Leechs", "i.e: full/partial sources")); finishedListModel->setHeaderData(STATUS, Qt::Horizontal, tr("Status")); finishedListModel->setHeaderData(ETA, Qt::Horizontal, tr("ETA", "i.e: Estimated Time of Arrival / Time left")); finishedList->setModel(finishedListModel); // Hide ETA & hash column finishedList->hideColumn(HASH); finishedList->hideColumn(ETA); finishedList->hideColumn(DLSPEED); // Load last columns width for download list if(!loadColWidthFinishedList()){ finishedList->header()->resizeSection(0, 200); } // Make download list header clickable for sorting finishedList->header()->setClickable(true); finishedList->header()->setSortIndicatorShown(true); connect(finishedList->header(), SIGNAL(sectionPressed(int)), this, SLOT(sortFinishedList(int))); finishedListDelegate = new DLListDelegate(finishedList); finishedList->setItemDelegate(finishedListDelegate); connect(finishedList, SIGNAL(customContextMenuRequested(const QPoint&)), this, SLOT(displayFinishedListMenu(const QPoint&))); connect(finishedList, SIGNAL(doubleClicked(const QModelIndex&)), this, SLOT(propertiesSelection())); actionDelete->setIcon(QIcon(QString::fromUtf8(":/Icons/skin/delete.png"))); actionPreview_file->setIcon(QIcon(QString::fromUtf8(":/Icons/skin/preview.png"))); actionDelete_Permanently->setIcon(QIcon(QString::fromUtf8(":/Icons/skin/delete_perm.png"))); actionTorrent_Properties->setIcon(QIcon(QString::fromUtf8(":/Icons/skin/properties.png"))); actionSet_upload_limit->setIcon(QIcon(QString::fromUtf8(":/Icons/skin/seeding.png"))); connect(actionDelete, SIGNAL(triggered()), (GUI*)parent, SLOT(on_actionDelete_triggered())); connect(actionPreview_file, SIGNAL(triggered()), (GUI*)parent, SLOT(on_actionPreview_file_triggered())); connect(actionDelete_Permanently, SIGNAL(triggered()), (GUI*)parent, SLOT(on_actionDelete_Permanently_triggered())); connect(actionTorrent_Properties, SIGNAL(triggered()), this, SLOT(propertiesSelection())); } FinishedTorrents::~FinishedTorrents(){ saveColWidthFinishedList(); delete finishedListDelegate; delete finishedListModel; } void FinishedTorrents::addFinishedSHA(QString hash){ if(BTSession->getUncheckedTorrentsList().indexOf(hash) != -1){ BTSession->setTorrentFinishedChecking(hash); } if(finishedSHAs.indexOf(hash) == -1) { finishedSHAs << hash; int row = finishedListModel->rowCount(); torrent_handle h = BTSession->getTorrentHandle(hash); // Adding torrent to download list finishedListModel->insertRow(row); finishedListModel->setData(finishedListModel->index(row, NAME), QVariant(h.name().c_str())); finishedListModel->setData(finishedListModel->index(row, SIZE), QVariant((qlonglong)h.get_torrent_info().total_size())); finishedListModel->setData(finishedListModel->index(row, DLSPEED), QVariant((double)0.)); finishedListModel->setData(finishedListModel->index(row, UPSPEED), QVariant((double)0.)); finishedListModel->setData(finishedListModel->index(row, SEEDSLEECH), QVariant("0/0")); finishedListModel->setData(finishedListModel->index(row, ETA), QVariant((qlonglong)-1)); finishedListModel->setData(finishedListModel->index(row, HASH), QVariant(hash)); finishedListModel->setData(finishedListModel->index(row, PROGRESS), QVariant((double)1.)); // Start the torrent if it was paused if(h.is_paused()) { h.resume(); if(QFile::exists(misc::qBittorrentPath()+"BT_backup"+QDir::separator()+hash+".paused")) { QFile::remove(misc::qBittorrentPath()+"BT_backup"+QDir::separator()+hash+".paused"); } } finishedListModel->setData(finishedListModel->index(row, STATUS), QVariant(tr("Finished", "i.e: Torrent has finished downloading"))); finishedListModel->setData(finishedListModel->index(row, NAME), QVariant(QIcon(":/Icons/skin/seeding.png")), Qt::DecorationRole); setRowColor(row, "orange"); // Create .finished file QFile finished_file(misc::qBittorrentPath()+"BT_backup"+QDir::separator()+hash+".finished"); finished_file.open(QIODevice::WriteOnly | QIODevice::Text); finished_file.close(); // Update the number of finished torrents ++nbFinished; ((GUI*)parent)->setTabText(1, tr("Finished") +" ("+QString(misc::toString(nbFinished).c_str())+")"); } else { qDebug("Problem: this torrent (%s) has finished twice...", hash.toStdString().c_str()); } } // Set the color of a row in data model void FinishedTorrents::setRowColor(int row, const QString& color){ for(int i=0; icolumnCount(); ++i){ finishedListModel->setData(finishedListModel->index(row, i), QVariant(QColor(color)), Qt::TextColorRole); } } void FinishedTorrents::sortFinishedList(int index){ static Qt::SortOrder sortOrder = Qt::AscendingOrder; if(finishedList->header()->sortIndicatorSection() == index){ if(sortOrder == Qt::AscendingOrder){ sortOrder = Qt::DescendingOrder; }else{ sortOrder = Qt::AscendingOrder; } } finishedList->header()->setSortIndicator(index, sortOrder); switch(index){ case SIZE: case ETA: case UPSPEED: case DLSPEED: case PROGRESS: sortFinishedListFloat(index, sortOrder); break; default: sortFinishedListString(index, sortOrder); } } void FinishedTorrents::sortFinishedListFloat(int index, Qt::SortOrder sortOrder){ QList > lines; // insertion sorting unsigned int nbRows = finishedListModel->rowCount(); for(unsigned int i=0; i(i, finishedListModel->data(finishedListModel->index(i, index)).toDouble()), sortOrder); } // Insert items in new model, in correct order unsigned int nbRows_old = lines.size(); for(unsigned int row=0; rowinsertRow(finishedListModel->rowCount()); unsigned int sourceRow = lines[row].first; unsigned int nbColumns = finishedListModel->columnCount(); for(unsigned int col=0; colsetData(finishedListModel->index(nbRows_old+row, col), finishedListModel->data(finishedListModel->index(sourceRow, col))); finishedListModel->setData(finishedListModel->index(nbRows_old+row, col), finishedListModel->data(finishedListModel->index(sourceRow, col), Qt::DecorationRole), Qt::DecorationRole); finishedListModel->setData(finishedListModel->index(nbRows_old+row, col), finishedListModel->data(finishedListModel->index(sourceRow, col), Qt::TextColorRole), Qt::TextColorRole); } } // Remove old rows finishedListModel->removeRows(0, nbRows_old); } void FinishedTorrents::sortFinishedListString(int index, Qt::SortOrder sortOrder){ QList > lines; // Insertion sorting unsigned int nbRows = finishedListModel->rowCount(); for(unsigned int i=0; i(i, finishedListModel->data(finishedListModel->index(i, index)).toString()), sortOrder); } // Insert items in new model, in correct order unsigned int nbRows_old = lines.size(); for(unsigned int row=0; rowinsertRow(finishedListModel->rowCount()); unsigned int sourceRow = lines[row].first; unsigned int nbColumns = finishedListModel->columnCount(); for(unsigned int col=0; colsetData(finishedListModel->index(nbRows_old+row, col), finishedListModel->data(finishedListModel->index(sourceRow, col))); finishedListModel->setData(finishedListModel->index(nbRows_old+row, col), finishedListModel->data(finishedListModel->index(sourceRow, col), Qt::DecorationRole), Qt::DecorationRole); finishedListModel->setData(finishedListModel->index(nbRows_old+row, col), finishedListModel->data(finishedListModel->index(sourceRow, col), Qt::TextColorRole), Qt::TextColorRole); } } // Remove old rows finishedListModel->removeRows(0, nbRows_old); } // Load columns width in a file that were saved previously // (finished list) bool FinishedTorrents::loadColWidthFinishedList(){ qDebug("Loading columns width for finished list"); QSettings settings("qBittorrent", "qBittorrent"); QString line = settings.value("FinishedListColsWidth", QString()).toString(); if(line.isEmpty()) return false; QStringList width_list = line.split(' '); if(width_list.size() != finishedListModel->columnCount()) return false; unsigned int listSize = width_list.size(); for(unsigned int i=0; iheader()->resizeSection(i, width_list.at(i).toInt()); } qDebug("Finished list columns width loaded"); return true; } // Save columns width in a file to remember them // (finished list) void FinishedTorrents::saveColWidthFinishedList() const{ qDebug("Saving columns width in finished list"); QSettings settings("qBittorrent", "qBittorrent"); QStringList width_list; unsigned int nbColumns = finishedListModel->columnCount(); for(unsigned int i=0; icolumnWidth(i)).c_str()); } settings.setValue("FinishedListColsWidth", width_list.join(" ")); qDebug("Finished list columns width saved"); } void FinishedTorrents::on_actionSet_upload_limit_triggered(){ QModelIndexList selectedIndexes = finishedList->selectionModel()->selectedIndexes(); QModelIndex index; QStringList hashes; foreach(index, selectedIndexes){ if(index.column() == NAME){ // Get the file hash hashes << finishedListModel->data(finishedListModel->index(index.row(), HASH)).toString(); } } new BandwidthAllocationDialog(this, true, BTSession, hashes); } void FinishedTorrents::updateFinishedList(){ // qDebug("Updating finished list"); QString hash; foreach(hash, finishedSHAs){ torrent_handle h = BTSession->getTorrentHandle(hash); if(!h.is_valid()){ qDebug("Problem: This torrent is not valid in finished list"); continue; } if(h.is_paused()){ h.resume(); // No paused torrents in finished list } torrent_status torrentStatus = h.status(); if(torrentStatus.state == torrent_status::downloading || (torrentStatus.state != torrent_status::checking_files && torrentStatus.state != torrent_status::queued_for_checking && torrentStatus.progress != 1.)) { // What are you doing here, go back to download tab! qDebug("Info: a torrent was moved from finished to download tab"); deleteFromFinishedList(hash); emit torrentMovedFromFinishedList(h); continue; } int row = getRowFromHash(hash); if(row == -1){ std::cerr << "ERROR: Can't find torrent in finished list\n"; continue; } finishedListModel->setData(finishedListModel->index(row, UPSPEED), QVariant((double)torrentStatus.upload_payload_rate)); finishedListModel->setData(finishedListModel->index(row, SEEDSLEECH), QVariant(QString(misc::toString(torrentStatus.num_seeds, true).c_str())+"/"+QString(misc::toString(torrentStatus.num_peers - torrentStatus.num_seeds, true).c_str()))); } } QStringList FinishedTorrents::getFinishedSHAs(){ return finishedSHAs; } int FinishedTorrents::getRowFromHash(const QString& hash) const{ unsigned int nbRows = finishedListModel->rowCount(); for(unsigned int i=0; idata(finishedListModel->index(i, HASH)) == hash){ return i; } } return -1; } // Will move it to download tab void FinishedTorrents::deleteFromFinishedList(QString hash){ int row = getRowFromHash(hash); if(row == -1){ std::cerr << "Error: couldn't find hash in finished list\n"; } finishedListModel->removeRow(row); QFile::remove(misc::qBittorrentPath()+"BT_backup"+QDir::separator()+hash+".finished"); --nbFinished; ((GUI*)parent)->setTabText(1, tr("Finished") +" ("+QString(misc::toString(nbFinished).c_str())+")"); finishedSHAs.removeAll(hash); } QTreeView* FinishedTorrents::getFinishedList(){ return finishedList; } QStandardItemModel* FinishedTorrents::getFinishedListModel(){ return finishedListModel; } // Show torrent properties dialog void FinishedTorrents::showProperties(const QModelIndex &index){ int row = index.row(); QString fileHash = finishedListModel->data(finishedListModel->index(row, HASH)).toString(); torrent_handle h = BTSession->getTorrentHandle(fileHash); QStringList errors = ((GUI*)parent)->trackerErrors.value(fileHash, QStringList(tr("None", "i.e: No error message"))); properties *prop = new properties(this, BTSession, h, errors); connect(prop, SIGNAL(changedFilteredFiles(torrent_handle, bool)), BTSession, SLOT(reloadTorrent(torrent_handle, bool))); prop->show(); } // display properties of selected items void FinishedTorrents::propertiesSelection(){ QModelIndexList selectedIndexes = finishedList->selectionModel()->selectedIndexes(); QModelIndex index; foreach(index, selectedIndexes){ if(index.column() == NAME){ showProperties(index); } } } void FinishedTorrents::displayFinishedListMenu(const QPoint& pos){ QMenu myFinishedListMenu(this); QModelIndex index; // Enable/disable pause/start action given the DL state QModelIndexList selectedIndexes = finishedList->selectionModel()->selectedIndexes(); QSettings settings("qBittorrent", "qBittorrent"); QString previewProgram = settings.value("Options/Misc/PreviewProgram", QString()).toString(); foreach(index, selectedIndexes){ if(index.column() == NAME){ // Get the file name QString fileHash = finishedListModel->data(finishedListModel->index(index.row(), HASH)).toString(); // Get handle and pause the torrent torrent_handle h = BTSession->getTorrentHandle(fileHash); myFinishedListMenu.addAction(actionDelete); myFinishedListMenu.addAction(actionDelete_Permanently); myFinishedListMenu.addAction(actionSet_upload_limit); myFinishedListMenu.addAction(actionTorrent_Properties); if(!previewProgram.isEmpty() && BTSession->isFilePreviewPossible(fileHash) && selectedIndexes.size()<=finishedListModel->columnCount()){ myFinishedListMenu.addAction(actionPreview_file); } break; } } // Call menu // XXX: why mapToGlobal() is not enough? myFinishedListMenu.exec(mapToGlobal(pos)+QPoint(10,55)); }