1
mirror of https://invent.kde.org/network/falkon.git synced 2024-09-23 10:42:11 +02:00
falkonOfficial/src/lib/tools/iconprovider.cpp

280 lines
8.4 KiB
C++
Raw Normal View History

2011-04-27 09:05:54 +02:00
/* ============================================================
* QupZilla - WebKit based browser
* Copyright (C) 2010-2016 David Rosca <nowrep@gmail.com>
2011-04-27 09:05:54 +02: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/>.
* ============================================================ */
#include "iconprovider.h"
#include "mainapplication.h"
2015-09-29 11:45:39 +02:00
#include "networkmanager.h"
#include "sqldatabase.h"
#include "autosaver.h"
#include "webview.h"
#include "qztools.h"
#include <QTimer>
#include <QBuffer>
#include <QFutureWatcher>
#include <QtConcurrent/QtConcurrentRun>
Q_GLOBAL_STATIC(IconProvider, qz_icon_provider)
IconProvider::IconProvider()
: QWidget()
{
m_autoSaver = new AutoSaver(this);
connect(m_autoSaver, SIGNAL(save()), this, SLOT(saveIconsToDatabase()));
}
void IconProvider::saveIcon(WebView* view)
{
// Don't save icons in private mode.
if (mApp->isPrivate()) {
return;
}
static const char *ignoredSchemes[] = { "qupzilla", "ftp", "file", "view-source" };
for (unsigned i = 0; i < sizeof(ignoredSchemes) / sizeof(ignoredSchemes[0]); ++i)
if (view->url().scheme() == ignoredSchemes[i])
return;
BufferedIcon item;
item.first = view->url();
item.second = view->icon().pixmap(32).toImage();
if (item.second == IconProvider::emptyWebImage()) {
return;
}
if (m_iconBuffer.contains(item)) {
return;
}
m_autoSaver->changeOccurred();
m_iconBuffer.append(item);
}
QIcon IconProvider::bookmarkIcon() const
{
return QIcon::fromTheme(QSL("bookmarks"), m_bookmarkIcon);
}
void IconProvider::setBookmarkIcon(const QIcon &icon)
{
m_bookmarkIcon = icon;
}
QIcon IconProvider::standardIcon(QStyle::StandardPixmap icon)
{
switch (icon) {
case QStyle::SP_MessageBoxCritical:
return QIcon::fromTheme(QSL("dialog-error"), QApplication::style()->standardIcon(icon));
case QStyle::SP_MessageBoxInformation:
return QIcon::fromTheme(QSL("dialog-information"), QApplication::style()->standardIcon(icon));
case QStyle::SP_MessageBoxQuestion:
return QIcon::fromTheme(QSL("dialog-question"), QApplication::style()->standardIcon(icon));
case QStyle::SP_MessageBoxWarning:
return QIcon::fromTheme(QSL("dialog-warning"), QApplication::style()->standardIcon(icon));
case QStyle::SP_DialogCloseButton:
return QIcon::fromTheme(QSL("dialog-close"), QApplication::style()->standardIcon(icon));
case QStyle::SP_BrowserStop:
return QIcon::fromTheme(QSL("process-stop"), QApplication::style()->standardIcon(icon));
case QStyle::SP_BrowserReload:
return QIcon::fromTheme(QSL("view-refresh"), QApplication::style()->standardIcon(icon));
case QStyle::SP_FileDialogToParent:
return QIcon::fromTheme(QSL("go-up"), QApplication::style()->standardIcon(icon));
case QStyle::SP_ArrowUp:
return QIcon::fromTheme(QSL("go-up"), QApplication::style()->standardIcon(icon));
case QStyle::SP_ArrowDown:
return QIcon::fromTheme(QSL("go-down"), QApplication::style()->standardIcon(icon));
case QStyle::SP_ArrowForward:
if (QApplication::layoutDirection() == Qt::RightToLeft) {
return QIcon::fromTheme(QSL("go-previous"), QApplication::style()->standardIcon(icon));
}
return QIcon::fromTheme(QSL("go-next"), QApplication::style()->standardIcon(icon));
2011-10-01 17:55:10 +02:00
case QStyle::SP_ArrowBack:
if (QApplication::layoutDirection() == Qt::RightToLeft) {
return QIcon::fromTheme(QSL("go-next"), QApplication::style()->standardIcon(icon));
}
return QIcon::fromTheme(QSL("go-previous"), QApplication::style()->standardIcon(icon));
default:
return QApplication::style()->standardIcon(icon);
}
}
QIcon IconProvider::newTabIcon()
{
return QIcon::fromTheme(QSL("tab-new"), QIcon(QSL(":/icons/menu/tab-new.png")));
}
QIcon IconProvider::newWindowIcon()
{
return QIcon::fromTheme(QSL("window-new"), QIcon(QSL(":/icons/menu/window-new.png")));
}
QIcon IconProvider::privateBrowsingIcon()
{
return QIcon(QSL(":/icons/menu/privatebrowsing.png"));
}
QIcon IconProvider::settingsIcon()
{
return QIcon::fromTheme(QSL("configure"), QIcon(QSL(":/icons/menu/settings.png")));
}
QIcon IconProvider::emptyWebIcon()
{
return QPixmap::fromImage(instance()->emptyWebImage());
}
QImage IconProvider::emptyWebImage()
{
if (instance()->m_emptyWebImage.isNull()) {
instance()->m_emptyWebImage = QPixmap(":icons/other/empty-page.png").toImage();
}
return instance()->m_emptyWebImage;
}
QIcon IconProvider::iconForUrl(const QUrl &url, bool allowEmpty)
{
return instance()->iconFromImage(imageForUrl(url, allowEmpty));
}
QImage IconProvider::imageForUrl(const QUrl &url, bool allowEmpty)
{
if (url.path().isEmpty()) {
return allowEmpty ? QImage() : IconProvider::emptyWebImage();
}
foreach (const BufferedIcon &ic, instance()->m_iconBuffer) {
if (ic.first.toString().startsWith(url.toString())) {
return ic.second;
}
}
QSqlQuery query;
query.prepare(QSL("SELECT icon FROM icons WHERE url LIKE ? ESCAPE ? LIMIT 1"));
query.addBindValue(QString("%1%").arg(QzTools::escapeSqlString(QString::fromUtf8(url.toEncoded(QUrl::RemoveFragment)))));
query.addBindValue(QL1S("!"));
query = SqlDatabase::instance()->exec(query);
if (query.next()) {
return QImage::fromData(query.value(0).toByteArray());
}
return allowEmpty ? QImage() : IconProvider::emptyWebImage();
}
void IconProvider::imageForUrlAsync(const QUrl &url, QObject *receiver, std::function<void(const QImage &)> callback)
{
QFutureWatcher<QImage> *watcher = new QFutureWatcher<QImage>();
connect(watcher, &QFutureWatcher<QImage>::finished, receiver, [=]() {
watcher->deleteLater();
callback(watcher->result());
});
watcher->setFuture(QtConcurrent::run(imageForUrl, url, false));
}
QIcon IconProvider::iconForDomain(const QUrl &url, bool allowEmpty)
{
return instance()->iconFromImage(imageForDomain(url, allowEmpty));
}
QImage IconProvider::imageForDomain(const QUrl &url, bool allowEmpty)
{
if (url.host().isEmpty()) {
return allowEmpty ? QImage() : IconProvider::emptyWebImage();
}
foreach (const BufferedIcon &ic, instance()->m_iconBuffer) {
if (ic.first.host() == url.host()) {
return ic.second;
}
}
QSqlQuery query;
query.prepare(QSL("SELECT icon FROM icons WHERE url LIKE ? ESCAPE ? LIMIT 1"));
query.addBindValue(QString("%%1%").arg(QzTools::escapeSqlString(url.host())));
query.addBindValue(QL1S("!"));
query.exec();
if (query.next()) {
return QImage::fromData(query.value(0).toByteArray());
}
return allowEmpty ? QImage() : IconProvider::emptyWebImage();
}
IconProvider* IconProvider::instance()
{
return qz_icon_provider();
}
void IconProvider::saveIconsToDatabase()
{
foreach (const BufferedIcon &ic, m_iconBuffer) {
QSqlQuery query;
query.prepare("SELECT id FROM icons WHERE url = ?");
query.bindValue(0, ic.first.toEncoded(QUrl::RemoveFragment));
query.exec();
if (query.next()) {
query.prepare("UPDATE icons SET icon = ? WHERE url = ?");
}
else {
query.prepare("INSERT INTO icons (icon, url) VALUES (?,?)");
}
QByteArray ba;
QBuffer buffer(&ba);
buffer.open(QIODevice::WriteOnly);
ic.second.save(&buffer, "PNG");
query.bindValue(0, buffer.data());
query.bindValue(1, ic.first.toEncoded(QUrl::RemoveFragment));
SqlDatabase::instance()->execAsync(query);
}
m_iconBuffer.clear();
}
void IconProvider::clearIconsDatabase()
{
QSqlQuery query;
query.exec("DELETE FROM icons");
query.exec("VACUUM");
m_iconBuffer.clear();
}
QIcon IconProvider::iconFromImage(const QImage &image)
{
return QIcon(QPixmap::fromImage(image));
}