mirror of
https://github.com/d47081/qBittorrent.git
synced 2025-03-13 05:41:17 +00:00
Merge pull request #2120 from sorokin/ip-filter-moc
split filterparserthread into .h and .cpp and (probably) fixes #2119
This commit is contained in:
commit
c9e13dfe8b
@ -89,6 +89,8 @@ void qt_mac_set_dock_menu(QMenu *menu);
|
|||||||
#include "downloadthread.h"
|
#include "downloadthread.h"
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
#include <libtorrent/session.hpp>
|
||||||
|
|
||||||
using namespace libtorrent;
|
using namespace libtorrent;
|
||||||
|
|
||||||
#define TIME_TRAY_BALLOON 5000
|
#define TIME_TRAY_BALLOON 5000
|
||||||
|
394
src/qtlibtorrent/filterparserthread.cpp
Normal file
394
src/qtlibtorrent/filterparserthread.cpp
Normal file
@ -0,0 +1,394 @@
|
|||||||
|
/*
|
||||||
|
* 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 "filterparserthread.h"
|
||||||
|
|
||||||
|
#include <QFile>
|
||||||
|
#include <QHostAddress>
|
||||||
|
|
||||||
|
#include <libtorrent/session.hpp>
|
||||||
|
#include <libtorrent/ip_filter.hpp>
|
||||||
|
|
||||||
|
FilterParserThread::FilterParserThread(QObject* parent, libtorrent::session *s) : QThread(parent), s(s), abort(false) {
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
FilterParserThread::~FilterParserThread() {
|
||||||
|
abort = true;
|
||||||
|
wait();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Parser for eMule ip filter in DAT format
|
||||||
|
int FilterParserThread::parseDATFilterFile(QString filePath, libtorrent::ip_filter& filter) {
|
||||||
|
int ruleCount = 0;
|
||||||
|
QFile file(filePath);
|
||||||
|
if (file.exists()) {
|
||||||
|
if (!file.open(QIODevice::ReadOnly | QIODevice::Text)) {
|
||||||
|
std::cerr << "I/O Error: Could not open ip filer file in read mode." << std::endl;
|
||||||
|
return ruleCount;
|
||||||
|
}
|
||||||
|
unsigned int nbLine = 0;
|
||||||
|
while (!file.atEnd() && !abort) {
|
||||||
|
++nbLine;
|
||||||
|
QByteArray line = file.readLine();
|
||||||
|
// Ignoring empty lines
|
||||||
|
line = line.trimmed();
|
||||||
|
if (line.isEmpty()) continue;
|
||||||
|
// Ignoring commented lines
|
||||||
|
if (line.startsWith('#') || line.startsWith("//")) continue;
|
||||||
|
|
||||||
|
// Line should be splitted by commas
|
||||||
|
QList<QByteArray> partsList = line.split(',');
|
||||||
|
const uint nbElem = partsList.size();
|
||||||
|
|
||||||
|
// IP Range should be splitted by a dash
|
||||||
|
QList<QByteArray> IPs = partsList.first().split('-');
|
||||||
|
if (IPs.size() != 2) {
|
||||||
|
qDebug("Ipfilter.dat: line %d is malformed.", nbLine);
|
||||||
|
qDebug("Line was %s", line.constData());
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
boost::system::error_code ec;
|
||||||
|
const QString strStartIP = cleanupIPAddress(IPs.at(0));
|
||||||
|
if (strStartIP.isEmpty()) {
|
||||||
|
qDebug("Ipfilter.dat: line %d is malformed.", nbLine);
|
||||||
|
qDebug("Start IP of the range is malformated: %s", qPrintable(strStartIP));
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
libtorrent::address startAddr = libtorrent::address::from_string(qPrintable(strStartIP), ec);
|
||||||
|
if (ec) {
|
||||||
|
qDebug("Ipfilter.dat: line %d is malformed.", nbLine);
|
||||||
|
qDebug("Start IP of the range is malformated: %s", qPrintable(strStartIP));
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
const QString strEndIP = cleanupIPAddress(IPs.at(1));
|
||||||
|
if (strEndIP.isEmpty()) {
|
||||||
|
qDebug("Ipfilter.dat: line %d is malformed.", nbLine);
|
||||||
|
qDebug("End IP of the range is malformated: %s", qPrintable(strEndIP));
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
libtorrent::address endAddr = libtorrent::address::from_string(qPrintable(strEndIP), ec);
|
||||||
|
if (ec) {
|
||||||
|
qDebug("Ipfilter.dat: line %d is malformed.", nbLine);
|
||||||
|
qDebug("End IP of the range is malformated: %s", qPrintable(strEndIP));
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if (startAddr.is_v4() != endAddr.is_v4()) {
|
||||||
|
qDebug("Ipfilter.dat: line %d is malformed.", nbLine);
|
||||||
|
qDebug("One IP is IPv4 and the other is IPv6!");
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check if there is an access value (apparently not mandatory)
|
||||||
|
int nbAccess = 0;
|
||||||
|
if (nbElem > 1) {
|
||||||
|
// There is possibly one
|
||||||
|
nbAccess = partsList.at(1).trimmed().toInt();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (nbAccess > 127) {
|
||||||
|
// Ignoring this rule because access value is too high
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
// Now Add to the filter
|
||||||
|
try {
|
||||||
|
filter.add_rule(startAddr, endAddr, libtorrent::ip_filter::blocked);
|
||||||
|
++ruleCount;
|
||||||
|
}catch(exception) {
|
||||||
|
qDebug("Bad line in filter file, avoided crash...");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
file.close();
|
||||||
|
}
|
||||||
|
return ruleCount;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Parser for PeerGuardian ip filter in p2p format
|
||||||
|
int FilterParserThread::parseP2PFilterFile(QString filePath, libtorrent::ip_filter& filter) {
|
||||||
|
int ruleCount = 0;
|
||||||
|
QFile file(filePath);
|
||||||
|
if (file.exists()) {
|
||||||
|
if (!file.open(QIODevice::ReadOnly | QIODevice::Text)) {
|
||||||
|
std::cerr << "I/O Error: Could not open ip filer file in read mode." << std::endl;
|
||||||
|
return ruleCount;
|
||||||
|
}
|
||||||
|
unsigned int nbLine = 0;
|
||||||
|
while (!file.atEnd() && !abort) {
|
||||||
|
++nbLine;
|
||||||
|
QByteArray line = file.readLine().trimmed();
|
||||||
|
if (line.isEmpty()) continue;
|
||||||
|
// Ignoring commented lines
|
||||||
|
if (line.startsWith('#') || line.startsWith("//")) continue;
|
||||||
|
// Line is splitted by :
|
||||||
|
QList<QByteArray> partsList = line.split(':');
|
||||||
|
if (partsList.size() < 2) {
|
||||||
|
qDebug("p2p file: line %d is malformed.", nbLine);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
// Get IP range
|
||||||
|
QList<QByteArray> IPs = partsList.last().split('-');
|
||||||
|
if (IPs.size() != 2) {
|
||||||
|
qDebug("p2p file: line %d is malformed.", nbLine);
|
||||||
|
qDebug("line was: %s", line.constData());
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
boost::system::error_code ec;
|
||||||
|
QString strStartIP = cleanupIPAddress(IPs.at(0));
|
||||||
|
if (strStartIP.isEmpty()) {
|
||||||
|
qDebug("p2p file: line %d is malformed.", nbLine);
|
||||||
|
qDebug("Start IP is invalid: %s", qPrintable(strStartIP));
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
libtorrent::address startAddr = libtorrent::address::from_string(qPrintable(strStartIP), ec);
|
||||||
|
if (ec) {
|
||||||
|
qDebug("p2p file: line %d is malformed.", nbLine);
|
||||||
|
qDebug("Start IP is invalid: %s", qPrintable(strStartIP));
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
QString strEndIP = cleanupIPAddress(IPs.at(1));
|
||||||
|
if (strEndIP.isEmpty()) {
|
||||||
|
qDebug("p2p file: line %d is malformed.", nbLine);
|
||||||
|
qDebug("End IP is invalid: %s", qPrintable(strStartIP));
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
libtorrent::address endAddr = libtorrent::address::from_string(qPrintable(strEndIP), ec);
|
||||||
|
if (ec) {
|
||||||
|
qDebug("p2p file: line %d is malformed.", nbLine);
|
||||||
|
qDebug("End IP is invalid: %s", qPrintable(strStartIP));
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if (startAddr.is_v4() != endAddr.is_v4()) {
|
||||||
|
qDebug("p2p file: line %d is malformed.", nbLine);
|
||||||
|
qDebug("Line was: %s", line.constData());
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
try {
|
||||||
|
filter.add_rule(startAddr, endAddr, libtorrent::ip_filter::blocked);
|
||||||
|
++ruleCount;
|
||||||
|
} catch(std::exception&) {
|
||||||
|
qDebug("p2p file: line %d is malformed.", nbLine);
|
||||||
|
qDebug("Line was: %s", line.constData());
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
file.close();
|
||||||
|
}
|
||||||
|
return ruleCount;
|
||||||
|
}
|
||||||
|
|
||||||
|
int FilterParserThread::getlineInStream(QDataStream& stream, string& name, char delim) {
|
||||||
|
char c;
|
||||||
|
int total_read = 0;
|
||||||
|
int read;
|
||||||
|
do {
|
||||||
|
read = stream.readRawData(&c, 1);
|
||||||
|
total_read += read;
|
||||||
|
if (read > 0) {
|
||||||
|
if (c != delim) {
|
||||||
|
name += c;
|
||||||
|
} else {
|
||||||
|
// Delim found
|
||||||
|
return total_read;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} while(read > 0);
|
||||||
|
return total_read;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Parser for PeerGuardian ip filter in p2p format
|
||||||
|
int FilterParserThread::parseP2BFilterFile(QString filePath, libtorrent::ip_filter& filter) {
|
||||||
|
int ruleCount = 0;
|
||||||
|
QFile file(filePath);
|
||||||
|
if (file.exists()) {
|
||||||
|
if (!file.open(QIODevice::ReadOnly)) {
|
||||||
|
std::cerr << "I/O Error: Could not open ip filer file in read mode." << std::endl;
|
||||||
|
return ruleCount;
|
||||||
|
}
|
||||||
|
QDataStream stream(&file);
|
||||||
|
// Read header
|
||||||
|
char buf[7];
|
||||||
|
unsigned char version;
|
||||||
|
if (
|
||||||
|
!stream.readRawData(buf, sizeof(buf)) ||
|
||||||
|
memcmp(buf, "\xFF\xFF\xFF\xFFP2B", 7) ||
|
||||||
|
!stream.readRawData((char*)&version, sizeof(version))
|
||||||
|
) {
|
||||||
|
std::cerr << "Parsing Error: The filter file is not a valid PeerGuardian P2B file." << std::endl;
|
||||||
|
return ruleCount;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (version==1 || version==2) {
|
||||||
|
qDebug ("p2b version 1 or 2");
|
||||||
|
unsigned int start, end;
|
||||||
|
|
||||||
|
string name;
|
||||||
|
while(getlineInStream(stream, name, '\0') && !abort) {
|
||||||
|
if (
|
||||||
|
!stream.readRawData((char*)&start, sizeof(start)) ||
|
||||||
|
!stream.readRawData((char*)&end, sizeof(end))
|
||||||
|
) {
|
||||||
|
std::cerr << "Parsing Error: The filter file is not a valid PeerGuardian P2B file." << std::endl;
|
||||||
|
return ruleCount;
|
||||||
|
}
|
||||||
|
// Network byte order to Host byte order
|
||||||
|
// asio address_v4 contructor expects it
|
||||||
|
// that way
|
||||||
|
libtorrent::address_v4 first(ntohl(start));
|
||||||
|
libtorrent::address_v4 last(ntohl(end));
|
||||||
|
// Apply to bittorrent session
|
||||||
|
try {
|
||||||
|
filter.add_rule(first, last, libtorrent::ip_filter::blocked);
|
||||||
|
++ruleCount;
|
||||||
|
} catch(std::exception&) {}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else if (version==3) {
|
||||||
|
qDebug ("p2b version 3");
|
||||||
|
unsigned int namecount;
|
||||||
|
if (!stream.readRawData((char*)&namecount, sizeof(namecount))) {
|
||||||
|
std::cerr << "Parsing Error: The filter file is not a valid PeerGuardian P2B file." << std::endl;
|
||||||
|
return ruleCount;
|
||||||
|
}
|
||||||
|
namecount=ntohl(namecount);
|
||||||
|
// Reading names although, we don't really care about them
|
||||||
|
for (unsigned int i=0; i<namecount; i++) {
|
||||||
|
string name;
|
||||||
|
if (!getlineInStream(stream, name, '\0')) {
|
||||||
|
std::cerr << "Parsing Error: The filter file is not a valid PeerGuardian P2B file." << std::endl;
|
||||||
|
return ruleCount;
|
||||||
|
}
|
||||||
|
if (abort) return ruleCount;
|
||||||
|
}
|
||||||
|
// Reading the ranges
|
||||||
|
unsigned int rangecount;
|
||||||
|
if (!stream.readRawData((char*)&rangecount, sizeof(rangecount))) {
|
||||||
|
std::cerr << "Parsing Error: The filter file is not a valid PeerGuardian P2B file." << std::endl;
|
||||||
|
return ruleCount;
|
||||||
|
}
|
||||||
|
rangecount=ntohl(rangecount);
|
||||||
|
|
||||||
|
unsigned int name, start, end;
|
||||||
|
|
||||||
|
for (unsigned int i=0; i<rangecount; i++) {
|
||||||
|
if (
|
||||||
|
!stream.readRawData((char*)&name, sizeof(name)) ||
|
||||||
|
!stream.readRawData((char*)&start, sizeof(start)) ||
|
||||||
|
!stream.readRawData((char*)&end, sizeof(end))
|
||||||
|
) {
|
||||||
|
std::cerr << "Parsing Error: The filter file is not a valid PeerGuardian P2B file." << std::endl;
|
||||||
|
return ruleCount;
|
||||||
|
}
|
||||||
|
// Network byte order to Host byte order
|
||||||
|
// asio address_v4 contructor expects it
|
||||||
|
// that way
|
||||||
|
libtorrent::address_v4 first(ntohl(start));
|
||||||
|
libtorrent::address_v4 last(ntohl(end));
|
||||||
|
// Apply to bittorrent session
|
||||||
|
try {
|
||||||
|
filter.add_rule(first, last, libtorrent::ip_filter::blocked);
|
||||||
|
++ruleCount;
|
||||||
|
} catch(std::exception&) {}
|
||||||
|
if (abort) return ruleCount;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
std::cerr << "Parsing Error: The filter file is not a valid PeerGuardian P2B file." << std::endl;
|
||||||
|
return ruleCount;
|
||||||
|
}
|
||||||
|
file.close();
|
||||||
|
}
|
||||||
|
return ruleCount;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Process ip filter file
|
||||||
|
// Supported formats:
|
||||||
|
// * eMule IP list (DAT): http://wiki.phoenixlabs.org/wiki/DAT_Format
|
||||||
|
// * PeerGuardian Text (P2P): http://wiki.phoenixlabs.org/wiki/P2P_Format
|
||||||
|
// * PeerGuardian Binary (P2B): http://wiki.phoenixlabs.org/wiki/P2B_Format
|
||||||
|
void FilterParserThread::processFilterFile(QString _filePath) {
|
||||||
|
if (isRunning()) {
|
||||||
|
// Already parsing a filter, abort first
|
||||||
|
abort = true;
|
||||||
|
wait();
|
||||||
|
}
|
||||||
|
abort = false;
|
||||||
|
filePath = _filePath;
|
||||||
|
// Run it
|
||||||
|
start();
|
||||||
|
}
|
||||||
|
|
||||||
|
void FilterParserThread::processFilterList(libtorrent::session *s, const QStringList& IPs) {
|
||||||
|
// First, import current filter
|
||||||
|
libtorrent::ip_filter filter = s->get_ip_filter();
|
||||||
|
foreach (const QString &ip, IPs) {
|
||||||
|
qDebug("Manual ban of peer %s", ip.toLocal8Bit().constData());
|
||||||
|
boost::system::error_code ec;
|
||||||
|
libtorrent::address addr = libtorrent::address::from_string(ip.toLocal8Bit().constData(), ec);
|
||||||
|
Q_ASSERT(!ec);
|
||||||
|
if (!ec)
|
||||||
|
filter.add_rule(addr, addr, libtorrent::ip_filter::blocked);
|
||||||
|
}
|
||||||
|
s->set_ip_filter(filter);
|
||||||
|
}
|
||||||
|
|
||||||
|
QString FilterParserThread::cleanupIPAddress(QString _ip) {
|
||||||
|
QHostAddress ip(_ip.trimmed());
|
||||||
|
if (ip.isNull()) {
|
||||||
|
return QString();
|
||||||
|
}
|
||||||
|
return ip.toString();
|
||||||
|
}
|
||||||
|
|
||||||
|
void FilterParserThread::run() {
|
||||||
|
qDebug("Processing filter file");
|
||||||
|
libtorrent::ip_filter filter = s->get_ip_filter();
|
||||||
|
int ruleCount = 0;
|
||||||
|
if (filePath.endsWith(".p2p", Qt::CaseInsensitive)) {
|
||||||
|
// PeerGuardian p2p file
|
||||||
|
ruleCount = parseP2PFilterFile(filePath, filter);
|
||||||
|
} else {
|
||||||
|
if (filePath.endsWith(".p2b", Qt::CaseInsensitive)) {
|
||||||
|
// PeerGuardian p2b file
|
||||||
|
ruleCount = parseP2BFilterFile(filePath, filter);
|
||||||
|
} else {
|
||||||
|
// Default: eMule DAT format
|
||||||
|
ruleCount = parseDATFilterFile(filePath, filter);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (abort)
|
||||||
|
return;
|
||||||
|
try {
|
||||||
|
s->set_ip_filter(filter);
|
||||||
|
emit IPFilterParsed(ruleCount);
|
||||||
|
} catch(std::exception&) {
|
||||||
|
emit IPFilterError();
|
||||||
|
}
|
||||||
|
qDebug("IP Filter thread: finished parsing, filter applied");
|
||||||
|
}
|
@ -32,13 +32,13 @@
|
|||||||
#define FILTERPARSERTHREAD_H
|
#define FILTERPARSERTHREAD_H
|
||||||
|
|
||||||
#include <QThread>
|
#include <QThread>
|
||||||
#include <QFile>
|
|
||||||
#include <QDataStream>
|
#include <QDataStream>
|
||||||
#include <QStringList>
|
#include <QStringList>
|
||||||
#include <QHostAddress>
|
|
||||||
|
|
||||||
#include <libtorrent/session.hpp>
|
namespace libtorrent {
|
||||||
#include <libtorrent/ip_filter.hpp>
|
class session;
|
||||||
|
struct ip_filter;
|
||||||
|
}
|
||||||
|
|
||||||
using namespace std;
|
using namespace std;
|
||||||
|
|
||||||
@ -55,375 +55,28 @@ class FilterParserThread : public QThread {
|
|||||||
Q_OBJECT
|
Q_OBJECT
|
||||||
|
|
||||||
public:
|
public:
|
||||||
FilterParserThread(QObject* parent, libtorrent::session *s) : QThread(parent), s(s), abort(false) {
|
FilterParserThread(QObject* parent, libtorrent::session *s);
|
||||||
|
~FilterParserThread();
|
||||||
|
|
||||||
}
|
int parseDATFilterFile(QString filePath, libtorrent::ip_filter& filter);
|
||||||
|
int parseP2PFilterFile(QString filePath, libtorrent::ip_filter& filter);
|
||||||
~FilterParserThread() {
|
int getlineInStream(QDataStream& stream, string& name, char delim);
|
||||||
abort = true;
|
int parseP2BFilterFile(QString filePath, libtorrent::ip_filter& filter);
|
||||||
wait();
|
void processFilterFile(QString _filePath);
|
||||||
}
|
static void processFilterList(libtorrent::session *s, const QStringList& IPs);
|
||||||
|
|
||||||
// Parser for eMule ip filter in DAT format
|
|
||||||
int parseDATFilterFile(QString filePath) {
|
|
||||||
int ruleCount = 0;
|
|
||||||
QFile file(filePath);
|
|
||||||
if (file.exists()) {
|
|
||||||
if (!file.open(QIODevice::ReadOnly | QIODevice::Text)) {
|
|
||||||
std::cerr << "I/O Error: Could not open ip filer file in read mode." << std::endl;
|
|
||||||
return ruleCount;
|
|
||||||
}
|
|
||||||
unsigned int nbLine = 0;
|
|
||||||
while (!file.atEnd() && !abort) {
|
|
||||||
++nbLine;
|
|
||||||
QByteArray line = file.readLine();
|
|
||||||
// Ignoring empty lines
|
|
||||||
line = line.trimmed();
|
|
||||||
if (line.isEmpty()) continue;
|
|
||||||
// Ignoring commented lines
|
|
||||||
if (line.startsWith('#') || line.startsWith("//")) continue;
|
|
||||||
|
|
||||||
// Line should be splitted by commas
|
|
||||||
QList<QByteArray> partsList = line.split(',');
|
|
||||||
const uint nbElem = partsList.size();
|
|
||||||
|
|
||||||
// IP Range should be splitted by a dash
|
|
||||||
QList<QByteArray> IPs = partsList.first().split('-');
|
|
||||||
if (IPs.size() != 2) {
|
|
||||||
qDebug("Ipfilter.dat: line %d is malformed.", nbLine);
|
|
||||||
qDebug("Line was %s", line.constData());
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
boost::system::error_code ec;
|
|
||||||
const QString strStartIP = cleanupIPAddress(IPs.at(0));
|
|
||||||
if (strStartIP.isEmpty()) {
|
|
||||||
qDebug("Ipfilter.dat: line %d is malformed.", nbLine);
|
|
||||||
qDebug("Start IP of the range is malformated: %s", qPrintable(strStartIP));
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
libtorrent::address startAddr = libtorrent::address::from_string(qPrintable(strStartIP), ec);
|
|
||||||
if (ec) {
|
|
||||||
qDebug("Ipfilter.dat: line %d is malformed.", nbLine);
|
|
||||||
qDebug("Start IP of the range is malformated: %s", qPrintable(strStartIP));
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
const QString strEndIP = cleanupIPAddress(IPs.at(1));
|
|
||||||
if (strEndIP.isEmpty()) {
|
|
||||||
qDebug("Ipfilter.dat: line %d is malformed.", nbLine);
|
|
||||||
qDebug("End IP of the range is malformated: %s", qPrintable(strEndIP));
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
libtorrent::address endAddr = libtorrent::address::from_string(qPrintable(strEndIP), ec);
|
|
||||||
if (ec) {
|
|
||||||
qDebug("Ipfilter.dat: line %d is malformed.", nbLine);
|
|
||||||
qDebug("End IP of the range is malformated: %s", qPrintable(strEndIP));
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
if (startAddr.is_v4() != endAddr.is_v4()) {
|
|
||||||
qDebug("Ipfilter.dat: line %d is malformed.", nbLine);
|
|
||||||
qDebug("One IP is IPv4 and the other is IPv6!");
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Check if there is an access value (apparently not mandatory)
|
|
||||||
int nbAccess = 0;
|
|
||||||
if (nbElem > 1) {
|
|
||||||
// There is possibly one
|
|
||||||
nbAccess = partsList.at(1).trimmed().toInt();
|
|
||||||
}
|
|
||||||
|
|
||||||
if (nbAccess > 127) {
|
|
||||||
// Ignoring this rule because access value is too high
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
// Now Add to the filter
|
|
||||||
try {
|
|
||||||
filter.add_rule(startAddr, endAddr, libtorrent::ip_filter::blocked);
|
|
||||||
++ruleCount;
|
|
||||||
}catch(exception) {
|
|
||||||
qDebug("Bad line in filter file, avoided crash...");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
file.close();
|
|
||||||
}
|
|
||||||
return ruleCount;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Parser for PeerGuardian ip filter in p2p format
|
|
||||||
int parseP2PFilterFile(QString filePath) {
|
|
||||||
int ruleCount = 0;
|
|
||||||
QFile file(filePath);
|
|
||||||
if (file.exists()) {
|
|
||||||
if (!file.open(QIODevice::ReadOnly | QIODevice::Text)) {
|
|
||||||
std::cerr << "I/O Error: Could not open ip filer file in read mode." << std::endl;
|
|
||||||
return ruleCount;
|
|
||||||
}
|
|
||||||
unsigned int nbLine = 0;
|
|
||||||
while (!file.atEnd() && !abort) {
|
|
||||||
++nbLine;
|
|
||||||
QByteArray line = file.readLine().trimmed();
|
|
||||||
if (line.isEmpty()) continue;
|
|
||||||
// Ignoring commented lines
|
|
||||||
if (line.startsWith('#') || line.startsWith("//")) continue;
|
|
||||||
// Line is splitted by :
|
|
||||||
QList<QByteArray> partsList = line.split(':');
|
|
||||||
if (partsList.size() < 2) {
|
|
||||||
qDebug("p2p file: line %d is malformed.", nbLine);
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
// Get IP range
|
|
||||||
QList<QByteArray> IPs = partsList.last().split('-');
|
|
||||||
if (IPs.size() != 2) {
|
|
||||||
qDebug("p2p file: line %d is malformed.", nbLine);
|
|
||||||
qDebug("line was: %s", line.constData());
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
boost::system::error_code ec;
|
|
||||||
QString strStartIP = cleanupIPAddress(IPs.at(0));
|
|
||||||
if (strStartIP.isEmpty()) {
|
|
||||||
qDebug("p2p file: line %d is malformed.", nbLine);
|
|
||||||
qDebug("Start IP is invalid: %s", qPrintable(strStartIP));
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
libtorrent::address startAddr = libtorrent::address::from_string(qPrintable(strStartIP), ec);
|
|
||||||
if (ec) {
|
|
||||||
qDebug("p2p file: line %d is malformed.", nbLine);
|
|
||||||
qDebug("Start IP is invalid: %s", qPrintable(strStartIP));
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
QString strEndIP = cleanupIPAddress(IPs.at(1));
|
|
||||||
if (strEndIP.isEmpty()) {
|
|
||||||
qDebug("p2p file: line %d is malformed.", nbLine);
|
|
||||||
qDebug("End IP is invalid: %s", qPrintable(strStartIP));
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
libtorrent::address endAddr = libtorrent::address::from_string(qPrintable(strEndIP), ec);
|
|
||||||
if (ec) {
|
|
||||||
qDebug("p2p file: line %d is malformed.", nbLine);
|
|
||||||
qDebug("End IP is invalid: %s", qPrintable(strStartIP));
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
if (startAddr.is_v4() != endAddr.is_v4()) {
|
|
||||||
qDebug("p2p file: line %d is malformed.", nbLine);
|
|
||||||
qDebug("Line was: %s", line.constData());
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
try {
|
|
||||||
filter.add_rule(startAddr, endAddr, libtorrent::ip_filter::blocked);
|
|
||||||
++ruleCount;
|
|
||||||
} catch(std::exception&) {
|
|
||||||
qDebug("p2p file: line %d is malformed.", nbLine);
|
|
||||||
qDebug("Line was: %s", line.constData());
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
file.close();
|
|
||||||
}
|
|
||||||
return ruleCount;
|
|
||||||
}
|
|
||||||
|
|
||||||
int getlineInStream(QDataStream& stream, string& name, char delim) {
|
|
||||||
char c;
|
|
||||||
int total_read = 0;
|
|
||||||
int read;
|
|
||||||
do {
|
|
||||||
read = stream.readRawData(&c, 1);
|
|
||||||
total_read += read;
|
|
||||||
if (read > 0) {
|
|
||||||
if (c != delim) {
|
|
||||||
name += c;
|
|
||||||
} else {
|
|
||||||
// Delim found
|
|
||||||
return total_read;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} while(read > 0);
|
|
||||||
return total_read;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Parser for PeerGuardian ip filter in p2p format
|
|
||||||
int parseP2BFilterFile(QString filePath) {
|
|
||||||
int ruleCount = 0;
|
|
||||||
QFile file(filePath);
|
|
||||||
if (file.exists()) {
|
|
||||||
if (!file.open(QIODevice::ReadOnly)) {
|
|
||||||
std::cerr << "I/O Error: Could not open ip filer file in read mode." << std::endl;
|
|
||||||
return ruleCount;
|
|
||||||
}
|
|
||||||
QDataStream stream(&file);
|
|
||||||
// Read header
|
|
||||||
char buf[7];
|
|
||||||
unsigned char version;
|
|
||||||
if (
|
|
||||||
!stream.readRawData(buf, sizeof(buf)) ||
|
|
||||||
memcmp(buf, "\xFF\xFF\xFF\xFFP2B", 7) ||
|
|
||||||
!stream.readRawData((char*)&version, sizeof(version))
|
|
||||||
) {
|
|
||||||
std::cerr << "Parsing Error: The filter file is not a valid PeerGuardian P2B file." << std::endl;
|
|
||||||
return ruleCount;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (version==1 || version==2) {
|
|
||||||
qDebug ("p2b version 1 or 2");
|
|
||||||
unsigned int start, end;
|
|
||||||
|
|
||||||
string name;
|
|
||||||
while(getlineInStream(stream, name, '\0') && !abort) {
|
|
||||||
if (
|
|
||||||
!stream.readRawData((char*)&start, sizeof(start)) ||
|
|
||||||
!stream.readRawData((char*)&end, sizeof(end))
|
|
||||||
) {
|
|
||||||
std::cerr << "Parsing Error: The filter file is not a valid PeerGuardian P2B file." << std::endl;
|
|
||||||
return ruleCount;
|
|
||||||
}
|
|
||||||
// Network byte order to Host byte order
|
|
||||||
// asio address_v4 contructor expects it
|
|
||||||
// that way
|
|
||||||
libtorrent::address_v4 first(ntohl(start));
|
|
||||||
libtorrent::address_v4 last(ntohl(end));
|
|
||||||
// Apply to bittorrent session
|
|
||||||
try {
|
|
||||||
filter.add_rule(first, last, libtorrent::ip_filter::blocked);
|
|
||||||
++ruleCount;
|
|
||||||
} catch(std::exception&) {}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else if (version==3) {
|
|
||||||
qDebug ("p2b version 3");
|
|
||||||
unsigned int namecount;
|
|
||||||
if (!stream.readRawData((char*)&namecount, sizeof(namecount))) {
|
|
||||||
std::cerr << "Parsing Error: The filter file is not a valid PeerGuardian P2B file." << std::endl;
|
|
||||||
return ruleCount;
|
|
||||||
}
|
|
||||||
namecount=ntohl(namecount);
|
|
||||||
// Reading names although, we don't really care about them
|
|
||||||
for (unsigned int i=0; i<namecount; i++) {
|
|
||||||
string name;
|
|
||||||
if (!getlineInStream(stream, name, '\0')) {
|
|
||||||
std::cerr << "Parsing Error: The filter file is not a valid PeerGuardian P2B file." << std::endl;
|
|
||||||
return ruleCount;
|
|
||||||
}
|
|
||||||
if (abort) return ruleCount;
|
|
||||||
}
|
|
||||||
// Reading the ranges
|
|
||||||
unsigned int rangecount;
|
|
||||||
if (!stream.readRawData((char*)&rangecount, sizeof(rangecount))) {
|
|
||||||
std::cerr << "Parsing Error: The filter file is not a valid PeerGuardian P2B file." << std::endl;
|
|
||||||
return ruleCount;
|
|
||||||
}
|
|
||||||
rangecount=ntohl(rangecount);
|
|
||||||
|
|
||||||
unsigned int name, start, end;
|
|
||||||
|
|
||||||
for (unsigned int i=0; i<rangecount; i++) {
|
|
||||||
if (
|
|
||||||
!stream.readRawData((char*)&name, sizeof(name)) ||
|
|
||||||
!stream.readRawData((char*)&start, sizeof(start)) ||
|
|
||||||
!stream.readRawData((char*)&end, sizeof(end))
|
|
||||||
) {
|
|
||||||
std::cerr << "Parsing Error: The filter file is not a valid PeerGuardian P2B file." << std::endl;
|
|
||||||
return ruleCount;
|
|
||||||
}
|
|
||||||
// Network byte order to Host byte order
|
|
||||||
// asio address_v4 contructor expects it
|
|
||||||
// that way
|
|
||||||
libtorrent::address_v4 first(ntohl(start));
|
|
||||||
libtorrent::address_v4 last(ntohl(end));
|
|
||||||
// Apply to bittorrent session
|
|
||||||
try {
|
|
||||||
filter.add_rule(first, last, libtorrent::ip_filter::blocked);
|
|
||||||
++ruleCount;
|
|
||||||
} catch(std::exception&) {}
|
|
||||||
if (abort) return ruleCount;
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
std::cerr << "Parsing Error: The filter file is not a valid PeerGuardian P2B file." << std::endl;
|
|
||||||
return ruleCount;
|
|
||||||
}
|
|
||||||
file.close();
|
|
||||||
}
|
|
||||||
return ruleCount;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Process ip filter file
|
|
||||||
// Supported formats:
|
|
||||||
// * eMule IP list (DAT): http://wiki.phoenixlabs.org/wiki/DAT_Format
|
|
||||||
// * PeerGuardian Text (P2P): http://wiki.phoenixlabs.org/wiki/P2P_Format
|
|
||||||
// * PeerGuardian Binary (P2B): http://wiki.phoenixlabs.org/wiki/P2B_Format
|
|
||||||
void processFilterFile(QString _filePath) {
|
|
||||||
// First, import current filter
|
|
||||||
filter = s->get_ip_filter();
|
|
||||||
if (isRunning()) {
|
|
||||||
// Already parsing a filter, abort first
|
|
||||||
abort = true;
|
|
||||||
wait();
|
|
||||||
}
|
|
||||||
abort = false;
|
|
||||||
filePath = _filePath;
|
|
||||||
// Run it
|
|
||||||
start();
|
|
||||||
}
|
|
||||||
|
|
||||||
static void processFilterList(libtorrent::session *s, const QStringList& IPs) {
|
|
||||||
// First, import current filter
|
|
||||||
libtorrent::ip_filter filter = s->get_ip_filter();
|
|
||||||
foreach (const QString &ip, IPs) {
|
|
||||||
qDebug("Manual ban of peer %s", ip.toLocal8Bit().constData());
|
|
||||||
boost::system::error_code ec;
|
|
||||||
libtorrent::address addr = libtorrent::address::from_string(ip.toLocal8Bit().constData(), ec);
|
|
||||||
Q_ASSERT(!ec);
|
|
||||||
if (!ec)
|
|
||||||
filter.add_rule(addr, addr, libtorrent::ip_filter::blocked);
|
|
||||||
}
|
|
||||||
s->set_ip_filter(filter);
|
|
||||||
}
|
|
||||||
|
|
||||||
signals:
|
signals:
|
||||||
void IPFilterParsed(int ruleCount);
|
void IPFilterParsed(int ruleCount);
|
||||||
void IPFilterError();
|
void IPFilterError();
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
QString cleanupIPAddress(QString _ip) {
|
QString cleanupIPAddress(QString _ip);
|
||||||
QHostAddress ip(_ip.trimmed());
|
void run();
|
||||||
if (ip.isNull()) {
|
|
||||||
return QString();
|
|
||||||
}
|
|
||||||
return ip.toString();
|
|
||||||
}
|
|
||||||
|
|
||||||
void run() {
|
|
||||||
qDebug("Processing filter file");
|
|
||||||
int ruleCount = 0;
|
|
||||||
if (filePath.endsWith(".p2p", Qt::CaseInsensitive)) {
|
|
||||||
// PeerGuardian p2p file
|
|
||||||
ruleCount = parseP2PFilterFile(filePath);
|
|
||||||
} else {
|
|
||||||
if (filePath.endsWith(".p2b", Qt::CaseInsensitive)) {
|
|
||||||
// PeerGuardian p2b file
|
|
||||||
ruleCount = parseP2BFilterFile(filePath);
|
|
||||||
} else {
|
|
||||||
// Default: eMule DAT format
|
|
||||||
ruleCount = parseDATFilterFile(filePath);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (abort)
|
|
||||||
return;
|
|
||||||
try {
|
|
||||||
s->set_ip_filter(filter);
|
|
||||||
emit IPFilterParsed(ruleCount);
|
|
||||||
} catch(std::exception&) {
|
|
||||||
emit IPFilterError();
|
|
||||||
}
|
|
||||||
qDebug("IP Filter thread: finished parsing, filter applied");
|
|
||||||
}
|
|
||||||
|
|
||||||
private:
|
private:
|
||||||
libtorrent::session *s;
|
libtorrent::session *s;
|
||||||
libtorrent::ip_filter filter;
|
|
||||||
bool abort;
|
bool abort;
|
||||||
QString filePath;
|
QString filePath;
|
||||||
|
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
@ -13,7 +13,8 @@ SOURCES += $$PWD/qbtsession.cpp \
|
|||||||
$$PWD/qtorrenthandle.cpp \
|
$$PWD/qtorrenthandle.cpp \
|
||||||
$$PWD/torrentspeedmonitor.cpp \
|
$$PWD/torrentspeedmonitor.cpp \
|
||||||
$$PWD/alertdispatcher.cpp \
|
$$PWD/alertdispatcher.cpp \
|
||||||
$$PWD/torrentstatistics.cpp
|
$$PWD/torrentstatistics.cpp \
|
||||||
|
$$PWD/filterparserthread.cpp
|
||||||
|
|
||||||
!contains(DEFINES, DISABLE_GUI) {
|
!contains(DEFINES, DISABLE_GUI) {
|
||||||
HEADERS += $$PWD/torrentmodel.h \
|
HEADERS += $$PWD/torrentmodel.h \
|
||||||
|
@ -178,7 +178,9 @@ nox {
|
|||||||
autoexpandabledialog.cpp \
|
autoexpandabledialog.cpp \
|
||||||
statsdialog.cpp \
|
statsdialog.cpp \
|
||||||
messageboxraised.cpp \
|
messageboxraised.cpp \
|
||||||
statussortfilterproxymodel.cpp
|
statussortfilterproxymodel.cpp \
|
||||||
|
statusbar.cpp \
|
||||||
|
trackerlogin.cpp
|
||||||
|
|
||||||
win32 {
|
win32 {
|
||||||
HEADERS += programupdater.h
|
HEADERS += programupdater.h
|
||||||
|
250
src/statusbar.cpp
Normal file
250
src/statusbar.cpp
Normal file
@ -0,0 +1,250 @@
|
|||||||
|
/*
|
||||||
|
* 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 "statusbar.h"
|
||||||
|
|
||||||
|
#include <QFontMetrics>
|
||||||
|
#include <QDebug>
|
||||||
|
|
||||||
|
#include "qbtsession.h"
|
||||||
|
#include "speedlimitdlg.h"
|
||||||
|
#include "iconprovider.h"
|
||||||
|
#include "preferences.h"
|
||||||
|
#include "misc.h"
|
||||||
|
|
||||||
|
#include <libtorrent/session.hpp>
|
||||||
|
#include <libtorrent/session_status.hpp>
|
||||||
|
|
||||||
|
StatusBar::StatusBar(QStatusBar *bar)
|
||||||
|
: m_bar(bar)
|
||||||
|
{
|
||||||
|
Preferences* const pref = Preferences::instance();
|
||||||
|
connect(QBtSession::instance(), SIGNAL(alternativeSpeedsModeChanged(bool)), this, SLOT(updateAltSpeedsBtn(bool)));
|
||||||
|
container = new QWidget(bar);
|
||||||
|
layout = new QHBoxLayout(container);
|
||||||
|
layout->setContentsMargins(0,0,0,0);
|
||||||
|
|
||||||
|
container->setLayout(layout);
|
||||||
|
connecStatusLblIcon = new QPushButton(bar);
|
||||||
|
connecStatusLblIcon->setFlat(true);
|
||||||
|
connecStatusLblIcon->setFocusPolicy(Qt::NoFocus);
|
||||||
|
connecStatusLblIcon->setFixedWidth(32);
|
||||||
|
connecStatusLblIcon->setCursor(Qt::PointingHandCursor);
|
||||||
|
connecStatusLblIcon->setIcon(QIcon(":/Icons/skin/firewalled.png"));
|
||||||
|
connecStatusLblIcon->setToolTip(QString::fromUtf8("<b>")+tr("Connection status:")+QString::fromUtf8("</b><br>")+QString::fromUtf8("<i>")+tr("No direct connections. This may indicate network configuration problems.")+QString::fromUtf8("</i>"));
|
||||||
|
dlSpeedLbl = new QPushButton(bar);
|
||||||
|
dlSpeedLbl->setIconSize(QSize(16,16));
|
||||||
|
dlSpeedLbl->setIcon(QIcon(":/Icons/skin/download.png"));
|
||||||
|
//dlSpeedLbl->setSizePolicy(QSizePolicy::Maximum, QSizePolicy::Preferred);
|
||||||
|
connect(dlSpeedLbl, SIGNAL(clicked()), this, SLOT(capDownloadSpeed()));
|
||||||
|
dlSpeedLbl->setFlat(true);
|
||||||
|
dlSpeedLbl->setFocusPolicy(Qt::NoFocus);
|
||||||
|
dlSpeedLbl->setCursor(Qt::PointingHandCursor);
|
||||||
|
|
||||||
|
altSpeedsBtn = new QPushButton(bar);
|
||||||
|
altSpeedsBtn->setFixedWidth(36);
|
||||||
|
altSpeedsBtn->setIconSize(QSize(32,32));
|
||||||
|
altSpeedsBtn->setFlat(true);
|
||||||
|
altSpeedsBtn->setFocusPolicy(Qt::NoFocus);
|
||||||
|
altSpeedsBtn->setCursor(Qt::PointingHandCursor);
|
||||||
|
updateAltSpeedsBtn(pref->isAltBandwidthEnabled());
|
||||||
|
|
||||||
|
connect(altSpeedsBtn, SIGNAL(clicked()), this, SLOT(toggleAlternativeSpeeds()));
|
||||||
|
|
||||||
|
upSpeedLbl = new QPushButton(bar);
|
||||||
|
upSpeedLbl->setIconSize(QSize(16,16));
|
||||||
|
upSpeedLbl->setIcon(QIcon(":/Icons/skin/seeding.png"));
|
||||||
|
//upSpeedLbl->setSizePolicy(QSizePolicy::Maximum, QSizePolicy::Preferred);
|
||||||
|
connect(upSpeedLbl, SIGNAL(clicked()), this, SLOT(capUploadSpeed()));
|
||||||
|
upSpeedLbl->setFlat(true);
|
||||||
|
upSpeedLbl->setFocusPolicy(Qt::NoFocus);
|
||||||
|
upSpeedLbl->setCursor(Qt::PointingHandCursor);
|
||||||
|
DHTLbl = new QLabel(tr("DHT: %1 nodes").arg(0), bar);
|
||||||
|
DHTLbl->setSizePolicy(QSizePolicy::Maximum, QSizePolicy::Preferred);
|
||||||
|
statusSep1 = new QFrame(bar);
|
||||||
|
statusSep1->setFixedSize(3, dlSpeedLbl->fontMetrics().height());
|
||||||
|
statusSep1->setFrameStyle(QFrame::VLine);
|
||||||
|
statusSep1->setFrameShadow(QFrame::Raised);
|
||||||
|
statusSep2 = new QFrame(bar);
|
||||||
|
statusSep2->setFixedSize(3, dlSpeedLbl->fontMetrics().height());
|
||||||
|
statusSep2->setFrameStyle(QFrame::VLine);
|
||||||
|
statusSep2->setFrameShadow(QFrame::Raised);
|
||||||
|
statusSep3 = new QFrame(bar);
|
||||||
|
statusSep3->setFixedSize(3, dlSpeedLbl->fontMetrics().height());
|
||||||
|
statusSep3->setFrameStyle(QFrame::VLine);
|
||||||
|
statusSep3->setFrameShadow(QFrame::Raised);
|
||||||
|
statusSep4 = new QFrame(bar);
|
||||||
|
statusSep4->setFixedSize(3, dlSpeedLbl->fontMetrics().height());
|
||||||
|
statusSep4->setFrameStyle(QFrame::VLine);
|
||||||
|
statusSep4->setFrameShadow(QFrame::Raised);
|
||||||
|
layout->addWidget(DHTLbl);
|
||||||
|
layout->addWidget(statusSep1);
|
||||||
|
layout->addWidget(connecStatusLblIcon);
|
||||||
|
layout->addWidget(statusSep2);
|
||||||
|
layout->addWidget(altSpeedsBtn);
|
||||||
|
layout->addWidget(statusSep4);
|
||||||
|
layout->addWidget(dlSpeedLbl);
|
||||||
|
layout->addWidget(statusSep3);
|
||||||
|
layout->addWidget(upSpeedLbl);
|
||||||
|
|
||||||
|
bar->addPermanentWidget(container);
|
||||||
|
container->setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Preferred);
|
||||||
|
//bar->setStyleSheet("QWidget {padding-top: 0; padding-bottom: 0; margin-top: 0; margin-bottom: 0;}\n");
|
||||||
|
container->setContentsMargins(0, 0, 0, 1);
|
||||||
|
bar->setContentsMargins(0, 0, 0, 0);
|
||||||
|
container->setFixedHeight(dlSpeedLbl->fontMetrics().height()+7);
|
||||||
|
bar->setFixedHeight(dlSpeedLbl->fontMetrics().height()+9);
|
||||||
|
// Is DHT enabled
|
||||||
|
DHTLbl->setVisible(pref->isDHTEnabled());
|
||||||
|
refreshTimer = new QTimer(bar);
|
||||||
|
refreshStatusBar();
|
||||||
|
connect(refreshTimer, SIGNAL(timeout()), this, SLOT(refreshStatusBar()));
|
||||||
|
refreshTimer->start(1500);
|
||||||
|
}
|
||||||
|
|
||||||
|
StatusBar::~StatusBar() {
|
||||||
|
qDebug() << Q_FUNC_INFO;
|
||||||
|
}
|
||||||
|
|
||||||
|
QPushButton* StatusBar::connectionStatusButton() const {
|
||||||
|
return connecStatusLblIcon;
|
||||||
|
}
|
||||||
|
|
||||||
|
void StatusBar::showRestartRequired() {
|
||||||
|
// Restart required notification
|
||||||
|
const QString restart_text = tr("qBittorrent needs to be restarted");
|
||||||
|
QLabel *restartIconLbl = new QLabel(m_bar);
|
||||||
|
restartIconLbl->setPixmap(QPixmap(":/Icons/oxygen/dialog-warning.png").scaled(QSize(24,24)));
|
||||||
|
restartIconLbl->setToolTip(restart_text);
|
||||||
|
m_bar->insertWidget(0,restartIconLbl);
|
||||||
|
QLabel *restartLbl = new QLabel(m_bar);
|
||||||
|
restartLbl->setSizePolicy(QSizePolicy::MinimumExpanding, QSizePolicy::Preferred);
|
||||||
|
m_bar->insertWidget(1, restartLbl);
|
||||||
|
QFontMetrics fm(restartLbl->font());
|
||||||
|
restartLbl->setText(fm.elidedText(restart_text, Qt::ElideRight, restartLbl->width()));
|
||||||
|
QBtSession::instance()->addConsoleMessage(tr("qBittorrent was just updated and needs to be restarted for the changes to be effective."), "red");
|
||||||
|
}
|
||||||
|
|
||||||
|
void StatusBar::stopTimer() {
|
||||||
|
refreshTimer->stop();
|
||||||
|
}
|
||||||
|
|
||||||
|
void StatusBar::refreshStatusBar() {
|
||||||
|
// Update connection status
|
||||||
|
const libtorrent::session_status sessionStatus = QBtSession::instance()->getSessionStatus();
|
||||||
|
if (!QBtSession::instance()->getSession()->is_listening()) {
|
||||||
|
connecStatusLblIcon->setIcon(QIcon(QString::fromUtf8(":/Icons/skin/disconnected.png")));
|
||||||
|
connecStatusLblIcon->setToolTip(QString::fromUtf8("<b>")+tr("Connection Status:")+QString::fromUtf8("</b><br>")+tr("Offline. This usually means that qBittorrent failed to listen on the selected port for incoming connections."));
|
||||||
|
} else {
|
||||||
|
if (sessionStatus.has_incoming_connections) {
|
||||||
|
// Connection OK
|
||||||
|
connecStatusLblIcon->setIcon(QIcon(QString::fromUtf8(":/Icons/skin/connected.png")));
|
||||||
|
connecStatusLblIcon->setToolTip(QString::fromUtf8("<b>")+tr("Connection Status:")+QString::fromUtf8("</b><br>")+tr("Online"));
|
||||||
|
}else{
|
||||||
|
connecStatusLblIcon->setIcon(QIcon(QString::fromUtf8(":/Icons/skin/firewalled.png")));
|
||||||
|
connecStatusLblIcon->setToolTip(QString::fromUtf8("<b>")+tr("Connection status:")+QString::fromUtf8("</b><br>")+QString::fromUtf8("<i>")+tr("No direct connections. This may indicate network configuration problems.")+QString::fromUtf8("</i>"));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// Update Number of DHT nodes
|
||||||
|
if (QBtSession::instance()->isDHTEnabled()) {
|
||||||
|
DHTLbl->setVisible(true);
|
||||||
|
//statusSep1->setVisible(true);
|
||||||
|
DHTLbl->setText(tr("DHT: %1 nodes").arg(QString::number(sessionStatus.dht_nodes)));
|
||||||
|
} else {
|
||||||
|
DHTLbl->setVisible(false);
|
||||||
|
//statusSep1->setVisible(false);
|
||||||
|
}
|
||||||
|
// Update speed labels
|
||||||
|
dlSpeedLbl->setText(tr("%1/s", "Per second").arg(misc::friendlyUnit(sessionStatus.payload_download_rate))+" ("+misc::friendlyUnit(sessionStatus.total_payload_download)+")");
|
||||||
|
upSpeedLbl->setText(tr("%1/s", "Per second").arg(misc::friendlyUnit(sessionStatus.payload_upload_rate))+" ("+misc::friendlyUnit(sessionStatus.total_payload_upload)+")");
|
||||||
|
}
|
||||||
|
|
||||||
|
void StatusBar::updateAltSpeedsBtn(bool alternative) {
|
||||||
|
if (alternative) {
|
||||||
|
altSpeedsBtn->setIcon(QIcon(":/Icons/slow.png"));
|
||||||
|
altSpeedsBtn->setToolTip(tr("Click to switch to regular speed limits"));
|
||||||
|
altSpeedsBtn->setDown(true);
|
||||||
|
} else {
|
||||||
|
altSpeedsBtn->setIcon(QIcon(":/Icons/slow_off.png"));
|
||||||
|
altSpeedsBtn->setToolTip(tr("Click to switch to alternative speed limits"));
|
||||||
|
altSpeedsBtn->setDown(false);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void StatusBar::toggleAlternativeSpeeds() {
|
||||||
|
Preferences* const pref = Preferences::instance();
|
||||||
|
if (pref->isSchedulerEnabled()) {
|
||||||
|
pref->setSchedulerEnabled(false);
|
||||||
|
m_bar->showMessage(tr("Manual change of rate limits mode. The scheduler is disabled."), 5000);
|
||||||
|
}
|
||||||
|
QBtSession::instance()->useAlternativeSpeedsLimit(!pref->isAltBandwidthEnabled());
|
||||||
|
}
|
||||||
|
|
||||||
|
void StatusBar::capDownloadSpeed() {
|
||||||
|
bool ok = false;
|
||||||
|
int cur_limit = QBtSession::instance()->getSession()->settings().download_rate_limit;
|
||||||
|
long new_limit = SpeedLimitDialog::askSpeedLimit(&ok, tr("Global Download Speed Limit"), cur_limit);
|
||||||
|
if (ok) {
|
||||||
|
Preferences* const pref = Preferences::instance();
|
||||||
|
bool alt = pref->isAltBandwidthEnabled();
|
||||||
|
if (new_limit <= 0) {
|
||||||
|
qDebug("Setting global download rate limit to Unlimited");
|
||||||
|
QBtSession::instance()->setDownloadRateLimit(-1);
|
||||||
|
if (!alt)
|
||||||
|
pref->setGlobalDownloadLimit(-1);
|
||||||
|
} else {
|
||||||
|
qDebug("Setting global download rate limit to %.1fKb/s", new_limit/1024.);
|
||||||
|
QBtSession::instance()->setDownloadRateLimit(new_limit);
|
||||||
|
if (!alt)
|
||||||
|
pref->setGlobalDownloadLimit(new_limit/1024.);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void StatusBar::capUploadSpeed() {
|
||||||
|
bool ok = false;
|
||||||
|
int cur_limit = QBtSession::instance()->getSession()->settings().upload_rate_limit;
|
||||||
|
long new_limit = SpeedLimitDialog::askSpeedLimit(&ok, tr("Global Upload Speed Limit"), cur_limit);
|
||||||
|
if (ok) {
|
||||||
|
Preferences* const pref = Preferences::instance();
|
||||||
|
bool alt = pref->isAltBandwidthEnabled();
|
||||||
|
if (new_limit <= 0) {
|
||||||
|
qDebug("Setting global upload rate limit to Unlimited");
|
||||||
|
QBtSession::instance()->setUploadRateLimit(-1);
|
||||||
|
if (!alt)
|
||||||
|
Preferences::instance()->setGlobalUploadLimit(-1);
|
||||||
|
} else {
|
||||||
|
qDebug("Setting global upload rate limit to %.1fKb/s", new_limit/1024.);
|
||||||
|
QBtSession::instance()->setUploadRateLimit(new_limit);
|
||||||
|
if (!alt)
|
||||||
|
Preferences::instance()->setGlobalUploadLimit(new_limit/1024.);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
222
src/statusbar.h
222
src/statusbar.h
@ -38,226 +38,24 @@
|
|||||||
#include <QGridLayout>
|
#include <QGridLayout>
|
||||||
#include <QPushButton>
|
#include <QPushButton>
|
||||||
#include <QHBoxLayout>
|
#include <QHBoxLayout>
|
||||||
#include <QFontMetrics>
|
|
||||||
#include <QDebug>
|
|
||||||
#include "qbtsession.h"
|
|
||||||
#include "speedlimitdlg.h"
|
|
||||||
#include "iconprovider.h"
|
|
||||||
#include "preferences.h"
|
|
||||||
#include "misc.h"
|
|
||||||
|
|
||||||
#include <libtorrent/session.hpp>
|
|
||||||
#include <libtorrent/session_status.hpp>
|
|
||||||
|
|
||||||
class StatusBar: public QObject {
|
class StatusBar: public QObject {
|
||||||
Q_OBJECT
|
Q_OBJECT
|
||||||
|
|
||||||
public:
|
public:
|
||||||
StatusBar(QStatusBar *bar): m_bar(bar) {
|
StatusBar(QStatusBar *bar);
|
||||||
Preferences* const pref = Preferences::instance();
|
~StatusBar();
|
||||||
connect(QBtSession::instance(), SIGNAL(alternativeSpeedsModeChanged(bool)), this, SLOT(updateAltSpeedsBtn(bool)));
|
|
||||||
container = new QWidget(bar);
|
|
||||||
layout = new QHBoxLayout(container);
|
|
||||||
layout->setContentsMargins(0,0,0,0);
|
|
||||||
|
|
||||||
container->setLayout(layout);
|
QPushButton* connectionStatusButton() const;
|
||||||
connecStatusLblIcon = new QPushButton(bar);
|
|
||||||
connecStatusLblIcon->setFlat(true);
|
|
||||||
connecStatusLblIcon->setFocusPolicy(Qt::NoFocus);
|
|
||||||
connecStatusLblIcon->setFixedWidth(32);
|
|
||||||
connecStatusLblIcon->setCursor(Qt::PointingHandCursor);
|
|
||||||
connecStatusLblIcon->setIcon(QIcon(":/Icons/skin/firewalled.png"));
|
|
||||||
connecStatusLblIcon->setToolTip(QString::fromUtf8("<b>")+tr("Connection status:")+QString::fromUtf8("</b><br>")+QString::fromUtf8("<i>")+tr("No direct connections. This may indicate network configuration problems.")+QString::fromUtf8("</i>"));
|
|
||||||
dlSpeedLbl = new QPushButton(bar);
|
|
||||||
dlSpeedLbl->setIconSize(QSize(16,16));
|
|
||||||
dlSpeedLbl->setIcon(QIcon(":/Icons/skin/download.png"));
|
|
||||||
//dlSpeedLbl->setSizePolicy(QSizePolicy::Maximum, QSizePolicy::Preferred);
|
|
||||||
connect(dlSpeedLbl, SIGNAL(clicked()), this, SLOT(capDownloadSpeed()));
|
|
||||||
dlSpeedLbl->setFlat(true);
|
|
||||||
dlSpeedLbl->setFocusPolicy(Qt::NoFocus);
|
|
||||||
dlSpeedLbl->setCursor(Qt::PointingHandCursor);
|
|
||||||
|
|
||||||
altSpeedsBtn = new QPushButton(bar);
|
|
||||||
altSpeedsBtn->setFixedWidth(36);
|
|
||||||
altSpeedsBtn->setIconSize(QSize(32,32));
|
|
||||||
altSpeedsBtn->setFlat(true);
|
|
||||||
altSpeedsBtn->setFocusPolicy(Qt::NoFocus);
|
|
||||||
altSpeedsBtn->setCursor(Qt::PointingHandCursor);
|
|
||||||
updateAltSpeedsBtn(pref->isAltBandwidthEnabled());
|
|
||||||
|
|
||||||
connect(altSpeedsBtn, SIGNAL(clicked()), this, SLOT(toggleAlternativeSpeeds()));
|
|
||||||
|
|
||||||
upSpeedLbl = new QPushButton(bar);
|
|
||||||
upSpeedLbl->setIconSize(QSize(16,16));
|
|
||||||
upSpeedLbl->setIcon(QIcon(":/Icons/skin/seeding.png"));
|
|
||||||
//upSpeedLbl->setSizePolicy(QSizePolicy::Maximum, QSizePolicy::Preferred);
|
|
||||||
connect(upSpeedLbl, SIGNAL(clicked()), this, SLOT(capUploadSpeed()));
|
|
||||||
upSpeedLbl->setFlat(true);
|
|
||||||
upSpeedLbl->setFocusPolicy(Qt::NoFocus);
|
|
||||||
upSpeedLbl->setCursor(Qt::PointingHandCursor);
|
|
||||||
DHTLbl = new QLabel(tr("DHT: %1 nodes").arg(0), bar);
|
|
||||||
DHTLbl->setSizePolicy(QSizePolicy::Maximum, QSizePolicy::Preferred);
|
|
||||||
statusSep1 = new QFrame(bar);
|
|
||||||
statusSep1->setFixedSize(3, dlSpeedLbl->fontMetrics().height());
|
|
||||||
statusSep1->setFrameStyle(QFrame::VLine);
|
|
||||||
statusSep1->setFrameShadow(QFrame::Raised);
|
|
||||||
statusSep2 = new QFrame(bar);
|
|
||||||
statusSep2->setFixedSize(3, dlSpeedLbl->fontMetrics().height());
|
|
||||||
statusSep2->setFrameStyle(QFrame::VLine);
|
|
||||||
statusSep2->setFrameShadow(QFrame::Raised);
|
|
||||||
statusSep3 = new QFrame(bar);
|
|
||||||
statusSep3->setFixedSize(3, dlSpeedLbl->fontMetrics().height());
|
|
||||||
statusSep3->setFrameStyle(QFrame::VLine);
|
|
||||||
statusSep3->setFrameShadow(QFrame::Raised);
|
|
||||||
statusSep4 = new QFrame(bar);
|
|
||||||
statusSep4->setFixedSize(3, dlSpeedLbl->fontMetrics().height());
|
|
||||||
statusSep4->setFrameStyle(QFrame::VLine);
|
|
||||||
statusSep4->setFrameShadow(QFrame::Raised);
|
|
||||||
layout->addWidget(DHTLbl);
|
|
||||||
layout->addWidget(statusSep1);
|
|
||||||
layout->addWidget(connecStatusLblIcon);
|
|
||||||
layout->addWidget(statusSep2);
|
|
||||||
layout->addWidget(altSpeedsBtn);
|
|
||||||
layout->addWidget(statusSep4);
|
|
||||||
layout->addWidget(dlSpeedLbl);
|
|
||||||
layout->addWidget(statusSep3);
|
|
||||||
layout->addWidget(upSpeedLbl);
|
|
||||||
|
|
||||||
bar->addPermanentWidget(container);
|
|
||||||
container->setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Preferred);
|
|
||||||
//bar->setStyleSheet("QWidget {padding-top: 0; padding-bottom: 0; margin-top: 0; margin-bottom: 0;}\n");
|
|
||||||
container->setContentsMargins(0, 0, 0, 1);
|
|
||||||
bar->setContentsMargins(0, 0, 0, 0);
|
|
||||||
container->setFixedHeight(dlSpeedLbl->fontMetrics().height()+7);
|
|
||||||
bar->setFixedHeight(dlSpeedLbl->fontMetrics().height()+9);
|
|
||||||
// Is DHT enabled
|
|
||||||
DHTLbl->setVisible(pref->isDHTEnabled());
|
|
||||||
refreshTimer = new QTimer(bar);
|
|
||||||
refreshStatusBar();
|
|
||||||
connect(refreshTimer, SIGNAL(timeout()), this, SLOT(refreshStatusBar()));
|
|
||||||
refreshTimer->start(1500);
|
|
||||||
}
|
|
||||||
|
|
||||||
~StatusBar() {
|
|
||||||
qDebug() << Q_FUNC_INFO;
|
|
||||||
}
|
|
||||||
|
|
||||||
QPushButton* connectionStatusButton() const {
|
|
||||||
return connecStatusLblIcon;
|
|
||||||
}
|
|
||||||
|
|
||||||
public slots:
|
public slots:
|
||||||
void showRestartRequired() {
|
void showRestartRequired();
|
||||||
// Restart required notification
|
void stopTimer();
|
||||||
const QString restart_text = tr("qBittorrent needs to be restarted");
|
void refreshStatusBar();
|
||||||
QLabel *restartIconLbl = new QLabel(m_bar);
|
void updateAltSpeedsBtn(bool alternative);
|
||||||
restartIconLbl->setPixmap(QPixmap(":/Icons/oxygen/dialog-warning.png").scaled(QSize(24,24)));
|
void toggleAlternativeSpeeds();
|
||||||
restartIconLbl->setToolTip(restart_text);
|
void capDownloadSpeed();
|
||||||
m_bar->insertWidget(0,restartIconLbl);
|
void capUploadSpeed();
|
||||||
QLabel *restartLbl = new QLabel(m_bar);
|
|
||||||
restartLbl->setSizePolicy(QSizePolicy::MinimumExpanding, QSizePolicy::Preferred);
|
|
||||||
m_bar->insertWidget(1, restartLbl);
|
|
||||||
QFontMetrics fm(restartLbl->font());
|
|
||||||
restartLbl->setText(fm.elidedText(restart_text, Qt::ElideRight, restartLbl->width()));
|
|
||||||
QBtSession::instance()->addConsoleMessage(tr("qBittorrent was just updated and needs to be restarted for the changes to be effective."), "red");
|
|
||||||
}
|
|
||||||
|
|
||||||
void stopTimer() {
|
|
||||||
refreshTimer->stop();
|
|
||||||
}
|
|
||||||
|
|
||||||
void refreshStatusBar() {
|
|
||||||
// Update connection status
|
|
||||||
const libtorrent::session_status sessionStatus = QBtSession::instance()->getSessionStatus();
|
|
||||||
if (!QBtSession::instance()->getSession()->is_listening()) {
|
|
||||||
connecStatusLblIcon->setIcon(QIcon(QString::fromUtf8(":/Icons/skin/disconnected.png")));
|
|
||||||
connecStatusLblIcon->setToolTip(QString::fromUtf8("<b>")+tr("Connection Status:")+QString::fromUtf8("</b><br>")+tr("Offline. This usually means that qBittorrent failed to listen on the selected port for incoming connections."));
|
|
||||||
} else {
|
|
||||||
if (sessionStatus.has_incoming_connections) {
|
|
||||||
// Connection OK
|
|
||||||
connecStatusLblIcon->setIcon(QIcon(QString::fromUtf8(":/Icons/skin/connected.png")));
|
|
||||||
connecStatusLblIcon->setToolTip(QString::fromUtf8("<b>")+tr("Connection Status:")+QString::fromUtf8("</b><br>")+tr("Online"));
|
|
||||||
}else{
|
|
||||||
connecStatusLblIcon->setIcon(QIcon(QString::fromUtf8(":/Icons/skin/firewalled.png")));
|
|
||||||
connecStatusLblIcon->setToolTip(QString::fromUtf8("<b>")+tr("Connection status:")+QString::fromUtf8("</b><br>")+QString::fromUtf8("<i>")+tr("No direct connections. This may indicate network configuration problems.")+QString::fromUtf8("</i>"));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
// Update Number of DHT nodes
|
|
||||||
if (QBtSession::instance()->isDHTEnabled()) {
|
|
||||||
DHTLbl->setVisible(true);
|
|
||||||
//statusSep1->setVisible(true);
|
|
||||||
DHTLbl->setText(tr("DHT: %1 nodes").arg(QString::number(sessionStatus.dht_nodes)));
|
|
||||||
} else {
|
|
||||||
DHTLbl->setVisible(false);
|
|
||||||
//statusSep1->setVisible(false);
|
|
||||||
}
|
|
||||||
// Update speed labels
|
|
||||||
dlSpeedLbl->setText(tr("%1/s", "Per second").arg(misc::friendlyUnit(sessionStatus.payload_download_rate))+" ("+misc::friendlyUnit(sessionStatus.total_payload_download)+")");
|
|
||||||
upSpeedLbl->setText(tr("%1/s", "Per second").arg(misc::friendlyUnit(sessionStatus.payload_upload_rate))+" ("+misc::friendlyUnit(sessionStatus.total_payload_upload)+")");
|
|
||||||
}
|
|
||||||
|
|
||||||
void updateAltSpeedsBtn(bool alternative) {
|
|
||||||
if (alternative) {
|
|
||||||
altSpeedsBtn->setIcon(QIcon(":/Icons/slow.png"));
|
|
||||||
altSpeedsBtn->setToolTip(tr("Click to switch to regular speed limits"));
|
|
||||||
altSpeedsBtn->setDown(true);
|
|
||||||
} else {
|
|
||||||
altSpeedsBtn->setIcon(QIcon(":/Icons/slow_off.png"));
|
|
||||||
altSpeedsBtn->setToolTip(tr("Click to switch to alternative speed limits"));
|
|
||||||
altSpeedsBtn->setDown(false);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void toggleAlternativeSpeeds() {
|
|
||||||
Preferences* const pref = Preferences::instance();
|
|
||||||
if (pref->isSchedulerEnabled()) {
|
|
||||||
pref->setSchedulerEnabled(false);
|
|
||||||
m_bar->showMessage(tr("Manual change of rate limits mode. The scheduler is disabled."), 5000);
|
|
||||||
}
|
|
||||||
QBtSession::instance()->useAlternativeSpeedsLimit(!pref->isAltBandwidthEnabled());
|
|
||||||
}
|
|
||||||
|
|
||||||
void capDownloadSpeed() {
|
|
||||||
bool ok = false;
|
|
||||||
int cur_limit = QBtSession::instance()->getSession()->settings().download_rate_limit;
|
|
||||||
long new_limit = SpeedLimitDialog::askSpeedLimit(&ok, tr("Global Download Speed Limit"), cur_limit);
|
|
||||||
if (ok) {
|
|
||||||
Preferences* const pref = Preferences::instance();
|
|
||||||
bool alt = pref->isAltBandwidthEnabled();
|
|
||||||
if (new_limit <= 0) {
|
|
||||||
qDebug("Setting global download rate limit to Unlimited");
|
|
||||||
QBtSession::instance()->setDownloadRateLimit(-1);
|
|
||||||
if (!alt)
|
|
||||||
pref->setGlobalDownloadLimit(-1);
|
|
||||||
} else {
|
|
||||||
qDebug("Setting global download rate limit to %.1fKb/s", new_limit/1024.);
|
|
||||||
QBtSession::instance()->setDownloadRateLimit(new_limit);
|
|
||||||
if (!alt)
|
|
||||||
pref->setGlobalDownloadLimit(new_limit/1024.);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void capUploadSpeed() {
|
|
||||||
bool ok = false;
|
|
||||||
int cur_limit = QBtSession::instance()->getSession()->settings().upload_rate_limit;
|
|
||||||
long new_limit = SpeedLimitDialog::askSpeedLimit(&ok, tr("Global Upload Speed Limit"), cur_limit);
|
|
||||||
if (ok) {
|
|
||||||
Preferences* const pref = Preferences::instance();
|
|
||||||
bool alt = pref->isAltBandwidthEnabled();
|
|
||||||
if (new_limit <= 0) {
|
|
||||||
qDebug("Setting global upload rate limit to Unlimited");
|
|
||||||
QBtSession::instance()->setUploadRateLimit(-1);
|
|
||||||
if (!alt)
|
|
||||||
Preferences::instance()->setGlobalUploadLimit(-1);
|
|
||||||
} else {
|
|
||||||
qDebug("Setting global upload rate limit to %.1fKb/s", new_limit/1024.);
|
|
||||||
QBtSession::instance()->setUploadRateLimit(new_limit);
|
|
||||||
if (!alt)
|
|
||||||
Preferences::instance()->setGlobalUploadLimit(new_limit/1024.);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private:
|
private:
|
||||||
QStatusBar *m_bar;
|
QStatusBar *m_bar;
|
||||||
|
57
src/trackerlogin.cpp
Normal file
57
src/trackerlogin.cpp
Normal file
@ -0,0 +1,57 @@
|
|||||||
|
/*
|
||||||
|
* 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 "trackerlogin.h"
|
||||||
|
|
||||||
|
trackerLogin::trackerLogin(QWidget *parent, QTorrentHandle h)
|
||||||
|
: QDialog(parent)
|
||||||
|
, h(h)
|
||||||
|
{
|
||||||
|
setupUi(this);
|
||||||
|
setAttribute(Qt::WA_DeleteOnClose);
|
||||||
|
login_logo->setPixmap(QPixmap(QString::fromUtf8(":/Icons/oxygen/encrypted.png")));
|
||||||
|
tracker_url->setText(h.current_tracker());
|
||||||
|
connect(this, SIGNAL(trackerLoginCancelled(QPair<QTorrentHandle,QString>)), parent, SLOT(addUnauthenticatedTracker(QPair<QTorrentHandle,QString>)));
|
||||||
|
show();
|
||||||
|
}
|
||||||
|
|
||||||
|
trackerLogin::~trackerLogin() {}
|
||||||
|
|
||||||
|
void trackerLogin::on_loginButton_clicked() {
|
||||||
|
// login
|
||||||
|
h.set_tracker_login(lineUsername->text(), linePasswd->text());
|
||||||
|
close();
|
||||||
|
}
|
||||||
|
|
||||||
|
void trackerLogin::on_cancelButton_clicked() {
|
||||||
|
// Emit a signal to GUI to stop asking for authentication
|
||||||
|
emit trackerLoginCancelled(QPair<QTorrentHandle,QString>(h, h.current_tracker()));
|
||||||
|
close();
|
||||||
|
}
|
@ -32,9 +32,6 @@
|
|||||||
#define TRACKERLOGIN_H
|
#define TRACKERLOGIN_H
|
||||||
|
|
||||||
#include <QDialog>
|
#include <QDialog>
|
||||||
#include <QMessageBox>
|
|
||||||
|
|
||||||
#include <libtorrent/session.hpp>
|
|
||||||
|
|
||||||
#include "ui_login.h"
|
#include "ui_login.h"
|
||||||
#include "qtorrenthandle.h"
|
#include "qtorrenthandle.h"
|
||||||
@ -46,32 +43,15 @@ class trackerLogin : public QDialog, private Ui::authentication{
|
|||||||
QTorrentHandle h;
|
QTorrentHandle h;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
trackerLogin(QWidget *parent, QTorrentHandle h): QDialog(parent), h(h) {
|
trackerLogin(QWidget *parent, QTorrentHandle h);
|
||||||
setupUi(this);
|
~trackerLogin();
|
||||||
setAttribute(Qt::WA_DeleteOnClose);
|
|
||||||
login_logo->setPixmap(QPixmap(QString::fromUtf8(":/Icons/oxygen/encrypted.png")));
|
|
||||||
tracker_url->setText(h.current_tracker());
|
|
||||||
connect(this, SIGNAL(trackerLoginCancelled(QPair<QTorrentHandle,QString>)), parent, SLOT(addUnauthenticatedTracker(QPair<QTorrentHandle,QString>)));
|
|
||||||
show();
|
|
||||||
}
|
|
||||||
|
|
||||||
~trackerLogin() {}
|
|
||||||
|
|
||||||
signals:
|
signals:
|
||||||
void trackerLoginCancelled(QPair<QTorrentHandle,QString> tracker);
|
void trackerLoginCancelled(QPair<QTorrentHandle,QString> tracker);
|
||||||
|
|
||||||
public slots:
|
public slots:
|
||||||
void on_loginButton_clicked() {
|
void on_loginButton_clicked();
|
||||||
// login
|
void on_cancelButton_clicked();
|
||||||
h.set_tracker_login(lineUsername->text(), linePasswd->text());
|
|
||||||
close();
|
|
||||||
}
|
|
||||||
|
|
||||||
void on_cancelButton_clicked() {
|
|
||||||
// Emit a signal to GUI to stop asking for authentication
|
|
||||||
emit trackerLoginCancelled(QPair<QTorrentHandle,QString>(h, h.current_tracker()));
|
|
||||||
close();
|
|
||||||
}
|
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
Loading…
x
Reference in New Issue
Block a user