1
mirror of https://invent.kde.org/network/falkon.git synced 2024-11-14 19:12:11 +01:00
falkonOfficial/src/lib/navigation/websearchbar.cpp

398 lines
12 KiB
C++
Raw Normal View History

2011-03-03 18:29:20 +01:00
/* ============================================================
* QupZilla - WebKit based browser
* Copyright (C) 2010-2013 David Rosca <nowrep@gmail.com>
2011-03-03 18:29:20 +01:00
*
* 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 3 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, see <http://www.gnu.org/licenses/>.
* ============================================================ */
2011-03-02 16:57:41 +01:00
#include "websearchbar.h"
#include "qupzilla.h"
#include "mainapplication.h"
#include "tabbedwebview.h"
#include "webpage.h"
#include "settings.h"
#include "qzsettings.h"
#include "tabwidget.h"
2011-03-02 16:57:41 +01:00
#include "clickablelabel.h"
#include "buttonwithmenu.h"
#include "searchenginesmanager.h"
#include "searchenginesdialog.h"
#include "networkmanager.h"
2011-03-02 16:57:41 +01:00
#include <QMimeData>
#include <QAbstractItemView>
#include <QCompleter>
#include <QStringListModel>
#include <QMenu>
#include <QTimer>
#include <QWebFrame>
#include <QClipboard>
2012-08-31 15:19:07 +02:00
#include <QContextMenuEvent>
WebSearchBar_Button::WebSearchBar_Button(QWidget* parent)
: ClickableLabel(parent)
{
setObjectName("websearchbar-searchbutton");
setCursor(QCursor(Qt::PointingHandCursor));
setFocusPolicy(Qt::ClickFocus);
}
void WebSearchBar_Button::contextMenuEvent(QContextMenuEvent* event)
{
event->accept();
}
WebSearchBar::WebSearchBar(QupZilla* mainClass)
: LineEdit(mainClass)
, p_QupZilla(mainClass)
2012-01-24 15:39:20 +01:00
, m_pasteAndGoAction(0)
, m_clearAction(0)
, m_reloadingEngines(false)
2011-03-02 16:57:41 +01:00
{
setObjectName("websearchbar");
setDragEnabled(true);
m_buttonSearch = new WebSearchBar_Button(this);
2011-03-02 16:57:41 +01:00
m_boxSearchType = new ButtonWithMenu(this);
m_boxSearchType->setObjectName("websearchbar-searchprovider-comobobox");
// RTL Support
// If we don't add 'm_boxSearchType' by following code, then we should use suitable padding-left value
// but then, when typing RTL text the layout dynamically changed and within RTL layout direction
// padding-left is equivalent to padding-right and vice versa, and because style sheet is
// not changed dynamically this create padding problems.
addWidget(m_boxSearchType, LineEdit::LeftSide);
2011-03-02 16:57:41 +01:00
addWidget(m_buttonSearch, LineEdit::RightSide);
connect(m_buttonSearch, SIGNAL(clicked(QPoint)), this, SLOT(search()));
connect(m_buttonSearch, SIGNAL(middleClicked(QPoint)), this, SLOT(searchInNewTab()));
connect(m_boxSearchType, SIGNAL(activeItemChanged(ButtonWithMenu::Item)), this, SLOT(searchChanged(ButtonWithMenu::Item)));
2011-03-02 16:57:41 +01:00
setWidgetSpacing(0);
m_searchManager = mApp->searchEnginesManager();
connect(m_boxSearchType->menu(), SIGNAL(aboutToShow()), this, SLOT(aboutToShowMenu()));
m_completer = new QCompleter(this);
m_completer->setCompletionMode(QCompleter::UnfilteredPopupCompletion);
m_completerModel = new QStringListModel(this);
m_completer->setModel(m_completerModel);
m_completer->popup()->setMinimumHeight(90);
setCompleter(m_completer);
m_openSearchEngine = new OpenSearchEngine(this);
m_openSearchEngine->setNetworkAccessManager(mApp->networkManager());
connect(m_openSearchEngine, SIGNAL(suggestions(QStringList)), this, SLOT(addSuggestions(QStringList)));
connect(this, SIGNAL(textEdited(QString)), m_openSearchEngine, SLOT(requestSuggestions(QString)));
QTimer::singleShot(0, this, SLOT(setupEngines()));
}
void WebSearchBar::aboutToShowMenu()
{
QMenu* menu = m_boxSearchType->menu();
menu->addSeparator();
completeMenuWithAvailableEngines(menu);
menu->addSeparator();
menu->addAction(QIcon(":icons/menu/gear.png"), tr("Manage Search Engines"), this, SLOT(openSearchEnginesDialog()));
}
void WebSearchBar::addSuggestions(const QStringList &list)
{
if (qzSettings->showSearchSuggestions) {
QStringList list_ = list.mid(0, 6);
m_completerModel->setStringList(list_);
}
2011-03-02 16:57:41 +01:00
}
void WebSearchBar::openSearchEnginesDialog()
2011-03-02 16:57:41 +01:00
{
if (m_searchDialog) {
m_searchDialog.data()->raise();
m_searchDialog.data()->activateWindow();
return;
}
m_searchDialog = new SearchEnginesDialog(this);
m_searchDialog.data()->show();
}
void WebSearchBar::enableSearchSuggestions(bool enable)
{
Settings settings;
settings.beginGroup("SearchEngines");
settings.setValue("showSuggestions", enable);
settings.endGroup();
qzSettings->showSearchSuggestions = enable;
m_completerModel->setStringList(QStringList());
}
void WebSearchBar::setupEngines()
{
disconnect(m_searchManager, SIGNAL(enginesChanged()), this, SLOT(setupEngines()));
m_reloadingEngines = true;
QString activeEngine = m_searchManager->startingEngineName();
if (m_boxSearchType->allItems().count() != 0) {
activeEngine = m_activeEngine.name;
}
m_boxSearchType->clearItems();
foreach (const SearchEngine &en, m_searchManager->allEngines()) {
ButtonWithMenu::Item item;
item.icon = en.icon;
item.text = en.name;
QVariant v;
v.setValue<SearchEngine>(en);
item.userData = v;
m_boxSearchType->addItem(item);
if (item.text == activeEngine) {
m_boxSearchType->setCurrentItem(item, false);
}
}
searchChanged(m_boxSearchType->currentItem());
connect(m_searchManager, SIGNAL(enginesChanged()), this, SLOT(setupEngines()));
m_reloadingEngines = false;
2011-03-02 16:57:41 +01:00
}
void WebSearchBar::searchChanged(const ButtonWithMenu::Item &item)
2011-03-02 16:57:41 +01:00
{
selectAll();
setFocus();
setPlaceholderText(item.text);
m_completerModel->setStringList(QStringList());
m_activeEngine = item.userData.value<SearchEngine>();
m_openSearchEngine->setSuggestionsUrl(m_activeEngine.suggestionsUrl);
m_openSearchEngine->setSuggestionsParameters(m_activeEngine.suggestionsParameters);
m_searchManager->setActiveEngine(m_activeEngine);
if (qzSettings->searchOnEngineChange && !m_reloadingEngines && !text().isEmpty()) {
search();
}
2011-03-02 16:57:41 +01:00
}
void WebSearchBar::instantSearchChanged(bool enable)
{
Settings settings;
settings.beginGroup("SearchEngines");
settings.setValue("SearchOnEngineChange", enable);
settings.endGroup();
qzSettings->searchOnEngineChange = enable;
}
2011-03-02 16:57:41 +01:00
void WebSearchBar::search()
{
p_QupZilla->weView()->setFocus();
SearchEnginesManager::SearchResult res = m_searchManager->searchResult(m_activeEngine, text());
p_QupZilla->weView()->load(res.request, res.operation, res.data);
}
void WebSearchBar::searchInNewTab()
{
2011-03-02 16:57:41 +01:00
p_QupZilla->weView()->setFocus();
int index = p_QupZilla->tabWidget()->addView(QUrl());
SearchEnginesManager::SearchResult res = m_searchManager->searchResult(m_activeEngine, text());
p_QupZilla->weView(index)->load(res.request, res.operation, res.data);
2011-03-02 16:57:41 +01:00
}
void WebSearchBar::completeMenuWithAvailableEngines(QMenu* menu)
{
WebView* view = p_QupZilla->weView();
QWebFrame* frame = view->page()->mainFrame();
QWebElementCollection elements = frame->documentElement().findAll(QLatin1String("link[rel=search]"));
foreach (const QWebElement &element, elements) {
if (element.attribute("type") != QLatin1String("application/opensearchdescription+xml")) {
continue;
}
QUrl url = view->url().resolved(QUrl::fromEncoded(element.attribute("href").toUtf8()));
QString title = element.attribute("title");
if (url.isEmpty()) {
continue;
}
if (title.isEmpty()) {
title = view->title();
}
menu->addAction(view->icon(), tr("Add %1 ...").arg(title), this, SLOT(addEngineFromAction()))->setData(url);
}
}
void WebSearchBar::addEngineFromAction()
{
if (QAction* action = qobject_cast<QAction*>(sender())) {
m_searchManager->addEngine(action->data().toUrl());
}
}
2012-01-24 15:39:20 +01:00
void WebSearchBar::pasteAndGo()
{
clear();
paste();
search();
}
void WebSearchBar::contextMenuEvent(QContextMenuEvent* event)
{
Q_UNUSED(event)
if (!m_pasteAndGoAction) {
m_pasteAndGoAction = new QAction(QIcon::fromTheme("edit-paste"), tr("Paste And &Search"), this);
m_pasteAndGoAction->setShortcut(QKeySequence("Ctrl+Shift+V"));
2012-01-24 15:39:20 +01:00
connect(m_pasteAndGoAction, SIGNAL(triggered()), this, SLOT(pasteAndGo()));
}
if (!m_clearAction) {
m_clearAction = new QAction(QIcon::fromTheme("edit-clear"), tr("Clear All"), this);
connect(m_clearAction, SIGNAL(triggered()), this, SLOT(clear()));
}
2012-01-24 15:39:20 +01:00
QMenu* tempMenu = createStandardContextMenu();
QMenu menu(this);
2012-01-24 15:39:20 +01:00
int i = 0;
foreach (QAction* act, tempMenu->actions()) {
menu.addAction(act);
2012-01-24 15:39:20 +01:00
switch (i) {
case 0:
act->setIcon(QIcon::fromTheme("edit-undo"));
break;
case 1:
act->setIcon(QIcon::fromTheme("edit-redo"));
break;
case 3:
act->setIcon(QIcon::fromTheme("edit-cut"));
break;
case 4:
act->setIcon(QIcon::fromTheme("edit-copy"));
break;
case 5:
act->setIcon(QIcon::fromTheme("edit-paste"));
menu.addAction(act);
menu.addAction(m_pasteAndGoAction);
break;
case 6:
act->setIcon(QIcon::fromTheme("edit-delete"));
menu.addAction(act);
menu.addAction(m_clearAction);
break;
case 8:
act->setIcon(QIcon::fromTheme("edit-select-all"));
break;
2012-01-24 15:39:20 +01:00
}
++i;
}
menu.addSeparator();
QAction* act = menu.addAction(tr("Show suggestions"));
act->setCheckable(true);
act->setChecked(qzSettings->showSearchSuggestions);
connect(act, SIGNAL(triggered(bool)), this, SLOT(enableSearchSuggestions(bool)));
QAction* instantSearch = menu.addAction(tr("Search when engine changed"));
instantSearch->setCheckable(true);
instantSearch->setChecked(qzSettings->searchOnEngineChange);
connect(instantSearch, SIGNAL(triggered(bool)), this, SLOT(instantSearchChanged(bool)));
2012-01-24 15:39:20 +01:00
m_pasteAndGoAction->setEnabled(!QApplication::clipboard()->text().isEmpty());
// Prevent choosing first option with double rightclick
QPoint pos = event->globalPos();
2012-01-24 15:39:20 +01:00
QPoint p(pos.x(), pos.y() + 1);
menu.exec(p);
tempMenu->deleteLater();
2012-01-24 15:39:20 +01:00
}
2011-03-17 17:03:04 +01:00
void WebSearchBar::focusOutEvent(QFocusEvent* e)
2011-03-02 16:57:41 +01:00
{
if (text().isEmpty()) {
QString search = m_boxSearchType->currentItem().text;
2011-03-02 16:57:41 +01:00
setPlaceholderText(search);
}
LineEdit::focusOutEvent(e);
2011-03-02 16:57:41 +01:00
}
void WebSearchBar::dropEvent(QDropEvent* event)
{
if (event->mimeData()->hasText()) {
QString dropText = event->mimeData()->text();
setText(dropText);
search();
QFocusEvent event(QFocusEvent::FocusOut);
LineEdit::focusOutEvent(&event);
return;
}
LineEdit::dropEvent(event);
}
void WebSearchBar::keyPressEvent(QKeyEvent* event)
{
switch (event->key()) {
case Qt::Key_V:
if (event->modifiers() == (Qt::ControlModifier | Qt::ShiftModifier)) {
pasteAndGo();
event->accept();
return;
}
break;
case Qt::Key_Return:
case Qt::Key_Enter:
2012-01-21 19:26:47 +01:00
if (event->modifiers() == Qt::AltModifier) {
searchInNewTab();
}
else {
search();
}
break;
case Qt::Key_Up:
if (event->modifiers() == Qt::ControlModifier) {
m_boxSearchType->selectPreviousItem();
}
break;
case Qt::Key_Down:
if (event->modifiers() == Qt::ControlModifier) {
m_boxSearchType->selectNextItem();
}
break;
default:
2012-01-31 11:35:03 +01:00
break;
}
2012-01-31 11:35:03 +01:00
LineEdit::keyPressEvent(event);
}