/* * 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 #include #include #include #include #include #include #include "trackerlist.h" #include "propertieswidget.h" #include "trackersadditiondlg.h" #include "misc.h" #include "qbtsession.h" #include "qinisettings.h" using namespace libtorrent; 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 TrackerList::getSelectedTrackerItems() const { QList selected_items = selectedItems(); QList 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; isetData(i, Qt::ForegroundRole, color); } } void TrackerList::moveSelectionUp() { #if LIBTORRENT_VERSION_MINOR < 15 QTorrentHandle h = properties->getCurrentTorrent(); if(!h.is_valid()) { clear(); return; } QList 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 trackers; for(int i=NB_STICKY_ITEM; idata(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 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 trackers; for(int i=NB_STICKY_ITEM; idata(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 peers; h.get_peer_info(peers); std::vector::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(QBtSession::instance()->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 if(QBtSession::instance()->isPexEnabled()) pex_item->setText(COL_STATUS, tr("Working")); else pex_item->setText(COL_STATUS, tr("Disabled")); pex_item->setText(COL_PEERS, QString::number(nb_pex)); // Load LSD Information if(QBtSession::instance()->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 trackers_data = QBtSession::instance()->getTrackersInfo(h.hash()); QStringList old_trackers_urls = tracker_items.keys(); const std::vector trackers = h.trackers(); for(std::vector::const_iterator it = trackers.begin(); it != trackers.end(); it++) { 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(); } } void TrackerList::deleteSelectedTrackers(){ QTorrentHandle h = properties->getCurrentTorrent(); if(!h.is_valid()) { clear(); return; } QList 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 trackers = h.trackers(); std::vector::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(); } void TrackerList::showTrackerListMenu(QPoint) { QTorrentHandle h = properties->getCurrentTorrent(); if(!h.is_valid() || !h.has_metadata()) return; //QList 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"), tr("Remove tracker")); } menu.addSeparator(); QAction *reannounceAct = menu.addAction(QIcon(":/Icons/oxygen/run-build.png"), tr("Force reannounce")); QAction *act = menu.exec(QCursor::pos()); if(act == 0) return; if(act == addAct) { askForTrackers(); return; } if(act == delAct) { deleteSelectedTrackers(); return; } if(act == reannounceAct) { properties->getCurrentTorrent().force_reannounce(); return; } } void TrackerList::loadSettings() { QIniSettings settings(QString::fromUtf8("qBittorrent"), QString::fromUtf8("qBittorrent")); QList contentColsWidths = misc::intListfromStringList(settings.value(QString::fromUtf8("TorrentProperties/Trackers/trackersColsWidth")).toStringList()); if(!contentColsWidths.empty()) { for(int i=0; i