Reorder code to match UI
[qBittorrent.git] / src / gui / search / searchwidget.cpp
blob1128a561e9cec748e66ea0b2c611d7fccf095c62
1 /*
2 * Bittorrent Client using Qt and libtorrent.
3 * Copyright (C) 2020, Will Da Silva <will@willdasilva.xyz>
4 * Copyright (C) 2015, 2018 Vladimir Golovnev <glassez@yandex.ru>
5 * Copyright (C) 2006 Christophe Dumez <chris@qbittorrent.org>
7 * This program is free software; you can redistribute it and/or
8 * modify it under the terms of the GNU General Public License
9 * as published by the Free Software Foundation; either version 2
10 * of the License, or (at your option) any later version.
12 * This program is distributed in the hope that it will be useful,
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15 * GNU General Public License for more details.
17 * You should have received a copy of the GNU General Public License
18 * along with this program; if not, write to the Free Software
19 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
21 * In addition, as a special exception, the copyright holders give permission to
22 * link this program with the OpenSSL project's "OpenSSL" library (or with
23 * modified versions of it that use the same license as the "OpenSSL" library),
24 * and distribute the linked executables. You must obey the GNU General Public
25 * License in all respects for all of the code used other than "OpenSSL". If you
26 * modify file(s), you may extend this exception to your version of the file(s),
27 * but you are not obligated to do so. If you do not wish to do so, delete this
28 * exception statement from your version.
31 #include "searchwidget.h"
33 #include <QtSystemDetection>
35 #include <utility>
37 #ifdef Q_OS_WIN
38 #include <cstdlib>
39 #endif
41 #include <QDebug>
42 #include <QEvent>
43 #include <QMessageBox>
44 #include <QMenu>
45 #include <QMouseEvent>
46 #include <QObject>
47 #include <QRegularExpression>
48 #include <QShortcut>
49 #include <QVector>
51 #include "base/global.h"
52 #include "base/search/searchhandler.h"
53 #include "base/search/searchpluginmanager.h"
54 #include "base/utils/foreignapps.h"
55 #include "gui/desktopintegration.h"
56 #include "gui/interfaces/iguiapplication.h"
57 #include "gui/mainwindow.h"
58 #include "gui/uithememanager.h"
59 #include "pluginselectdialog.h"
60 #include "searchjobwidget.h"
61 #include "ui_searchwidget.h"
63 #define SEARCHHISTORY_MAXSIZE 50
64 #define URL_COLUMN 5
66 namespace
68 QString statusIconName(const SearchJobWidget::Status st)
70 switch (st)
72 case SearchJobWidget::Status::Ongoing:
73 return u"queued"_s;
74 case SearchJobWidget::Status::Finished:
75 return u"task-complete"_s;
76 case SearchJobWidget::Status::Aborted:
77 return u"task-reject"_s;
78 case SearchJobWidget::Status::Error:
79 case SearchJobWidget::Status::NoResults:
80 return u"dialog-warning"_s;
81 default:
82 return {};
87 SearchWidget::SearchWidget(IGUIApplication *app, MainWindow *mainWindow)
88 : GUIApplicationComponent(app, mainWindow)
89 , m_ui {new Ui::SearchWidget()}
90 , m_mainWindow {mainWindow}
92 m_ui->setupUi(this);
93 m_ui->tabWidget->tabBar()->installEventFilter(this);
95 const QString searchPatternHint = u"<html><head/><body><p>"
96 + tr("A phrase to search for.") + u"<br>"
97 + tr("Spaces in a search term may be protected by double quotes.")
98 + u"</p><p>"
99 + tr("Example:", "Search phrase example")
100 + u"<br>"
101 + tr("<b>foo bar</b>: search for <b>foo</b> and <b>bar</b>",
102 "Search phrase example, illustrates quotes usage, a pair of "
103 "space delimited words, individual words are highlighted")
104 + u"<br>"
105 + tr("<b>&quot;foo bar&quot;</b>: search for <b>foo bar</b>",
106 "Search phrase example, illustrates quotes usage, double quoted"
107 "pair of space delimited words, the whole pair is highlighted")
108 + u"</p></body></html>";
109 m_ui->lineEditSearchPattern->setToolTip(searchPatternHint);
111 #ifndef Q_OS_MACOS
112 // Icons
113 m_ui->searchButton->setIcon(UIThemeManager::instance()->getIcon(u"edit-find"_s));
114 m_ui->pluginsButton->setIcon(UIThemeManager::instance()->getIcon(u"plugins"_s, u"preferences-system-network"_s));
115 #else
116 // On macOS the icons overlap the text otherwise
117 QSize iconSize = m_ui->tabWidget->iconSize();
118 iconSize.setWidth(iconSize.width() + 16);
119 m_ui->tabWidget->setIconSize(iconSize);
120 #endif
121 connect(m_ui->tabWidget, &QTabWidget::tabCloseRequested, this, &SearchWidget::closeTab);
122 connect(m_ui->tabWidget, &QTabWidget::currentChanged, this, &SearchWidget::tabChanged);
124 const auto *searchManager = SearchPluginManager::instance();
125 const auto onPluginChanged = [this]()
127 fillPluginComboBox();
128 fillCatCombobox();
129 selectActivePage();
131 connect(searchManager, &SearchPluginManager::pluginInstalled, this, onPluginChanged);
132 connect(searchManager, &SearchPluginManager::pluginUninstalled, this, onPluginChanged);
133 connect(searchManager, &SearchPluginManager::pluginUpdated, this, onPluginChanged);
134 connect(searchManager, &SearchPluginManager::pluginEnabled, this, onPluginChanged);
136 // Fill in category combobox
137 onPluginChanged();
139 connect(m_ui->lineEditSearchPattern, &LineEdit::returnPressed, m_ui->searchButton, &QPushButton::click);
140 connect(m_ui->lineEditSearchPattern, &LineEdit::textEdited, this, &SearchWidget::searchTextEdited);
141 connect(m_ui->selectPlugin, qOverload<int>(&QComboBox::currentIndexChanged)
142 , this, &SearchWidget::selectMultipleBox);
143 connect(m_ui->selectPlugin, qOverload<int>(&QComboBox::currentIndexChanged)
144 , this, &SearchWidget::fillCatCombobox);
146 const auto *focusSearchHotkey = new QShortcut(QKeySequence::Find, this);
147 connect(focusSearchHotkey, &QShortcut::activated, this, &SearchWidget::toggleFocusBetweenLineEdits);
148 const auto *focusSearchHotkeyAlternative = new QShortcut((Qt::CTRL | Qt::Key_E), this);
149 connect(focusSearchHotkeyAlternative, &QShortcut::activated, this, &SearchWidget::toggleFocusBetweenLineEdits);
152 bool SearchWidget::eventFilter(QObject *object, QEvent *event)
154 if (object == m_ui->tabWidget->tabBar())
156 // Close tabs when middle-clicked
157 if (event->type() != QEvent::MouseButtonRelease)
158 return false;
160 const auto *mouseEvent = static_cast<QMouseEvent *>(event);
161 const int tabIndex = m_ui->tabWidget->tabBar()->tabAt(mouseEvent->pos());
162 if ((mouseEvent->button() == Qt::MiddleButton) && (tabIndex >= 0))
164 closeTab(tabIndex);
165 return true;
167 if (mouseEvent->button() == Qt::RightButton)
169 QMenu *menu = new QMenu(this);
170 menu->setAttribute(Qt::WA_DeleteOnClose);
171 menu->addAction(tr("Close tab"), this, [this, tabIndex]() { closeTab(tabIndex); });
172 menu->addAction(tr("Close all tabs"), this, &SearchWidget::closeAllTabs);
173 menu->popup(QCursor::pos());
174 return true;
176 return false;
178 return QWidget::eventFilter(object, event);
181 void SearchWidget::fillCatCombobox()
183 m_ui->comboCategory->clear();
184 m_ui->comboCategory->addItem(SearchPluginManager::categoryFullName(u"all"_s), u"all"_s);
186 using QStrPair = std::pair<QString, QString>;
187 QVector<QStrPair> tmpList;
188 for (const QString &cat : asConst(SearchPluginManager::instance()->getPluginCategories(selectedPlugin())))
189 tmpList << std::make_pair(SearchPluginManager::categoryFullName(cat), cat);
190 std::sort(tmpList.begin(), tmpList.end(), [](const QStrPair &l, const QStrPair &r) { return (QString::localeAwareCompare(l.first, r.first) < 0); });
192 for (const QStrPair &p : asConst(tmpList))
194 qDebug("Supported category: %s", qUtf8Printable(p.second));
195 m_ui->comboCategory->addItem(p.first, p.second);
198 if (m_ui->comboCategory->count() > 1)
199 m_ui->comboCategory->insertSeparator(1);
202 void SearchWidget::fillPluginComboBox()
204 m_ui->selectPlugin->clear();
205 m_ui->selectPlugin->addItem(tr("Only enabled"), u"enabled"_s);
206 m_ui->selectPlugin->addItem(tr("All plugins"), u"all"_s);
207 m_ui->selectPlugin->addItem(tr("Select..."), u"multi"_s);
209 using QStrPair = std::pair<QString, QString>;
210 QVector<QStrPair> tmpList;
211 for (const QString &name : asConst(SearchPluginManager::instance()->enabledPlugins()))
212 tmpList << std::make_pair(SearchPluginManager::instance()->pluginFullName(name), name);
213 std::sort(tmpList.begin(), tmpList.end(), [](const QStrPair &l, const QStrPair &r) { return (l.first < r.first); } );
215 for (const QStrPair &p : asConst(tmpList))
216 m_ui->selectPlugin->addItem(p.first, p.second);
218 if (m_ui->selectPlugin->count() > 3)
219 m_ui->selectPlugin->insertSeparator(3);
222 QString SearchWidget::selectedCategory() const
224 return m_ui->comboCategory->itemData(m_ui->comboCategory->currentIndex()).toString();
227 QString SearchWidget::selectedPlugin() const
229 return m_ui->selectPlugin->itemData(m_ui->selectPlugin->currentIndex()).toString();
232 void SearchWidget::selectActivePage()
234 if (SearchPluginManager::instance()->allPlugins().isEmpty())
236 m_ui->stackedPages->setCurrentWidget(m_ui->emptyPage);
237 m_ui->lineEditSearchPattern->setEnabled(false);
238 m_ui->comboCategory->setEnabled(false);
239 m_ui->selectPlugin->setEnabled(false);
240 m_ui->searchButton->setEnabled(false);
242 else
244 m_ui->stackedPages->setCurrentWidget(m_ui->searchPage);
245 m_ui->lineEditSearchPattern->setEnabled(true);
246 m_ui->comboCategory->setEnabled(true);
247 m_ui->selectPlugin->setEnabled(true);
248 m_ui->searchButton->setEnabled(true);
252 SearchWidget::~SearchWidget()
254 qDebug("Search destruction");
255 delete m_ui;
258 void SearchWidget::tabChanged(const int index)
260 // when we switch from a tab that is not empty to another that is empty
261 // the download button doesn't have to be available
262 m_currentSearchTab = ((index < 0) ? nullptr : m_allTabs.at(m_ui->tabWidget->currentIndex()));
265 void SearchWidget::selectMultipleBox([[maybe_unused]] const int index)
267 if (selectedPlugin() == u"multi")
268 on_pluginsButton_clicked();
271 void SearchWidget::toggleFocusBetweenLineEdits()
273 if (m_ui->lineEditSearchPattern->hasFocus() && m_currentSearchTab)
275 m_currentSearchTab->lineEditSearchResultsFilter()->setFocus();
276 m_currentSearchTab->lineEditSearchResultsFilter()->selectAll();
278 else
280 m_ui->lineEditSearchPattern->setFocus();
281 m_ui->lineEditSearchPattern->selectAll();
285 void SearchWidget::on_pluginsButton_clicked()
287 auto *dlg = new PluginSelectDialog(SearchPluginManager::instance(), this);
288 dlg->setAttribute(Qt::WA_DeleteOnClose);
289 dlg->show();
292 void SearchWidget::searchTextEdited(const QString &)
294 // Enable search button
295 m_ui->searchButton->setText(tr("Search"));
296 m_isNewQueryString = true;
299 void SearchWidget::giveFocusToSearchInput()
301 m_ui->lineEditSearchPattern->setFocus();
304 // Function called when we click on search button
305 void SearchWidget::on_searchButton_clicked()
307 if (!Utils::ForeignApps::pythonInfo().isValid())
309 app()->desktopIntegration()->showNotification(tr("Search Engine"), tr("Please install Python to use the Search Engine."));
310 return;
313 if (m_activeSearchTab)
315 m_activeSearchTab->cancelSearch();
316 if (!m_isNewQueryString)
318 m_ui->searchButton->setText(tr("Search"));
319 return;
323 m_isNewQueryString = false;
325 const QString pattern = m_ui->lineEditSearchPattern->text().trimmed();
326 // No search pattern entered
327 if (pattern.isEmpty())
329 QMessageBox::critical(this, tr("Empty search pattern"), tr("Please type a search pattern first"));
330 return;
333 const QString plugin = selectedPlugin();
335 QStringList plugins;
336 if (plugin == u"all")
337 plugins = SearchPluginManager::instance()->allPlugins();
338 else if ((plugin == u"enabled") || (plugin == u"multi"))
339 plugins = SearchPluginManager::instance()->enabledPlugins();
340 else
341 plugins << plugin;
343 qDebug("Search with category: %s", qUtf8Printable(selectedCategory()));
345 // Launch search
346 auto *searchHandler = SearchPluginManager::instance()->startSearch(pattern, selectedCategory(), plugins);
348 // Tab Addition
349 auto *newTab = new SearchJobWidget(searchHandler, app(), this);
350 m_allTabs.append(newTab);
352 QString tabName = pattern;
353 tabName.replace(QRegularExpression(u"&{1}"_s), u"&&"_s);
354 m_ui->tabWidget->addTab(newTab, tabName);
355 m_ui->tabWidget->setCurrentWidget(newTab);
357 connect(newTab, &SearchJobWidget::statusChanged, this, [this, newTab]() { tabStatusChanged(newTab); });
359 m_ui->searchButton->setText(tr("Stop"));
360 m_activeSearchTab = newTab;
361 tabStatusChanged(newTab);
364 void SearchWidget::tabStatusChanged(QWidget *tab)
366 const int tabIndex = m_ui->tabWidget->indexOf(tab);
367 m_ui->tabWidget->setTabToolTip(tabIndex, tab->statusTip());
368 m_ui->tabWidget->setTabIcon(tabIndex, UIThemeManager::instance()->getIcon(
369 statusIconName(static_cast<SearchJobWidget *>(tab)->status())));
371 if ((tab == m_activeSearchTab) && (m_activeSearchTab->status() != SearchJobWidget::Status::Ongoing))
373 Q_ASSERT(m_activeSearchTab->status() != SearchJobWidget::Status::Ongoing);
375 if (app()->desktopIntegration()->isNotificationsEnabled() && (m_mainWindow->currentTabWidget() != this))
377 if (m_activeSearchTab->status() == SearchJobWidget::Status::Error)
378 app()->desktopIntegration()->showNotification(tr("Search Engine"), tr("Search has failed"));
379 else
380 app()->desktopIntegration()->showNotification(tr("Search Engine"), tr("Search has finished"));
383 m_activeSearchTab = nullptr;
384 m_ui->searchButton->setText(tr("Search"));
388 void SearchWidget::closeTab(int index)
390 SearchJobWidget *tab = m_allTabs.takeAt(index);
391 if (tab == m_activeSearchTab)
392 m_ui->searchButton->setText(tr("Search"));
394 delete tab;
397 void SearchWidget::closeAllTabs()
399 for (int i = (m_allTabs.size() - 1); i >= 0; --i)
400 closeTab(i);