|
|
|
/*
|
|
|
|
* 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.
|
|
|
|
*
|
|
|
|
* In addition, as a special exception, the copyright holders give permission to
|
|
|
|
* link this program with the OpenSSL project's "OpenSSL" library (or with
|
|
|
|
* modified versions of it that use the same license as the "OpenSSL" library),
|
|
|
|
* and distribute the linked executables. You must obey the GNU General Public
|
|
|
|
* License in all respects for all of the code used other than "OpenSSL". If you
|
|
|
|
* modify file(s), you may extend this exception to your version of the file(s),
|
|
|
|
* but you are not obligated to do so. If you do not wish to do so, delete this
|
|
|
|
* exception statement from your version.
|
|
|
|
*
|
|
|
|
* Contact : chris@qbittorrent.org
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <QTreeWidgetItem>
|
|
|
|
#include <QStringList>
|
|
|
|
#include <QMenu>
|
|
|
|
#include <QSettings>
|
|
|
|
#include <QHash>
|
|
|
|
#include <QAction>
|
|
|
|
#include <QColor>
|
|
|
|
#include <libtorrent/version.hpp>
|
|
|
|
|
|
|
|
#include "trackerlist.h"
|
|
|
|
#include "propertieswidget.h"
|
|
|
|
#include "trackersadditiondlg.h"
|
|
|
|
#include "misc.h"
|
|
|
|
#include "bittorrent.h"
|
|
|
|
|
|
|
|
TrackerList::TrackerList(PropertiesWidget *properties): QTreeWidget(), properties(properties) {
|
|
|
|
// Graphical settings
|
|
|
|
setRootIsDecorated(false);
|
|
|
|
setAllColumnsShowFocus(true);
|
|
|
|
setItemsExpandable(false);
|
|
|
|
setSelectionMode(QAbstractItemView::ExtendedSelection);
|
|
|
|
// Context menu
|
|
|
|
setContextMenuPolicy(Qt::CustomContextMenu);
|
|
|
|
connect(this, SIGNAL(customContextMenuRequested(QPoint)), this, SLOT(showTrackerListMenu(QPoint)));
|
|
|
|
// Set header
|
|
|
|
QStringList header;
|
|
|
|
header << tr("URL");
|
|
|
|
header << tr("Status");
|
|
|
|
header << tr("Peers");
|
|
|
|
header << tr("Message");
|
|
|
|
setHeaderItem(new QTreeWidgetItem(header));
|
|
|
|
dht_item = new QTreeWidgetItem(QStringList("** "+tr("[DHT]")+" **"));
|
|
|
|
insertTopLevelItem(0, dht_item);
|
|
|
|
setRowColor(0, QColor("grey"));
|
|
|
|
pex_item = new QTreeWidgetItem(QStringList("** "+tr("[PeX]")+" **"));
|
|
|
|
insertTopLevelItem(1, pex_item);
|
|
|
|
setRowColor(1, QColor("grey"));
|
|
|
|
lsd_item = new QTreeWidgetItem(QStringList("** "+tr("[LSD]")+" **"));
|
|
|
|
insertTopLevelItem(2, lsd_item);
|
|
|
|
setRowColor(2, QColor("grey"));
|
|
|
|
loadSettings();
|
|
|
|
}
|
|
|
|
|
|
|
|
TrackerList::~TrackerList() {
|
|
|
|
saveSettings();
|
|
|
|
}
|
|
|
|
|
|
|
|
QList<QTreeWidgetItem*> TrackerList::getSelectedTrackerItems() const {
|
|
|
|
QList<QTreeWidgetItem*> selected_items = selectedItems();
|
|
|
|
QList<QTreeWidgetItem*> selected_trackers;
|
|
|
|
foreach(QTreeWidgetItem *item, selectedItems()) {
|
|
|
|
if(indexOfTopLevelItem(item) >= NB_STICKY_ITEM) { // Ignore STICKY ITEMS
|
|
|
|
selected_trackers << item;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return selected_trackers;
|
|
|
|
}
|
|
|
|
|
|
|
|
void TrackerList::setRowColor(int row, QColor color) {
|
|
|
|
unsigned int nbColumns = columnCount();
|
|
|
|
QTreeWidgetItem *item = topLevelItem(row);
|
|
|
|
for(unsigned int i=0; i<nbColumns; ++i) {
|
|
|
|
item->setData(i, Qt::ForegroundRole, color);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void TrackerList::moveSelectionUp() {
|
|
|
|
#if LIBTORRENT_VERSION_MINOR < 15
|
|
|
|
QTorrentHandle h = properties->getCurrentTorrent();
|
|
|
|
if(!h.is_valid()) {
|
|
|
|
clear();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
QList<QTreeWidgetItem *> selected_items = getSelectedTrackerItems();
|
|
|
|
if(selected_items.isEmpty()) return;
|
|
|
|
bool change = false;
|
|
|
|
foreach(QTreeWidgetItem *item, selected_items){
|
|
|
|
int index = indexOfTopLevelItem(item);
|
|
|
|
if(index > NB_STICKY_ITEM) {
|
|
|
|
insertTopLevelItem(index-1, takeTopLevelItem(index));
|
|
|
|
change = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if(!change) return;
|
|
|
|
// Restore selection
|
|
|
|
QItemSelectionModel *selection = selectionModel();
|
|
|
|
foreach(QTreeWidgetItem *item, selected_items) {
|
|
|
|
selection->select(indexFromItem(item), QItemSelectionModel::Rows|QItemSelectionModel::Select);
|
|
|
|
}
|
|
|
|
setSelectionModel(selection);
|
|
|
|
// Update torrent trackers
|
|
|
|
std::vector<announce_entry> trackers;
|
|
|
|
for(int i=NB_STICKY_ITEM; i<topLevelItemCount(); ++i) {
|
|
|
|
QString tracker_url = topLevelItem(i)->data(COL_URL, Qt::DisplayRole).toString();
|
|
|
|
announce_entry e(tracker_url.toStdString());
|
|
|
|
e.tier = i-NB_STICKY_ITEM;
|
|
|
|
trackers.push_back(e);
|
|
|
|
}
|
|
|
|
h.replace_trackers(trackers);
|
|
|
|
// Reannounce
|
|
|
|
h.force_reannounce();
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
void TrackerList::moveSelectionDown() {
|
|
|
|
#if LIBTORRENT_VERSION_MINOR < 15
|
|
|
|
QTorrentHandle h = properties->getCurrentTorrent();
|
|
|
|
if(!h.is_valid()) {
|
|
|
|
clear();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
QList<QTreeWidgetItem *> selected_items = getSelectedTrackerItems();
|
|
|
|
if(selected_items.isEmpty()) return;
|
|
|
|
bool change = false;
|
|
|
|
for(int i=selectedItems().size()-1; i>= 0; --i) {
|
|
|
|
int index = indexOfTopLevelItem(selected_items.at(i));
|
|
|
|
if(index < topLevelItemCount()-1) {
|
|
|
|
insertTopLevelItem(index+1, takeTopLevelItem(index));
|
|
|
|
change = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if(!change) return;
|
|
|
|
// Restore selection
|
|
|
|
QItemSelectionModel *selection = selectionModel();
|
|
|
|
foreach(QTreeWidgetItem *item, selected_items) {
|
|
|
|
selection->select(indexFromItem(item), QItemSelectionModel::Rows|QItemSelectionModel::Select);
|
|
|
|
}
|
|
|
|
setSelectionModel(selection);
|
|
|
|
// Update torrent trackers
|
|
|
|
std::vector<announce_entry> trackers;
|
|
|
|
for(int i=NB_STICKY_ITEM; i<topLevelItemCount(); ++i) {
|
|
|
|
QString tracker_url = topLevelItem(i)->data(COL_URL, Qt::DisplayRole).toString();
|
|
|
|
announce_entry e(tracker_url.toStdString());
|
|
|
|
e.tier = i-NB_STICKY_ITEM;
|
|
|
|
trackers.push_back(e);
|
|
|
|
}
|
|
|
|
h.replace_trackers(trackers);
|
|
|
|
// Reannounce
|
|
|
|
h.force_reannounce();
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
void TrackerList::clear() {
|
|
|
|
qDeleteAll(tracker_items.values());
|
|
|
|
tracker_items.clear();
|
|
|
|
dht_item->setText(COL_PEERS, "");
|
|
|
|
dht_item->setText(COL_STATUS, "");
|
|
|
|
dht_item->setText(COL_MSG, "");
|
|
|
|
pex_item->setText(COL_PEERS, "");
|
|
|
|
pex_item->setText(COL_STATUS, "");
|
|
|
|
pex_item->setText(COL_MSG, "");
|
|
|
|
lsd_item->setText(COL_PEERS, "");
|
|
|
|
lsd_item->setText(COL_STATUS, "");
|
|
|
|
lsd_item->setText(COL_MSG, "");
|
|
|
|
}
|
|
|
|
|
|
|
|
void TrackerList::loadStickyItems(const QTorrentHandle &h) {
|
|
|
|
// XXX: libtorrent should provide this info...
|
|
|
|
// Count peers from DHT, LSD, PeX
|
|
|
|
uint nb_dht=0, nb_lsd=0, nb_pex=0;
|
|
|
|
std::vector<peer_info> peers;
|
|
|
|
h.get_peer_info(peers);
|
|
|
|
std::vector<peer_info>::iterator it;
|
|
|
|
for(it=peers.begin(); it!=peers.end(); it++) {
|
|
|
|
if(it->source & peer_info::dht)
|
|
|
|
++nb_dht;
|
|
|
|
if(it->source & peer_info::lsd)
|
|
|
|
++nb_lsd;
|
|
|
|
if(it->source & peer_info::pex)
|
|
|
|
++nb_pex;
|
|
|
|
}
|
|
|
|
// load DHT information
|
|
|
|
if(properties->getBTSession()->isDHTEnabled() && h.has_metadata() && !h.priv()) {
|
|
|
|
dht_item->setText(COL_STATUS, tr("Working"));
|
|
|
|
} else {
|
|
|
|
dht_item->setText(COL_STATUS, tr("Disabled"));
|
|
|
|
}
|
|
|
|
dht_item->setText(COL_PEERS, QString::number(nb_dht));
|
|
|
|
if(h.has_metadata() && h.priv()) {
|
|
|
|
dht_item->setText(COL_MSG, tr("This torrent is private"));
|
|
|
|
}
|
|
|
|
// Load PeX Information
|
|
|
|
pex_item->setText(COL_STATUS, tr("Working"));
|
|
|
|
pex_item->setText(COL_PEERS, QString::number(nb_pex));
|
|
|
|
// Load LSD Information
|
|
|
|
if(properties->getBTSession()->isLSDEnabled())
|
|
|
|
lsd_item->setText(COL_STATUS, tr("Working"));
|
|
|
|
else
|
|
|
|
lsd_item->setText(COL_STATUS, tr("Disabled"));
|
|
|
|
lsd_item->setText(COL_PEERS, QString::number(nb_lsd));
|
|
|
|
}
|
|
|
|
|
|
|
|
void TrackerList::loadTrackers() {
|
|
|
|
// Load trackers from torrent handle
|
|
|
|
QTorrentHandle h = properties->getCurrentTorrent();
|
|
|
|
if(!h.is_valid()) return;
|
|
|
|
loadStickyItems(h);
|
|
|
|
// Load actual trackers information
|
|
|
|
QHash<QString, TrackerInfos> trackers_data = properties->getBTSession()->getTrackersInfo(h.hash());
|
|
|
|
QStringList old_trackers_urls = tracker_items.keys();
|
|
|
|
std::vector<announce_entry> trackers = h.trackers();
|
|
|
|
std::vector<announce_entry>::iterator it;
|
|
|
|
for(it = trackers.begin(); it != trackers.end(); it++) {
|
|
|
|
QStringList item_list;
|
|
|
|
QString tracker_url = misc::toQString(it->url);
|
|
|
|
QTreeWidgetItem *item = tracker_items.value(tracker_url, 0);
|
|
|
|
if(!item) {
|
|
|
|
item = new QTreeWidgetItem();
|
|
|
|
item->setText(COL_URL, tracker_url);
|
|
|
|
addTopLevelItem(item);
|
|
|
|
tracker_items[tracker_url] = item;
|
|
|
|
} else {
|
|
|
|
old_trackers_urls.removeOne(tracker_url);
|
|
|
|
}
|
|
|
|
TrackerInfos data = trackers_data.value(tracker_url, TrackerInfos(tracker_url));
|
|
|
|
QString error_message = data.last_message.trimmed();
|
|
|
|
#if LIBTORRENT_VERSION_MINOR > 14
|
|
|
|
if(it->verified) {
|
|
|
|
item->setText(COL_STATUS, tr("Working"));
|
|
|
|
item->setText(COL_MSG, "");
|
|
|
|
} else {
|
|
|
|
if(it->updating && it->fails == 0) {
|
|
|
|
item->setText(COL_STATUS, tr("Updating..."));
|
|
|
|
item->setText(COL_MSG, "");
|
|
|
|
} else {
|
|
|
|
if(it->fails > 0) {
|
|
|
|
item->setText(COL_STATUS, tr("Not working"));
|
|
|
|
item->setText(COL_MSG, error_message);
|
|
|
|
} else {
|
|
|
|
item->setText(COL_STATUS, tr("Not contacted yet"));
|
|
|
|
item->setText(COL_MSG, "");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
if(data.verified) {
|
|
|
|
item->setText(COL_STATUS, tr("Working"));
|
|
|
|
item->setText(COL_MSG, "");
|
|
|
|
} else {
|
|
|
|
if(data.fail_count > 0) {
|
|
|
|
item->setText(COL_STATUS, tr("Not working"));
|
|
|
|
item->setText(COL_MSG, error_message);
|
|
|
|
} else {
|
|
|
|
item->setText(COL_STATUS, tr("Not contacted yet"));
|
|
|
|
item->setText(COL_MSG, "");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
item->setText(COL_PEERS, QString::number(trackers_data.value(tracker_url, TrackerInfos(tracker_url)).num_peers));
|
|
|
|
}
|
|
|
|
// Remove old trackers
|
|
|
|
foreach(const QString &tracker, old_trackers_urls) {
|
|
|
|
delete tracker_items.take(tracker);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Ask the user for new trackers and add them to the torrent
|
|
|
|
void TrackerList::askForTrackers(){
|
|
|
|
QTorrentHandle h = properties->getCurrentTorrent();
|
|
|
|
if(!h.is_valid()) return;
|
|
|
|
QStringList trackers = TrackersAdditionDlg::askForTrackers(h);
|
|
|
|
if(!trackers.empty()) {
|
|
|
|
foreach(const QString& tracker, trackers) {
|
|
|
|
announce_entry url(tracker.toStdString());
|
|
|
|
url.tier = 0;
|
|
|
|
h.add_tracker(url);
|
|
|
|
}
|
|
|
|
// Reannounce to new trackers
|
|
|
|
h.force_reannounce();
|
|
|
|
// Reload tracker list
|
|
|
|
loadTrackers();
|
|
|
|
// XXX: I don't think this is necessary now
|
|
|
|
//BTSession->saveTrackerFile(h.hash());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void TrackerList::deleteSelectedTrackers(){
|
|
|
|
QTorrentHandle h = properties->getCurrentTorrent();
|
|
|
|
if(!h.is_valid()) {
|
|
|
|
clear();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
QList<QTreeWidgetItem *> selected_items = getSelectedTrackerItems();
|
|
|
|
if(selected_items.isEmpty()) return;
|
|
|
|
QStringList urls_to_remove;
|
|
|
|
foreach(QTreeWidgetItem *item, selected_items){
|
|
|
|
QString tracker_url = item->data(COL_URL, Qt::DisplayRole).toString();
|
|
|
|
urls_to_remove << tracker_url;
|
|
|
|
tracker_items.remove(tracker_url);
|
|
|
|
delete item;
|
|
|
|
}
|
|
|
|
// Iterate of trackers and remove selected ones
|
|
|
|
std::vector<announce_entry> trackers = h.trackers();
|
|
|
|
std::vector<announce_entry>::iterator it = trackers.begin();
|
|
|
|
while(it != trackers.end()) {
|
|
|
|
int index = urls_to_remove.indexOf(misc::toQString((*it).url));
|
|
|
|
if(index >= 0) {
|
|
|
|
trackers.erase(it);
|
|
|
|
urls_to_remove.removeAt(index);
|
|
|
|
} else {
|
|
|
|
it++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
h.replace_trackers(trackers);
|
|
|
|
h.force_reannounce();
|
|
|
|
// Reload Trackers
|
|
|
|
loadTrackers();
|
|
|
|
//XXX: I don't think this is necessary
|
|
|
|
//BTSession->saveTrackerFile(h.hash());
|
|
|
|
}
|
|
|
|
|
|
|
|
void TrackerList::showTrackerListMenu(QPoint) {
|
|
|
|
QTorrentHandle h = properties->getCurrentTorrent();
|
|
|
|
if(!h.is_valid() || !h.has_metadata()) return;
|
|
|
|
QList<QTreeWidgetItem*> selected_items = getSelectedTrackerItems();
|
|
|
|
QMenu menu;
|
|
|
|
// Add actions
|
|
|
|
QAction *addAct = menu.addAction(QIcon(":/Icons/oxygen/list-add.png"), tr("Add a new tracker..."));
|
|
|
|
QAction *delAct = 0;
|
|
|
|
if(!getSelectedTrackerItems().isEmpty()) {
|
|
|
|
delAct = menu.addAction(QIcon(":/Icons/oxygen/list-remove.png"), "Remove tracker");
|
|
|
|
}
|
|
|
|
QAction *act = menu.exec(QCursor::pos());
|
|
|
|
if(act == 0) return;
|
|
|
|
if(act == addAct) {
|
|
|
|
askForTrackers();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if(act == delAct) {
|
|
|
|
deleteSelectedTrackers();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void TrackerList::loadSettings() {
|
|
|
|
QSettings settings(QString::fromUtf8("qBittorrent"), QString::fromUtf8("qBittorrent"));
|
|
|
|
QList<int> contentColsWidths = misc::intListfromStringList(settings.value(QString::fromUtf8("TorrentProperties/Trackers/trackersColsWidth")).toStringList());
|
|
|
|
if(!contentColsWidths.empty()) {
|
|
|
|
for(int i=0; i<contentColsWidths.size(); ++i) {
|
|
|
|
setColumnWidth(i, contentColsWidths.at(i));
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
setColumnWidth(0, 300);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void TrackerList::saveSettings() const {
|
|
|
|
QSettings settings(QString::fromUtf8("qBittorrent"), QString::fromUtf8("qBittorrent"));
|
|
|
|
QStringList contentColsWidths;
|
|
|
|
for(int i=0; i<columnCount(); ++i) {
|
|
|
|
contentColsWidths << QString::number(columnWidth(i));
|
|
|
|
}
|
|
|
|
settings.setValue(QString::fromUtf8("TorrentProperties/Trackers/trackersColsWidth"), contentColsWidths);
|
|
|
|
}
|