/* * 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 "trackerlist.h" #include "propertieswidget.h" #include "trackersadditiondlg.h" #include "iconprovider.h" #include "qbtsession.h" #include "qinisettings.h" #include "misc.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() << "" << "** "+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 { 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 h.force_reannounce(); } 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 h.force_reannounce(); } 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) { // 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")); } 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")); // Load LSD Information if (QBtSession::instance()->isLSDEnabled()) lsd_item->setText(COL_STATUS, tr("Working")); else lsd_item->setText(COL_STATUS, tr("Disabled")); // 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_TIER, QString::number(it->tier)); 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 (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; QStringList trackers = TrackersAdditionDlg::askForTrackers(h); if (!trackers.empty()) { foreach (const QString& tracker, trackers) { if (tracker.trimmed().isEmpty()) continue; announce_entry url(tracker.toStdString()); url.tier = 0; h.add_tracker(url); } // Reannounce to new trackers h.force_reannounce(); // Reload tracker list loadTrackers(); } } // Copy tracker URL to clipboard void TrackerList::copyTrackerUrl() { qDebug()<<"Copy tracker URL to clipboard"; QTorrentHandle h = properties->getCurrentTorrent(); if (!h.is_valid()) { clear(); return; } QList selected_items = getSelectedTrackerItems(); if (selected_items.isEmpty()) return; QString urls_to_copy; foreach (QTreeWidgetItem *item, selected_items) { QString tracker_url = item->data(COL_URL, Qt::DisplayRole).toString(); qDebug()<setText(urls_to_copy); qDebug()<<"Tracker URL copied clipboard!"; } 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 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); h.force_reannounce(); // Reload Trackers loadTrackers(); } 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 = menu.addAction(IconProvider::instance()->getIcon("list-add"), tr("Copy tracker url...")); QAction *delAct = 0; if (!getSelectedTrackerItems().isEmpty()) { delAct = menu.addAction(IconProvider::instance()->getIcon("list-remove"), tr("Remove tracker")); } menu.addSeparator(); QAction *reannounceAct = menu.addAction(IconProvider::instance()->getIcon("view-refresh"), tr("Force reannounce")); 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 (act == reannounceAct) { properties->getCurrentTorrent().force_reannounce(); return; } } void TrackerList::loadSettings() { QIniSettings settings(QString::fromUtf8("qBittorrent"), QString::fromUtf8("qBittorrent")); if (!header()->restoreState(settings.value("TorrentProperties/Trackers/TrackerListState").toByteArray())) { setColumnWidth(0, 30); setColumnWidth(1, 300); } } void TrackerList::saveSettings() const { QIniSettings settings(QString::fromUtf8("qBittorrent"), QString::fromUtf8("qBittorrent")); settings.setValue("TorrentProperties/Trackers/TrackerListState", header()->saveState()); }