2007-04-04 16:01:32 +04:00
|
|
|
/*
|
|
|
|
* Bittorrent Client using Qt4 and libtorrent.
|
|
|
|
* Copyright (C) 2006 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 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, write to the Free Software
|
|
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
|
|
*
|
|
|
|
* Contact : chris@qbittorrent.org
|
|
|
|
*/
|
|
|
|
#include "FinishedTorrents.h"
|
|
|
|
#include "misc.h"
|
|
|
|
#include "GUI.h"
|
2007-04-04 16:52:50 +04:00
|
|
|
#include "properties_imp.h"
|
2007-04-10 14:50:34 +04:00
|
|
|
#include "allocationDlg.h"
|
2007-04-04 16:01:32 +04:00
|
|
|
#include <QFile>
|
2007-04-04 17:12:11 +04:00
|
|
|
#include <QSettings>
|
2007-04-04 16:01:32 +04:00
|
|
|
|
|
|
|
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);
|
2007-04-11 22:31:21 +04:00
|
|
|
// Hide ETA & hash column
|
2007-04-04 16:01:32 +04:00
|
|
|
finishedList->hideColumn(HASH);
|
2007-04-11 22:31:21 +04:00
|
|
|
finishedList->hideColumn(ETA);
|
|
|
|
finishedList->hideColumn(DLSPEED);
|
2007-04-11 17:10:13 +04:00
|
|
|
// Load last columns width for download list
|
|
|
|
if(!loadColWidthFinishedList()){
|
|
|
|
finishedList->header()->resizeSection(0, 200);
|
|
|
|
}
|
2007-04-11 17:19:58 +04:00
|
|
|
// 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)));
|
2007-04-11 22:31:21 +04:00
|
|
|
finishedListDelegate = new DLListDelegate(finishedList);
|
2007-04-04 16:01:32 +04:00
|
|
|
finishedList->setItemDelegate(finishedListDelegate);
|
2007-04-04 17:12:11 +04:00
|
|
|
connect(finishedList, SIGNAL(customContextMenuRequested(const QPoint&)), this, SLOT(displayFinishedListMenu(const QPoint&)));
|
2007-04-11 17:23:17 +04:00
|
|
|
connect(finishedList, SIGNAL(doubleClicked(const QModelIndex&)), this, SLOT(propertiesSelection()));
|
2007-04-04 17:12:11 +04:00
|
|
|
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")));
|
2007-04-10 14:50:34 +04:00
|
|
|
actionSet_upload_limit->setIcon(QIcon(QString::fromUtf8(":/Icons/skin/seeding.png")));
|
2007-04-04 17:12:11 +04:00
|
|
|
connect(actionDelete, SIGNAL(triggered()), (GUI*)parent, SLOT(deleteSelection()));
|
2007-04-04 18:36:47 +04:00
|
|
|
connect(actionPreview_file, SIGNAL(triggered()), (GUI*)parent, SLOT(previewFileSelection()));
|
2007-04-04 17:12:11 +04:00
|
|
|
connect(actionDelete_Permanently, SIGNAL(triggered()), (GUI*)parent, SLOT(deletePermanently()));
|
2007-04-04 18:29:59 +04:00
|
|
|
connect(actionTorrent_Properties, SIGNAL(triggered()), this, SLOT(propertiesSelection()));
|
2007-04-04 16:01:32 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
FinishedTorrents::~FinishedTorrents(){
|
2007-04-11 17:10:13 +04:00
|
|
|
saveColWidthFinishedList();
|
2007-04-04 16:01:32 +04:00
|
|
|
delete finishedListDelegate;
|
|
|
|
delete finishedListModel;
|
|
|
|
}
|
|
|
|
|
|
|
|
void FinishedTorrents::addFinishedSHA(QString 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; i<finishedListModel->columnCount(); ++i){
|
|
|
|
finishedListModel->setData(finishedListModel->index(row, i), QVariant(QColor(color)), Qt::TextColorRole);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-04-11 17:19:58 +04:00
|
|
|
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<QPair<int, double> > lines;
|
|
|
|
// insertion sorting
|
|
|
|
unsigned int nbRows = finishedListModel->rowCount();
|
|
|
|
for(unsigned int i=0; i<nbRows; ++i){
|
|
|
|
misc::insertSort(lines, QPair<int,double>(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; row<nbRows_old; ++row){
|
|
|
|
finishedListModel->insertRow(finishedListModel->rowCount());
|
|
|
|
unsigned int sourceRow = lines[row].first;
|
|
|
|
unsigned int nbColumns = finishedListModel->columnCount();
|
|
|
|
for(unsigned int col=0; col<nbColumns; ++col){
|
|
|
|
finishedListModel->setData(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<QPair<int, QString> > lines;
|
|
|
|
// Insertion sorting
|
|
|
|
unsigned int nbRows = finishedListModel->rowCount();
|
|
|
|
for(unsigned int i=0; i<nbRows; ++i){
|
|
|
|
misc::insertSortString(lines, QPair<int, QString>(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; row<nbRows_old; ++row){
|
|
|
|
finishedListModel->insertRow(finishedListModel->rowCount());
|
|
|
|
unsigned int sourceRow = lines[row].first;
|
|
|
|
unsigned int nbColumns = finishedListModel->columnCount();
|
|
|
|
for(unsigned int col=0; col<nbColumns; ++col){
|
|
|
|
finishedListModel->setData(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);
|
|
|
|
}
|
|
|
|
|
2007-04-11 17:10:13 +04:00
|
|
|
// 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; i<listSize; ++i){
|
|
|
|
finishedList->header()->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; i<nbColumns; ++i){
|
|
|
|
width_list << QString(misc::toString(finishedList->columnWidth(i)).c_str());
|
|
|
|
}
|
|
|
|
settings.setValue("FinishedListColsWidth", width_list.join(" "));
|
|
|
|
qDebug("Finished list columns width saved");
|
|
|
|
}
|
|
|
|
|
2007-04-10 14:11:36 +04:00
|
|
|
void FinishedTorrents::on_actionSet_upload_limit_triggered(){
|
2007-04-10 14:50:34 +04:00
|
|
|
QModelIndexList selectedIndexes = finishedList->selectionModel()->selectedIndexes();
|
2007-04-10 14:11:36 +04:00
|
|
|
QModelIndex index;
|
|
|
|
QStringList hashes;
|
|
|
|
foreach(index, selectedIndexes){
|
|
|
|
if(index.column() == NAME){
|
|
|
|
// Get the file hash
|
2007-04-10 14:50:34 +04:00
|
|
|
hashes << finishedListModel->data(finishedListModel->index(index.row(), HASH)).toString();
|
2007-04-10 14:11:36 +04:00
|
|
|
}
|
|
|
|
}
|
2007-04-10 14:50:34 +04:00
|
|
|
new BandwidthAllocationDialog(this, true, BTSession, hashes);
|
2007-04-10 14:11:36 +04:00
|
|
|
}
|
|
|
|
|
2007-04-04 16:01:32 +04:00
|
|
|
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()) continue;
|
|
|
|
torrent_status torrentStatus = h.status();
|
|
|
|
if(torrentStatus.state == torrent_status::downloading) {
|
|
|
|
// What are you doing here, go back to download tab!
|
|
|
|
qDebug("Info: a torrent was moved from finished to download tab");
|
|
|
|
deleteFromFinishedList(hash);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
QList<QStandardItem *> items = finishedListModel->findItems(hash, Qt::MatchExactly, HASH );
|
|
|
|
if(items.size() != 1){
|
|
|
|
qDebug("Problem: Can't find torrent in finished list");
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
int row = items.at(0)->row();
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Will move it to download tab
|
|
|
|
void FinishedTorrents::deleteFromFinishedList(QString hash){
|
|
|
|
finishedSHAs.removeAll(hash);
|
|
|
|
QList<QStandardItem *> items = finishedListModel->findItems(hash, Qt::MatchExactly, HASH );
|
|
|
|
if(items.size() != 1){
|
|
|
|
qDebug("Problem: Can't delete torrent from finished list");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
finishedListModel->removeRow(finishedListModel->indexFromItem(items.at(0)).row());
|
|
|
|
QFile::remove(misc::qBittorrentPath()+"BT_backup"+QDir::separator()+hash+".finished");
|
|
|
|
--nbFinished;
|
|
|
|
((GUI*)parent)->setTabText(1, tr("Finished") +" ("+QString(misc::toString(nbFinished).c_str())+")");
|
|
|
|
}
|
2007-04-04 16:33:53 +04:00
|
|
|
|
|
|
|
QTreeView* FinishedTorrents::getFinishedList(){
|
|
|
|
return finishedList;
|
|
|
|
}
|
|
|
|
|
|
|
|
QStandardItemModel* FinishedTorrents::getFinishedListModel(){
|
|
|
|
return finishedListModel;
|
|
|
|
}
|
2007-04-04 16:52:50 +04:00
|
|
|
|
|
|
|
// 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, 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);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2007-04-04 17:12:11 +04:00
|
|
|
|
|
|
|
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);
|
2007-04-10 14:11:36 +04:00
|
|
|
myFinishedListMenu.addAction(actionSet_upload_limit);
|
2007-04-04 17:12:11 +04:00
|
|
|
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));
|
|
|
|
}
|