/* * 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 #include #include #include "trackerlist.h" #include "propertieswidget.h" #include "trackersadditiondlg.h" #include "iconprovider.h" #include "qbtsession.h" #include "core/preferences.h" #include "core/misc.h" #include "autoexpandabledialog.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 << "#"; header << tr("URL"); header << tr("Status"); header << tr("Peers"); header << tr("Message"); setHeaderItem(new QTreeWidgetItem(header)); dht_item = new QTreeWidgetItem(QStringList() << "" << "** [DHT] **"); insertTopLevelItem(0, dht_item); setRowColor(0, QColor("grey")); pex_item = new QTreeWidgetItem(QStringList() << "" << "** [PeX] **"); insertTopLevelItem(1, pex_item); setRowColor(1, QColor("grey")); lsd_item = new QTreeWidgetItem(QStringList() << "" << "** [LSD] **"); insertTopLevelItem(2, lsd_item); setRowColor(2, QColor("grey")); editHotkey = new QShortcut(QKeySequence("F2"), this, SLOT(editSelectedTracker()), 0, Qt::WidgetShortcut); connect(this, SIGNAL(doubleClicked(QModelIndex)), SLOT(editSelectedTracker())); deleteHotkey = new QShortcut(QKeySequence(QKeySequence::Delete), this, SLOT(deleteSelectedTrackers()), 0, Qt::WidgetShortcut); loadSettings(); } TrackerList::~TrackerList() { delete editHotkey; delete deleteHotkey; saveSettings(); } QList TrackerList::getSelectedTrackerItems() const { const QList selected_items = selectedItems(); QList selected_trackers; foreach (QTreeWidgetItem *item, selected_items) { 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() { 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 if (!h.is_paused()) h.force_reannounce(); loadTrackers(); } void TrackerList::moveSelectionDown() { 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 if (!h.is_paused()) h.force_reannounce(); loadTrackers(); } 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) { QString working = tr("Working"); QString disabled = tr("Disabled"); // load DHT information if (QBtSession::instance()->isDHTEnabled() && !h.priv()) dht_item->setText(COL_STATUS, working); else dht_item->setText(COL_STATUS, disabled); // Load PeX Information if (QBtSession::instance()->isPexEnabled() && !h.priv()) pex_item->setText(COL_STATUS, working); else pex_item->setText(COL_STATUS, disabled); // Load LSD Information if (QBtSession::instance()->isLSDEnabled() && !h.priv()) lsd_item->setText(COL_STATUS, working); else lsd_item->setText(COL_STATUS, disabled); if (h.priv()) { QString privateMsg = tr("This torrent is private"); dht_item->setText(COL_MSG, privateMsg); pex_item->setText(COL_MSG, privateMsg); lsd_item->setText(COL_MSG, privateMsg); } // 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 = peers.begin(); std::vector::iterator end = peers.end(); for ( ; it != 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; } dht_item->setText(COL_PEERS, QString::number(nb_dht)); pex_item->setText(COL_PEERS, QString::number(nb_pex)); 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(); std::vector::const_iterator it = trackers.begin(); std::vector::const_iterator end = trackers.end(); for ( ; it != 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); } item->setText(COL_TIER, QString::number(it->tier)); TrackerInfos data = trackers_data.value(tracker_url, TrackerInfos(tracker_url)); QString error_message = data.last_message.trimmed(); 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, ""); } } } 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; QString hash = h.hash(); QStringList trackers = TrackersAdditionDlg::askForTrackers(h); QBtSession::instance()->addTrackersAndUrlSeeds(hash, trackers, QStringList()); loadTrackers(); } void TrackerList::copyTrackerUrl() { QList selected_items = getSelectedTrackerItems(); if (selected_items.isEmpty()) return; QStringList urls_to_copy; foreach (QTreeWidgetItem *item, selected_items) { QString tracker_url = item->data(COL_URL, Qt::DisplayRole).toString(); qDebug() << QString("Copy: ") + tracker_url; urls_to_copy << tracker_url; } QApplication::clipboard()->setText(urls_to_copy.join("\n")); } void TrackerList::deleteSelectedTrackers() { QTorrentHandle h = properties->getCurrentTorrent(); if (!h.is_valid()) { clear(); return; } QString hash = h.hash(); 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 remaining_trackers; std::vector trackers = h.trackers(); std::vector::iterator it = trackers.begin(); std::vector::iterator itend = trackers.end(); for ( ; it != itend; ++it) { if (!urls_to_remove.contains(misc::toQString((*it).url))) { remaining_trackers.push_back(*it); } } h.replace_trackers(remaining_trackers); if (!urls_to_remove.empty()) emit trackersRemoved(urls_to_remove, hash); if (remaining_trackers.empty()) emit trackerlessChange(true, hash); if (!h.is_paused()) h.force_reannounce(); // Reload Trackers loadTrackers(); } void TrackerList::editSelectedTracker() { try { QTorrentHandle h = properties->getCurrentTorrent(); QString hash = h.hash(); QList selected_items = getSelectedTrackerItems(); if (selected_items.isEmpty()) return; // During multi-select only process item selected last QUrl tracker_url = selected_items.last()->text(COL_URL); bool ok; QUrl new_tracker_url = AutoExpandableDialog::getText(this, tr("Tracker editing"), tr("Tracker URL:"), QLineEdit::Normal, tracker_url.toString(), &ok).trimmed(); if (!ok) return; if (!new_tracker_url.isValid()) { QMessageBox::warning(this, tr("Tracker editing failed"), tr("The tracker URL entered is invalid.")); return; } if (new_tracker_url == tracker_url) return; std::vector trackers = h.trackers(); std::vector::iterator it = trackers.begin(); std::vector::iterator itend = trackers.end(); bool match = false; for ( ; it != itend; ++it) { if (new_tracker_url == QUrl(misc::toQString(it->url))) { QMessageBox::warning(this, tr("Tracker editing failed"), tr("The tracker URL already exists.")); return; } if (tracker_url == QUrl(misc::toQStringU(it->url)) && !match) { announce_entry new_entry(new_tracker_url.toString().toUtf8().constData()); new_entry.tier = it->tier; match = true; *it = new_entry; emit trackersRemoved(QStringList(tracker_url.toString()), hash); emit trackersAdded(QStringList(new_tracker_url.toString()), hash); } } h.replace_trackers(trackers); if (!h.is_paused()) { h.force_reannounce(); h.force_dht_announce(); } } catch(invalid_handle&) { return; } loadTrackers(); } #if LIBTORRENT_VERSION_NUM >= 10000 void TrackerList::reannounceSelected() { try { QTorrentHandle h = properties->getCurrentTorrent(); QList selected_items = getSelectedTrackerItems(); if (selected_items.isEmpty()) return; std::vector trackers = h.trackers(); for (size_t i = 0; i < trackers.size(); ++i) { foreach (QTreeWidgetItem* w, selected_items) { if (w->text(COL_URL) == misc::toQString(trackers[i].url)) { h.force_reannounce(0, i); break; } } } } catch(invalid_handle&) { return; } loadTrackers(); } #endif void TrackerList::showTrackerListMenu(QPoint) { QTorrentHandle h = properties->getCurrentTorrent(); if (!h.is_valid()) return; //QList selected_items = getSelectedTrackerItems(); QMenu menu; // Add actions QAction *addAct = menu.addAction(IconProvider::instance()->getIcon("list-add"), tr("Add a new tracker...")); QAction *copyAct = 0; QAction *delAct = 0; QAction *editAct = 0; if (!getSelectedTrackerItems().isEmpty()) { delAct = menu.addAction(IconProvider::instance()->getIcon("list-remove"), tr("Remove tracker")); copyAct = menu.addAction(IconProvider::instance()->getIcon("edit-copy"), tr("Copy tracker url")); editAct = menu.addAction(IconProvider::instance()->getIcon("edit-rename"),tr("Edit selected tracker URL")); } #if LIBTORRENT_VERSION_NUM >= 10000 QAction *reannounceSelAct = NULL; #endif QAction *reannounceAct = NULL; if (!h.is_paused()) { #if LIBTORRENT_VERSION_NUM >= 10000 reannounceSelAct = menu.addAction(IconProvider::instance()->getIcon("view-refresh"), tr("Force reannounce to selected trackers")); #endif menu.addSeparator(); reannounceAct = menu.addAction(IconProvider::instance()->getIcon("view-refresh"), tr("Force reannounce to all trackers")); } QAction *act = menu.exec(QCursor::pos()); if (act == 0) return; if (act == addAct) { askForTrackers(); return; } if (act == copyAct) { copyTrackerUrl(); return; } if (act == delAct) { deleteSelectedTrackers(); return; } #if LIBTORRENT_VERSION_NUM >= 10000 if (act == reannounceSelAct) { reannounceSelected(); return; } #endif if (act == reannounceAct) { properties->getCurrentTorrent().force_reannounce(); return; } if (act == editAct) { editSelectedTracker(); return; } } void TrackerList::loadSettings() { if (!header()->restoreState(Preferences::instance()->getPropTrackerListState())) { setColumnWidth(0, 30); setColumnWidth(1, 300); } } void TrackerList::saveSettings() const { Preferences::instance()->setPropTrackerListState(header()->saveState()); }