1
mirror of https://invent.kde.org/network/falkon.git synced 2024-11-11 17:42:10 +01:00
falkonOfficial/src/lib/navigation/locationbar.cpp

768 lines
21 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 "locationbar.h"
#include "qupzilla.h"
#include "tabbedwebview.h"
2011-03-02 16:57:41 +01:00
#include "rssmanager.h"
#include "mainapplication.h"
#include "clickablelabel.h"
#include "webpage.h"
#include "tabwidget.h"
#include "bookmarkicon.h"
#include "progressbar.h"
#include "statusbarmessage.h"
#include "toolbutton.h"
#include "searchenginesmanager.h"
#include "siteicon.h"
#include "goicon.h"
#include "rssicon.h"
#include "downicon.h"
#include "qztools.h"
#include "iconprovider.h"
#include "qzsettings.h"
#include "colors.h"
#include "autofillicon.h"
2011-03-02 16:57:41 +01:00
#include <QMimeData>
#include <QClipboard>
#include <QTimer>
2012-08-31 15:19:07 +02:00
#include <QContextMenuEvent>
#include <QAction>
#include <QMenu>
LocationBar::LocationBar(QupZilla* mainClass)
: LineEdit(mainClass)
, p_QupZilla(mainClass)
, m_webView(0)
2012-01-21 19:26:47 +01:00
, m_pasteAndGoAction(0)
, m_clearAction(0)
, m_rssIconVisible(false)
, m_holdingAlt(false)
, m_loadProgress(0)
, m_progressVisible(false)
, m_forcePaintEvent(false)
, m_inlineCompletionVisible(false)
, m_popupClosed(false)
2011-03-02 16:57:41 +01:00
{
setObjectName("locationbar");
setDragEnabled(true);
2011-03-02 16:57:41 +01:00
m_bookmarkIcon = new BookmarkIcon(this);
m_goIcon = new GoIcon(this);
m_rssIcon = new RssIcon(this);
m_siteIcon = new SiteIcon(p_QupZilla, this);
m_autofillIcon = new AutoFillIcon(this);
DownIcon* down = new DownIcon(this);
2011-03-02 16:57:41 +01:00
// RTL Support
// if we don't add 'm_siteIcon' 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_siteIcon, LineEdit::LeftSide);
addWidget(m_autofillIcon, LineEdit::RightSide);
addWidget(m_bookmarkIcon, LineEdit::RightSide);
2011-03-02 16:57:41 +01:00
addWidget(m_rssIcon, LineEdit::RightSide);
addWidget(m_goIcon, LineEdit::RightSide);
addWidget(down, LineEdit::RightSide);
2011-03-02 16:57:41 +01:00
m_completer.setLocationBar(this);
connect(&m_completer, SIGNAL(showCompletion(QString)), this, SLOT(showCompletion(QString)));
connect(&m_completer, SIGNAL(completionActivated()), this, SLOT(urlEnter()));
connect(&m_completer, SIGNAL(popupClosed()), this, SLOT(completionPopupClosed()));
2011-03-02 16:57:41 +01:00
connect(this, SIGNAL(textEdited(QString)), this, SLOT(textEdit()));
connect(m_goIcon, SIGNAL(clicked(QPoint)), this, SLOT(urlEnter()));
connect(down, SIGNAL(clicked(QPoint)), &m_completer, SLOT(showMostVisited()));
connect(mApp->searchEnginesManager(), SIGNAL(activeEngineChanged()), this, SLOT(updatePlaceHolderText()));
connect(mApp->searchEnginesManager(), SIGNAL(defaultEngineChanged()), this, SLOT(updatePlaceHolderText()));
connect(mApp, SIGNAL(message(Qz::AppMessageType,bool)), SLOT(onMessage(Qz::AppMessageType,bool)));
loadSettings();
2011-03-02 16:57:41 +01:00
clearIcon();
updatePlaceHolderText();
// Hide icons by default
hideGoButton();
m_rssIcon->hide();
m_autofillIcon->hide();
}
void LocationBar::setWebView(TabbedWebView* view)
{
m_webView = view;
m_bookmarkIcon->setWebView(m_webView);
m_rssIcon->setWebView(m_webView);
m_siteIcon->setWebView(m_webView);
m_autofillIcon->setWebView(m_webView);
connect(m_webView, SIGNAL(loadStarted()), SLOT(onLoadStarted()));
connect(m_webView, SIGNAL(loadProgress(int)), SLOT(onLoadProgress(int)));
connect(m_webView, SIGNAL(loadFinished(bool)), SLOT(onLoadFinished()));
}
void LocationBar::setText(const QString &text)
{
LineEdit::setText(text);
m_forcePaintEvent = true;
setCursorPosition(0);
}
void LocationBar::updatePlaceHolderText()
{
QString engineName = qzSettings->searchWithDefaultEngine ?
mApp->searchEnginesManager()->defaultEngine().name :
mApp->searchEnginesManager()->activeEngine().name;
setPlaceholderText(tr("Enter URL address or search on %1").arg(engineName));
2011-03-02 16:57:41 +01:00
}
void LocationBar::showCompletion(const QString &newText)
{
m_inlineCompletionVisible = false;
LineEdit::setText(newText);
// Move cursor to the end
end(false);
}
void LocationBar::completionPopupClosed()
{
m_inlineCompletionVisible = false;
m_popupClosed = true;
}
QUrl LocationBar::createUrl()
{
QUrl urlToLoad;
// Check for Search Engine shortcut
int firstSpacePos = text().indexOf(QLatin1Char(' '));
if (firstSpacePos != -1) {
QString shortcut = text().left(firstSpacePos);
QString searchedString = QUrl::toPercentEncoding(text().mid(firstSpacePos).trimmed());
SearchEngine en = mApp->searchEnginesManager()->engineForShortcut(shortcut);
if (!en.name.isEmpty()) {
urlToLoad = QUrl::fromEncoded(en.url.replace(QLatin1String("%s"), searchedString).toUtf8());
}
}
if (m_inlineCompletionVisible) {
urlToLoad = WebView::guessUrlFromString(text() + m_completer.domainCompletion());
}
if (urlToLoad.isEmpty()) {
QUrl guessedUrl = WebView::guessUrlFromString(text());
if (!guessedUrl.isEmpty()) {
urlToLoad = guessedUrl;
}
else {
urlToLoad = QUrl::fromEncoded(text().toUtf8());
}
}
return urlToLoad;
}
QString LocationBar::convertUrlToText(const QUrl &url) const
{
QString stringUrl = QzTools::urlEncodeQueryString(url);
if (stringUrl == QLatin1String("qupzilla:speeddial") || stringUrl == QLatin1String("about:blank")) {
stringUrl.clear();
}
return stringUrl;
}
void LocationBar::urlEnter()
{
const QUrl &url = createUrl();
const QString &urlString = convertUrlToText(url);
m_completer.closePopup();
m_webView->setFocus();
if (urlString != text()) {
setText(convertUrlToText(url));
}
emit loadUrl(url);
}
2011-03-02 16:57:41 +01:00
void LocationBar::textEdit()
{
if (!text().isEmpty()) {
m_completer.complete(text());
m_inlineCompletionVisible = true;
}
else {
m_completer.closePopup();
}
2011-03-02 16:57:41 +01:00
showGoButton();
}
void LocationBar::showGoButton()
{
m_rssIconVisible = m_rssIcon->isVisible();
m_bookmarkIcon->hide();
2011-03-02 16:57:41 +01:00
m_rssIcon->hide();
m_goIcon->show();
updateTextMargins();
2011-03-02 16:57:41 +01:00
}
void LocationBar::hideGoButton()
{
m_rssIcon->setVisible(m_rssIconVisible);
m_bookmarkIcon->show();
if (!qzSettings->alwaysShowGoIcon) {
m_goIcon->hide();
}
updateTextMargins();
2011-03-02 16:57:41 +01:00
}
void LocationBar::showRSSIcon(bool state)
{
m_rssIcon->setVisible(state);
updateTextMargins();
}
void LocationBar::showUrl(const QUrl &url)
2011-03-02 16:57:41 +01:00
{
if (hasFocus() || url.isEmpty()) {
2011-03-02 16:57:41 +01:00
return;
}
2011-03-02 16:57:41 +01:00
const QString &stringUrl = convertUrlToText(url);
if (stringUrl == text()) {
return;
2011-03-02 16:57:41 +01:00
}
setText(stringUrl);
2011-03-02 16:57:41 +01:00
hideGoButton();
m_bookmarkIcon->checkBookmark(url);
2011-03-02 16:57:41 +01:00
}
void LocationBar::siteIconChanged()
{
QIcon icon_ = m_webView->icon();
2011-03-02 16:57:41 +01:00
if (icon_.isNull()) {
clearIcon();
}
else {
m_siteIcon->setIcon(QIcon(icon_.pixmap(16, 16)));
2011-03-02 16:57:41 +01:00
}
}
void LocationBar::clearIcon()
{
m_siteIcon->setIcon(qIconProvider->emptyWebIcon());
}
2011-03-02 16:57:41 +01:00
void LocationBar::setPrivacy(bool state)
{
m_siteIcon->setProperty("secured", QVariant(state));
m_siteIcon->style()->unpolish(m_siteIcon);
m_siteIcon->style()->polish(m_siteIcon);
setProperty("secured", QVariant(state));
style()->unpolish(this);
style()->polish(this);
2011-03-02 16:57:41 +01:00
}
2012-01-21 19:26:47 +01:00
void LocationBar::pasteAndGo()
{
clear();
paste();
urlEnter();
}
void LocationBar::contextMenuEvent(QContextMenuEvent* event)
{
Q_UNUSED(event)
if (!m_pasteAndGoAction) {
m_pasteAndGoAction = new QAction(QIcon::fromTheme("edit-paste"), tr("Paste And &Go"), this);
m_pasteAndGoAction->setShortcut(QKeySequence("Ctrl+Shift+V"));
connect(m_pasteAndGoAction, SIGNAL(triggered()), this, SLOT(pasteAndGo()));
2012-01-21 19:26:47 +01:00
}
if (!m_clearAction) {
m_clearAction = new QAction(QIcon::fromTheme("edit-clear"), tr("Clear All"), this);
connect(m_clearAction, SIGNAL(triggered()), this, SLOT(clear()));
}
QMenu* tempMenu = createStandardContextMenu();
QMenu menu(this);
2012-01-21 19:26:47 +01:00
int i = 0;
foreach (QAction* act, tempMenu->actions()) {
menu.addAction(act);
2012-01-21 19:26:47 +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);
2012-01-21 19:26:47 +01:00
break;
case 6:
act->setIcon(QIcon::fromTheme("edit-delete"));
menu.addAction(act);
menu.addAction(m_clearAction);
2012-01-21 19:26:47 +01:00
break;
case 8:
act->setIcon(QIcon::fromTheme("edit-select-all"));
break;
}
++i;
}
m_pasteAndGoAction->setEnabled(!QApplication::clipboard()->text().isEmpty());
// Prevent choosing first option with double rightclick
QPoint pos = event->globalPos();
2012-01-21 19:26:47 +01:00
QPoint p(pos.x(), pos.y() + 1);
menu.exec(p);
tempMenu->deleteLater();
2012-01-21 19:26:47 +01:00
}
void LocationBar::focusInEvent(QFocusEvent* event)
{
const QString &stringUrl = convertUrlToText(m_webView->url());
// Text has been edited, let's show go button
if (stringUrl != text()) {
showGoButton();
}
LineEdit::focusInEvent(event);
}
2011-03-17 17:03:04 +01:00
void LocationBar::dropEvent(QDropEvent* event)
2011-03-02 16:57:41 +01:00
{
if (event->mimeData()->hasUrls()) {
QUrl dropUrl = event->mimeData()->urls().at(0);
if (WebView::isUrlValid(dropUrl)) {
setText(dropUrl.toString());
m_webView->setFocus();
emit loadUrl(dropUrl);
QFocusEvent event(QFocusEvent::FocusOut);
LineEdit::focusOutEvent(&event);
2011-03-02 16:57:41 +01:00
return;
}
}
else if (event->mimeData()->hasText()) {
QUrl dropUrl = QUrl(event->mimeData()->text().trimmed());
2011-03-02 16:57:41 +01:00
if (WebView::isUrlValid(dropUrl)) {
setText(dropUrl.toString());
m_webView->setFocus();
emit loadUrl(dropUrl);
QFocusEvent event(QFocusEvent::FocusOut);
LineEdit::focusOutEvent(&event);
2011-03-02 16:57:41 +01:00
return;
}
}
LineEdit::dropEvent(event);
2011-03-02 16:57:41 +01:00
}
void LocationBar::focusOutEvent(QFocusEvent* event)
{
LineEdit::focusOutEvent(event);
if (event->reason() == Qt::PopupFocusReason
|| (!selectedText().isEmpty() && event->reason() != Qt::TabFocusReason)) {
return;
}
m_popupClosed = false;
m_forcePaintEvent = true;
setCursorPosition(0);
hideGoButton();
if (text().trimmed().isEmpty()) {
clear();
}
}
void LocationBar::keyPressEvent(QKeyEvent* event)
2011-03-02 16:57:41 +01:00
{
switch (event->key()) {
case Qt::Key_V:
if (event->modifiers() == (Qt::ControlModifier | Qt::ShiftModifier)) {
pasteAndGo();
event->accept();
return;
}
break;
case Qt::Key_Down:
m_completer.complete(text());
break;
case Qt::Key_End:
case Qt::Key_Right:
if (m_inlineCompletionVisible) {
m_inlineCompletionVisible = false;
setText(text() + m_completer.domainCompletion());
setCursorPosition(text().size());
m_completer.closePopup();
}
if (m_completer.isPopupVisible()) {
m_completer.closePopup();
}
break;
case Qt::Key_Left:
if (m_completer.isPopupVisible()) {
m_completer.closePopup();
}
break;
case Qt::Key_Delete:
if (m_inlineCompletionVisible) {
m_inlineCompletionVisible = false;
update();
event->accept();
}
break;
case Qt::Key_Escape:
m_webView->setFocus();
showUrl(m_webView->url());
2011-03-02 16:57:41 +01:00
event->accept();
break;
2011-03-02 16:57:41 +01:00
case Qt::Key_Alt:
m_holdingAlt = true;
break;
2011-03-02 16:57:41 +01:00
case Qt::Key_Return:
case Qt::Key_Enter:
switch (event->modifiers()) {
case Qt::ControlModifier:
setText(text().append(QLatin1String(".com")));
urlEnter();
m_holdingAlt = false;
break;
case Qt::AltModifier:
m_completer.closePopup();
p_QupZilla->tabWidget()->addView(createUrl());
m_holdingAlt = false;
break;
default:
urlEnter();
m_holdingAlt = false;
}
break;
case Qt::Key_0:
case Qt::Key_1:
case Qt::Key_2:
case Qt::Key_3:
case Qt::Key_4:
case Qt::Key_5:
case Qt::Key_6:
case Qt::Key_7:
case Qt::Key_8:
case Qt::Key_9:
if (event->modifiers() & Qt::AltModifier || event->modifiers() & Qt::ControlModifier) {
event->ignore();
m_holdingAlt = false;
return;
}
break;
default:
m_holdingAlt = false;
}
LineEdit::keyPressEvent(event);
2011-03-02 16:57:41 +01:00
}
void LocationBar::keyReleaseEvent(QKeyEvent* event)
{
QString localDomain = tr(".co.uk", "Append domain name on ALT + Enter = Should be different for every country");
if (event->key() == Qt::Key_Alt && m_holdingAlt && qzSettings->addCountryWithAlt &&
!text().endsWith(localDomain) && !text().endsWith(QLatin1Char('/'))) {
LineEdit::setText(text().append(localDomain));
}
LineEdit::keyReleaseEvent(event);
}
void LocationBar::onLoadStarted()
{
m_progressVisible = true;
m_autofillIcon->hide();
}
void LocationBar::onLoadProgress(int progress)
{
if (qzSettings->showLoadingProgress) {
m_loadProgress = progress;
update();
}
}
void LocationBar::onLoadFinished()
{
if (qzSettings->showLoadingProgress) {
QTimer::singleShot(700, this, SLOT(hideProgress()));
}
WebPage* page = qobject_cast<WebPage*>(m_webView->page());
if (page && page->hasMultipleUsernames()) {
m_autofillIcon->setFormData(page->autoFillData());
m_autofillIcon->show();
}
}
void LocationBar::loadSettings()
{
Settings settings;
settings.beginGroup("AddressBar");
m_progressStyle = static_cast<ProgressStyle>(settings.value("ProgressStyle", 0).toInt());
bool customColor = settings.value("UseCustomProgressColor", false).toBool();
m_progressColor = customColor ? settings.value("CustomProgressColor", palette().color(QPalette::Highlight)).value<QColor>() : QColor();
settings.endGroup();
}
void LocationBar::onMessage(Qz::AppMessageType msg, bool state)
{
Q_UNUSED(state)
if (!qzSettings->showLoadingProgress) {
return;
}
if (msg == Qz::AM_ReloadSettings) {
loadSettings();
}
}
void LocationBar::hideProgress()
{
if (qzSettings->showLoadingProgress && m_loadProgress == 100) {
m_progressVisible = false;
update();
}
}
void LocationBar::paintEvent(QPaintEvent* event)
{
QStyleOptionFrameV3 option;
initStyleOption(&option);
int lm, tm, rm, bm;
getTextMargins(&lm, &tm, &rm, &bm);
QRect contentsRect = style()->subElementRect(QStyle::SE_LineEditContents, &option, this);
contentsRect.adjust(lm, tm, -rm, -bm);
const QFontMetrics &fm = fontMetrics();
const int x = contentsRect.x() + 4;
const int y = contentsRect.y() + (contentsRect.height() - fm.height() + 1) / 2;
const int width = contentsRect.width() - 6;
const int height = fm.height();
const QRect textRect(x, y, width, height);
QTextOption opt;
opt.setWrapMode(QTextOption::NoWrap);
QPainter p(this);
p.setRenderHint(QPainter::Antialiasing, true);
p.setRenderHint(QPainter::TextAntialiasing, true);
if (hasFocus() && m_inlineCompletionVisible) {
// Draw inline domain completion if available
const QString &completionText = m_completer.domainCompletion();
if (!completionText.isEmpty()) {
QRect completionRect = textRect;
completionRect.setX(completionRect.x() + fm.width(text()) + 1);
completionRect.setWidth(fm.width(completionText) + 1);
style()->drawPrimitive(QStyle::PE_PanelLineEdit, &option, &p, this);
// Text part
p.drawText(textRect, text(), opt);
// Completion part
p.fillRect(completionRect, palette().color(QPalette::Highlight));
p.setPen(palette().color(QPalette::HighlightedText));
p.drawText(completionRect, completionText, opt);
return;
}
}
if (m_completer.isPopupVisible() && !m_completer.showingMostVisited()) {
// We need to draw cursor when popup is visible
// But don't paint it if we are just showing most visited sites
const int cursorWidth = style()->pixelMetric(QStyle::PM_TextCursorWidth, &option, this);
const int cursorHeight = fm.height();
QString textPart = text().left(cursorPosition());
int cursorXpos = x + fontMetrics().width(textPart);
QRect cursor = cursorRect();
cursor.setX(cursorXpos + 1);
cursor.setWidth(cursorWidth);
cursor.setHeight(cursorHeight);
style()->drawPrimitive(QStyle::PE_PanelLineEdit, &option, &p, this);
p.drawText(textRect, text(), opt);
if (textRect.contains(cursor.center().x(), cursor.center().y())) {
p.fillRect(cursor, option.palette.text().color());
}
return;
}
if (hasFocus() || text().isEmpty() || m_forcePaintEvent) {
LineEdit::paintEvent(event);
if (m_forcePaintEvent) {
m_forcePaintEvent = false;
update();
}
return;
}
style()->drawPrimitive(QStyle::PE_PanelLineEdit, &option, &p, this);
QPen oldPen = p.pen();
if (qzSettings->showLoadingProgress && m_progressVisible) {
QColor bg = m_progressColor;
if (!bg.isValid() || bg.alpha() == 0) {
bg = Colors::mid(palette().color(QPalette::Base),
palette().color(QPalette::Text),
m_progressStyle > 0 ? 4 : 8, 1);
}
p.setBrush(QBrush(bg));
QPen outlinePen(bg.darker(110), 0.8);
p.setPen(outlinePen);
switch (m_progressStyle) {
case ProgressFilled: {
QRect bar = contentsRect.adjusted(0, 1, 0, -1);
bar.setWidth(bar.width() * m_loadProgress / 100);
const int roundness = bar.height() / 4.0;
p.drawRoundedRect(bar, roundness, roundness);
break;
}
case ProgressBottom: {
outlinePen.setWidthF(0.3);
outlinePen.setColor(outlinePen.color().darker(130));
p.setPen(outlinePen);
QRect bar(contentsRect.x(), contentsRect.bottom() - 3,
contentsRect.width() * m_loadProgress / 100.0, 3);
p.drawRoundedRect(bar, 1, 1);
break;
}
case ProgressTop: {
outlinePen.setWidthF(0.3);
outlinePen.setColor(outlinePen.color().darker(130));
p.setPen(outlinePen);
QRect bar(contentsRect.x(), contentsRect.top() + 1,
contentsRect.width() * m_loadProgress / 100.0, 3);
p.drawRoundedRect(bar, 1, 1);
break;
}
default:
break;
}
}
p.setPen(oldPen);
const QString hostName = m_webView->url().host();
QString currentText = text();
QRect currentRect = textRect;
if (!hostName.isEmpty()) {
const int hostPos = currentText.indexOf(hostName);
if (hostPos > 0) {
QPen lightPen = oldPen;
QColor lightColor = Colors::mid(palette().color(QPalette::Base),
palette().color(QPalette::Text),
1, 1);
lightPen.setColor(lightColor);
p.setPen(lightPen);
currentText = text().mid(0, hostPos);
currentRect.setWidth(fm.width(currentText));
p.drawText(currentRect, currentText, opt);
p.setPen(oldPen);
currentRect.setX(currentRect.x() + currentRect.width());
const int hostWidth = fm.width(hostName);
currentRect.setWidth(hostWidth);
p.drawText(currentRect, hostName, opt);
p.setFont(font());
currentText = text().mid(hostPos + hostName.length());
currentRect.setX(currentRect.x() + hostWidth);
currentRect.setWidth(textRect.width() - currentRect.x() + textRect.x());
p.setPen(lightPen);
}
}
p.drawText(currentRect, currentText, opt);
}
LocationBar::~LocationBar()
{
}