Some work about adaptive color scheme for Web UI (PR #19901)
http://[316:c51a:62a3:8b9::4]/d4708/qBittorrent/src/branch/adaptive-webui
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
405 lines
15 KiB
405 lines
15 KiB
/* |
|
* 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. |
|
* |
|
* Contact : chris@qbittorrent.org |
|
*/ |
|
|
|
#include <QStandardItemModel> |
|
#include <QHeaderView> |
|
#include <QCompleter> |
|
#include <QSettings> |
|
#include <QMessageBox> |
|
#include <QSystemTrayIcon> |
|
#include <QTemporaryFile> |
|
#include <QSystemTrayIcon> |
|
#include <iostream> |
|
#include <QTimer> |
|
#include <QDir> |
|
|
|
#include "searchEngine.h" |
|
#include "bittorrent.h" |
|
#include "downloadThread.h" |
|
#include "misc.h" |
|
#include "SearchListDelegate.h" |
|
|
|
#define SEARCHHISTORY_MAXSIZE 50 |
|
|
|
/*SEARCH ENGINE START*/ |
|
SearchEngine::SearchEngine(bittorrent *BTSession, QSystemTrayIcon *myTrayIcon, bool systrayIntegration) : QWidget(), BTSession(BTSession), myTrayIcon(myTrayIcon), systrayIntegration(systrayIntegration){ |
|
setupUi(this); |
|
downloader = new downloadThread(this); |
|
// new qCompleter to the search pattern |
|
startSearchHistory(); |
|
searchCompleter = new QCompleter(searchHistory, this); |
|
searchCompleter->setCaseSensitivity(Qt::CaseInsensitive); |
|
search_pattern->setCompleter(searchCompleter); |
|
// Add close tab button |
|
closeTab_button = new QPushButton(); |
|
closeTab_button->setIcon(QIcon(QString::fromUtf8(":/Icons/gnome-shutdown.png"))); |
|
closeTab_button->setFlat(true); |
|
connect(closeTab_button, SIGNAL(clicked()), this, SLOT(closeTab_button_clicked())); |
|
tabWidget->setCornerWidget(closeTab_button); |
|
// Boolean initialization |
|
search_stopped = false; |
|
// Creating Search Process |
|
searchProcess = new QProcess(this); |
|
connect(searchProcess, SIGNAL(started()), this, SLOT(searchStarted())); |
|
connect(searchProcess, SIGNAL(readyReadStandardOutput()), this, SLOT(readSearchOutput())); |
|
connect(searchProcess, SIGNAL(finished(int, QProcess::ExitStatus)), this, SLOT(searchFinished(int,QProcess::ExitStatus))); |
|
connect(tabWidget, SIGNAL(currentChanged(int)), this, SLOT(tab_changed(int))); |
|
searchTimeout = new QTimer(this); |
|
searchTimeout->setSingleShot(true); |
|
connect(searchTimeout, SIGNAL(timeout()), this, SLOT(on_stop_search_button_clicked())); |
|
// Check last enabled search engines |
|
loadEngineSettings(); |
|
// Update nova.py search plugin if necessary |
|
updateNova(); |
|
} |
|
|
|
SearchEngine::~SearchEngine(){ |
|
qDebug("Search destruction"); |
|
// save the searchHistory for later uses |
|
saveSearchHistory(); |
|
searchProcess->kill(); |
|
searchProcess->waitForFinished(); |
|
delete searchTimeout; |
|
delete searchProcess; |
|
delete searchCompleter; |
|
delete downloader; |
|
} |
|
|
|
void SearchEngine::tab_changed(int t) |
|
{//when we switch from a tab that is not empty to another that is empty the download button |
|
//doesn't have to be available |
|
if(t>-1) |
|
{//-1 = no more tab |
|
if(all_tab.at(tabWidget->currentIndex())->getCurrentSearchListModel()->rowCount()) { |
|
download_button->setEnabled(true); |
|
} else { |
|
download_button->setEnabled(false); |
|
} |
|
} |
|
} |
|
|
|
void SearchEngine::on_enginesButton_clicked() { |
|
engineSelectDlg *dlg = new engineSelectDlg(this); |
|
connect(dlg, SIGNAL(enginesChanged()), this, SLOT(loadEngineSettings())); |
|
} |
|
|
|
// get the last searchs from a QSettings to a QStringList |
|
void SearchEngine::startSearchHistory(){ |
|
QSettings settings("qBittorrent", "qBittorrent"); |
|
settings.beginGroup("Search"); |
|
searchHistory = settings.value("searchHistory",-1).toStringList(); |
|
settings.endGroup(); |
|
} |
|
|
|
void SearchEngine::loadEngineSettings() { |
|
qDebug("Loading engine settings"); |
|
enabled_engines.clear(); |
|
QSettings settings(QString::fromUtf8("qBittorrent"), QString::fromUtf8("qBittorrent")); |
|
QStringList known_engines = settings.value(QString::fromUtf8("SearchEngines/knownEngines"), QStringList()).toStringList(); |
|
QVariantList known_enginesEnabled = settings.value(QString::fromUtf8("SearchEngines/knownEnginesEnabled"), QList<QVariant>()).toList(); |
|
QString engine; |
|
unsigned int i = 0; |
|
foreach(engine, known_engines) { |
|
if(known_enginesEnabled.at(i).toBool()) |
|
enabled_engines << engine; |
|
++i; |
|
} |
|
if(enabled_engines.empty()) |
|
enabled_engines << "all"; |
|
qDebug("Engine settings loaded"); |
|
} |
|
|
|
// Save the history list into the QSettings for the next session |
|
void SearchEngine::saveSearchHistory() |
|
{ |
|
QSettings settings("qBittorrent", "qBittorrent"); |
|
settings.beginGroup("Search"); |
|
settings.setValue("searchHistory",searchHistory); |
|
settings.endGroup(); |
|
} |
|
|
|
// Function called when we click on search button |
|
void SearchEngine::on_search_button_clicked(){ |
|
if(searchProcess->state() != QProcess::NotRunning){ |
|
searchProcess->kill(); |
|
searchProcess->waitForFinished(); |
|
} |
|
if(searchTimeout->isActive()) { |
|
searchTimeout->stop(); |
|
} |
|
QString pattern = search_pattern->text().trimmed(); |
|
// No search pattern entered |
|
if(pattern.isEmpty()){ |
|
QMessageBox::critical(0, tr("Empty search pattern"), tr("Please type a search pattern first")); |
|
return; |
|
} |
|
// Tab Addition |
|
currentSearchTab=new SearchTab(this); |
|
all_tab.append(currentSearchTab); |
|
tabWidget->addTab(currentSearchTab, pattern); |
|
tabWidget->setCurrentWidget(currentSearchTab); |
|
closeTab_button->setEnabled(true); |
|
// if the pattern is not in the pattern |
|
if(searchHistory.indexOf(pattern) == -1){ |
|
//update the searchHistory list |
|
searchHistory.append(pattern); |
|
// verify the max size of the history |
|
if(searchHistory.size() > SEARCHHISTORY_MAXSIZE) |
|
searchHistory = searchHistory.mid(searchHistory.size()/2,searchHistory.size()/2); |
|
searchCompleter = new QCompleter(searchHistory, this); |
|
searchCompleter->setCaseSensitivity(Qt::CaseInsensitive); |
|
search_pattern->setCompleter(searchCompleter); |
|
} |
|
|
|
|
|
// Getting checked search engines |
|
Q_ASSERT(!enabled_engines.empty()); |
|
QStringList params; |
|
QStringList engineNames; |
|
search_stopped = false; |
|
params << misc::qBittorrentPath()+"search_engine"+QDir::separator()+"nova2.py"; |
|
params << enabled_engines.join(","); |
|
params << pattern.split(" "); |
|
// Update SearchEngine widgets |
|
no_search_results = true; |
|
nb_search_results = 0; |
|
search_result_line_truncated.clear(); |
|
//on change le texte du label courrant |
|
currentSearchTab->getCurrentLabel()->setText(tr("Results")+" <i>(0)</i>:"); |
|
// Launch search |
|
searchProcess->start("python", params, QIODevice::ReadOnly); |
|
searchTimeout->start(180000); // 3min |
|
} |
|
|
|
void SearchEngine::searchStarted(){ |
|
// Update SearchEngine widgets |
|
search_status->setText(tr("Searching...")); |
|
search_status->repaint(); |
|
stop_search_button->setEnabled(true); |
|
stop_search_button->repaint(); |
|
// clear results window ... not needed since we got Tabbed |
|
//SearchListModel->removeRows(0, SearchListModel->rowCount()); |
|
// Clear previous results urls too |
|
//searchResultsUrls.clear(); |
|
} |
|
|
|
// Download the given item from search results list |
|
void SearchEngine::downloadSelectedItem(const QModelIndex& index){ |
|
int row = index.row(); |
|
// Get Item url |
|
QString url = searchResultsUrls.value(all_tab.at(tabWidget->currentIndex())->getCurrentSearchListModel()->data(all_tab.at(tabWidget->currentIndex())->getCurrentSearchListModel()->index(row, NAME)).toString()); |
|
// Download from url |
|
BTSession->downloadFromUrl(url); |
|
// Set item color to RED |
|
all_tab.at(tabWidget->currentIndex())->setRowColor(row, "red"); |
|
} |
|
|
|
// search Qprocess return output as soon as it gets new |
|
// stuff to read. We split it into lines and add each |
|
// line to search results calling appendSearchResult(). |
|
void SearchEngine::readSearchOutput(){ |
|
QByteArray output = searchProcess->readAllStandardOutput(); |
|
output.replace("\r", ""); |
|
QList<QByteArray> lines_list = output.split('\n'); |
|
QByteArray line; |
|
if(!search_result_line_truncated.isEmpty()){ |
|
QByteArray end_of_line = lines_list.takeFirst(); |
|
lines_list.prepend(search_result_line_truncated+end_of_line); |
|
} |
|
search_result_line_truncated = lines_list.takeLast().trimmed(); |
|
foreach(line, lines_list){ |
|
appendSearchResult(QString(line)); |
|
} |
|
currentSearchTab->getCurrentLabel()->setText(tr("Results")+QString::fromUtf8(" <i>(")+misc::toQString(nb_search_results)+QString::fromUtf8(")</i>:")); |
|
} |
|
|
|
// Update nova.py search plugin if necessary |
|
void SearchEngine::updateNova() { |
|
qDebug("Updating nova"); |
|
// create search_engine directory if necessary |
|
QDir search_dir(misc::qBittorrentPath()+"search_engine"); |
|
if(!search_dir.exists()){ |
|
search_dir.mkdir(misc::qBittorrentPath()+"search_engine"); |
|
} |
|
QFile package_file(search_dir.path()+QDir::separator()+"__init__.py"); |
|
package_file.open(QIODevice::WriteOnly | QIODevice::Text); |
|
package_file.close(); |
|
if(!search_dir.exists("engines")){ |
|
search_dir.mkdir("engines"); |
|
} |
|
QFile package_file2(search_dir.path()+QDir::separator()+"engines"+QDir::separator()+"__init__.py"); |
|
package_file2.open(QIODevice::WriteOnly | QIODevice::Text); |
|
package_file2.close(); |
|
// Copy search plugin files (if necessary) |
|
QString filePath = misc::qBittorrentPath()+"search_engine"+QDir::separator()+"nova2.py"; |
|
if(misc::getPluginVersion(":/search_engine/nova2.py") > misc::getPluginVersion(filePath)) { |
|
if(QFile::exists(filePath)) |
|
QFile::remove(filePath); |
|
QFile::copy(":/search_engine/nova2.py", filePath); |
|
} |
|
// Set permissions |
|
QFile::Permissions perm=QFile::ReadOwner | QFile::WriteOwner | QFile::ExeOwner | QFile::ReadUser | QFile::WriteUser | QFile::ExeUser | QFile::ReadGroup | QFile::ReadGroup; |
|
QFile(misc::qBittorrentPath()+"search_engine"+QDir::separator()+"nova2.py").setPermissions(perm); |
|
filePath = misc::qBittorrentPath()+"search_engine"+QDir::separator()+"novaprinter.py"; |
|
if(misc::getPluginVersion(":/search_engine/novaprinter.py") > misc::getPluginVersion(filePath)) { |
|
if(QFile::exists(filePath)){ |
|
QFile::remove(filePath); |
|
} |
|
QFile::copy(":/search_engine/novaprinter.py", filePath); |
|
} |
|
QString destDir = misc::qBittorrentPath()+"search_engine"+QDir::separator()+"engines"+QDir::separator(); |
|
QDir shipped_subDir(":/search_engine/engines/"); |
|
QStringList files = shipped_subDir.entryList(); |
|
QString file; |
|
foreach(file, files){ |
|
QString shipped_file = shipped_subDir.path()+"/"+file; |
|
// Copy python classes |
|
if(file.endsWith(".py")) { |
|
if(misc::getPluginVersion(shipped_file) > misc::getPluginVersion(destDir+file) ) { |
|
qDebug("shippped %s is more recent then local plugin, updating", file.toUtf8().data()); |
|
if(QFile::exists(destDir+file)) { |
|
qDebug("Removing old %s", (destDir+file).toUtf8().data()); |
|
QFile::remove(destDir+file); |
|
} |
|
qDebug("%s copied to %s", shipped_file.toUtf8().data(), (destDir+file).toUtf8().data()); |
|
QFile::copy(shipped_file, destDir+file); |
|
} |
|
} else { |
|
// Copy icons |
|
if(file.endsWith(".png")) { |
|
if(!QFile::exists(destDir+file)) { |
|
QFile::copy(shipped_file, destDir+file); |
|
} |
|
} |
|
} |
|
} |
|
} |
|
|
|
// Slot called when search is Finished |
|
// Search can be finished for 3 reasons : |
|
// Error | Stopped by user | Finished normally |
|
void SearchEngine::searchFinished(int exitcode,QProcess::ExitStatus){ |
|
QSettings settings("qBittorrent", "qBittorrent"); |
|
bool useNotificationBalloons = settings.value("Preferences/General/NotificationBaloons", true).toBool(); |
|
if(systrayIntegration && useNotificationBalloons) { |
|
myTrayIcon->showMessage(tr("Search Engine"), tr("Search has finished"), QSystemTrayIcon::Information, TIME_TRAY_BALLOON); |
|
} |
|
if(exitcode){ |
|
search_status->setText(tr("An error occured during search...")); |
|
}else{ |
|
if(search_stopped){ |
|
search_status->setText(tr("Search aborted")); |
|
}else{ |
|
if(no_search_results){ |
|
search_status->setText(tr("Search returned no results")); |
|
}else{ |
|
search_status->setText(tr("Search has finished")); |
|
} |
|
} |
|
} |
|
if(currentSearchTab) |
|
currentSearchTab->getCurrentLabel()->setText(tr("Results", "i.e: Search results")+QString::fromUtf8(" <i>(")+misc::toQString(nb_search_results)+QString::fromUtf8(")</i>:")); |
|
search_button->setEnabled(true); |
|
stop_search_button->setEnabled(false); |
|
} |
|
|
|
// SLOT to append one line to search results list |
|
// Line is in the following form : |
|
// file url | file name | file size | nb seeds | nb leechers | Search engine url |
|
void SearchEngine::appendSearchResult(QString line){ |
|
QStringList parts = line.split("|"); |
|
if(parts.size() != 6){ |
|
return; |
|
} |
|
QString url = parts.takeFirst(); |
|
QString filename = parts.first(); |
|
// XXX: Two results can't have the same name (right?) |
|
if(searchResultsUrls.contains(filename)){ |
|
return; |
|
} |
|
// Add item to search result list |
|
int row = currentSearchTab->getCurrentSearchListModel()->rowCount(); |
|
currentSearchTab->getCurrentSearchListModel()->insertRow(row); |
|
for(int i=0; i<5; ++i){ |
|
if(parts.at(i).toFloat() == -1 && i != SIZE) |
|
currentSearchTab->getCurrentSearchListModel()->setData(currentSearchTab->getCurrentSearchListModel()->index(row, i), tr("Unknown")); |
|
else |
|
currentSearchTab->getCurrentSearchListModel()->setData(currentSearchTab->getCurrentSearchListModel()->index(row, i), QVariant(parts.at(i))); |
|
} |
|
// Add url to searchResultsUrls associative array |
|
searchResultsUrls.insert(filename, url); |
|
no_search_results = false; |
|
++nb_search_results; |
|
// Enable clear & download buttons |
|
download_button->setEnabled(true); |
|
} |
|
|
|
// Stop search while it is working in background |
|
void SearchEngine::on_stop_search_button_clicked(){ |
|
// Kill process |
|
searchProcess->terminate(); |
|
search_stopped = true; |
|
searchTimeout->stop(); |
|
} |
|
|
|
// Clear search results list |
|
void SearchEngine::closeTab_button_clicked(){ |
|
if(all_tab.size()) { |
|
qDebug("currentTab rank: %d", tabWidget->currentIndex()); |
|
qDebug("currentSearchTab rank: %d", tabWidget->indexOf(currentSearchTab)); |
|
if(tabWidget->currentIndex() == tabWidget->indexOf(currentSearchTab)) { |
|
qDebug("Deleted current search Tab"); |
|
if(searchProcess->state() != QProcess::NotRunning){ |
|
searchProcess->terminate(); |
|
} |
|
if(searchTimeout->isActive()) { |
|
searchTimeout->stop(); |
|
} |
|
search_stopped = true; |
|
currentSearchTab = 0; |
|
} |
|
delete all_tab.takeAt(tabWidget->currentIndex()); |
|
if(!all_tab.size()) { |
|
closeTab_button->setEnabled(false); |
|
download_button->setEnabled(false); |
|
} |
|
} |
|
} |
|
|
|
void SearchEngine::on_clearPatternButton_clicked() { |
|
search_pattern->clear(); |
|
search_pattern->setFocus(); |
|
} |
|
|
|
// Download selected items in search results list |
|
void SearchEngine::on_download_button_clicked(){ |
|
//QModelIndexList selectedIndexes = currentSearchTab->getCurrentTreeView()->selectionModel()->selectedIndexes(); |
|
QModelIndexList selectedIndexes = all_tab.at(tabWidget->currentIndex())->getCurrentTreeView()->selectionModel()->selectedIndexes(); |
|
QModelIndex index; |
|
foreach(index, selectedIndexes){ |
|
if(index.column() == NAME){ |
|
// Get Item url |
|
QString url = searchResultsUrls.value(index.data().toString()); |
|
BTSession->downloadFromUrl(url); |
|
all_tab.at(tabWidget->currentIndex())->setRowColor(index.row(), "red"); |
|
} |
|
} |
|
}
|
|
|