2015-12-10 05:06:13 +02:00
|
|
|
/****************************************************************************
|
|
|
|
**
|
|
|
|
** Copyright (C) 2015 The Qt Company Ltd.
|
|
|
|
** Contact: http://www.qt.io/licensing/
|
|
|
|
**
|
|
|
|
** This file is part of the QtGui module of the Qt Toolkit.
|
|
|
|
**
|
|
|
|
** $QT_BEGIN_LICENSE:LGPL$
|
|
|
|
** Commercial License Usage
|
|
|
|
** Licensees holding valid commercial Qt licenses may use this file in
|
|
|
|
** accordance with the commercial license agreement provided with the
|
|
|
|
** Software or, alternatively, in accordance with the terms contained in
|
|
|
|
** a written agreement between you and The Qt Company. For licensing terms
|
|
|
|
** and conditions see http://www.qt.io/terms-conditions. For further
|
|
|
|
** information use the contact form at http://www.qt.io/contact-us.
|
|
|
|
**
|
|
|
|
** GNU Lesser General Public License Usage
|
|
|
|
** Alternatively, this file may be used under the terms of the GNU Lesser
|
|
|
|
** General Public License version 2.1 or version 3 as published by the Free
|
|
|
|
** Software Foundation and appearing in the file LICENSE.LGPLv21 and
|
|
|
|
** LICENSE.LGPLv3 included in the packaging of this file. Please review the
|
|
|
|
** following information to ensure the GNU Lesser General Public License
|
|
|
|
** requirements will be met: https://www.gnu.org/licenses/lgpl.html and
|
|
|
|
** http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.
|
|
|
|
**
|
|
|
|
** As a special exception, The Qt Company gives you certain additional
|
|
|
|
** rights. These rights are described in The Qt Company LGPL Exception
|
|
|
|
** version 1.1, included in the file LGPL_EXCEPTION.txt in this package.
|
|
|
|
**
|
|
|
|
** GNU General Public License Usage
|
|
|
|
** Alternatively, this file may be used under the terms of the GNU
|
|
|
|
** General Public License version 3.0 as published by the Free Software
|
|
|
|
** Foundation and appearing in the file LICENSE.GPL included in the
|
|
|
|
** packaging of this file. Please review the following information to
|
|
|
|
** ensure the GNU General Public License version 3.0 requirements will be
|
|
|
|
** met: http://www.gnu.org/copyleft/gpl.html.
|
|
|
|
**
|
|
|
|
** $QT_END_LICENSE$
|
|
|
|
**
|
|
|
|
****************************************************************************/
|
|
|
|
#ifndef QT_NO_ICON
|
|
|
|
|
2016-09-06 06:05:43 +00:00
|
|
|
#include "qiconloader_p.h"
|
|
|
|
#include "qapplication_p.h"
|
|
|
|
#include "qicon_p.h"
|
2019-05-04 17:02:17 +00:00
|
|
|
#include "qguiplatformplugin.h"
|
2016-09-06 06:05:43 +00:00
|
|
|
#include "qfactoryloader_p.h"
|
|
|
|
#include "qguicommon_p.h"
|
2015-12-10 05:06:13 +02:00
|
|
|
|
|
|
|
#include <QtGui/QIconEnginePlugin>
|
|
|
|
#include <QtGui/QPixmapCache>
|
|
|
|
#include <QtGui/QIconEngine>
|
|
|
|
#include <QtGui/QStyleOption>
|
|
|
|
#include <QtCore/QList>
|
|
|
|
#include <QtCore/QHash>
|
|
|
|
#include <QtCore/QDir>
|
|
|
|
#include <QtCore/QSettings>
|
|
|
|
#include <QtGui/QPainter>
|
|
|
|
|
|
|
|
#ifdef Q_WS_X11
|
|
|
|
#include <qt_x11_p.h>
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#include <qstylehelper_p.h>
|
|
|
|
|
|
|
|
QT_BEGIN_NAMESPACE
|
|
|
|
|
|
|
|
Q_GLOBAL_STATIC(QIconLoader, iconLoaderInstance)
|
|
|
|
|
|
|
|
/* Theme to use in last resort, if the theme does not have the icon, neither the parents */
|
|
|
|
static QString fallbackTheme()
|
|
|
|
{
|
|
|
|
#ifdef Q_WS_X11
|
2019-05-04 16:59:05 +00:00
|
|
|
return QLatin1String("hicolor");
|
|
|
|
#else
|
2015-12-10 05:06:13 +02:00
|
|
|
return QString();
|
2019-05-04 16:59:05 +00:00
|
|
|
#endif
|
2015-12-10 05:06:13 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
QIconLoader::QIconLoader() :
|
2016-08-07 16:14:51 +00:00
|
|
|
m_themeKey(1), m_supportsSvg(false)
|
2015-12-10 05:06:13 +02:00
|
|
|
{
|
2016-08-07 16:14:51 +00:00
|
|
|
Q_ASSERT(qApp);
|
2015-12-10 05:06:13 +02:00
|
|
|
|
2016-08-07 16:14:51 +00:00
|
|
|
m_systemTheme = qt_guiPlatformPlugin()->systemIconThemeName();
|
|
|
|
if (m_systemTheme.isEmpty())
|
|
|
|
m_systemTheme = fallbackTheme();
|
2015-12-10 05:06:13 +02:00
|
|
|
#ifndef QT_NO_LIBRARY
|
2016-09-06 05:28:14 +00:00
|
|
|
if (iconloader()->keys().contains(QLatin1String("svg")))
|
2016-08-07 16:14:51 +00:00
|
|
|
m_supportsSvg = true;
|
2015-12-10 05:06:13 +02:00
|
|
|
#endif //QT_NO_LIBRARY
|
|
|
|
}
|
|
|
|
|
|
|
|
QIconLoader *QIconLoader::instance()
|
|
|
|
{
|
|
|
|
return iconLoaderInstance();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Queries the system theme and invalidates existing
|
|
|
|
// icons if the theme has changed.
|
|
|
|
void QIconLoader::updateSystemTheme()
|
|
|
|
{
|
|
|
|
// Only change if this is not explicitly set by the user
|
|
|
|
if (m_userTheme.isEmpty()) {
|
|
|
|
QString theme = qt_guiPlatformPlugin()->systemIconThemeName();
|
|
|
|
if (theme.isEmpty())
|
|
|
|
theme = fallbackTheme();
|
|
|
|
if (theme != m_systemTheme) {
|
|
|
|
m_systemTheme = theme;
|
|
|
|
invalidateKey();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void QIconLoader::setThemeName(const QString &themeName)
|
|
|
|
{
|
|
|
|
m_userTheme = themeName;
|
|
|
|
invalidateKey();
|
|
|
|
}
|
|
|
|
|
|
|
|
void QIconLoader::setThemeSearchPath(const QStringList &searchPaths)
|
|
|
|
{
|
|
|
|
m_iconDirs = searchPaths;
|
|
|
|
themeList.clear();
|
|
|
|
invalidateKey();
|
|
|
|
}
|
|
|
|
|
|
|
|
QStringList QIconLoader::themeSearchPaths() const
|
|
|
|
{
|
|
|
|
if (m_iconDirs.isEmpty()) {
|
|
|
|
m_iconDirs = qt_guiPlatformPlugin()->iconThemeSearchPaths();
|
|
|
|
// Always add resource directory as search path
|
|
|
|
m_iconDirs.append(QLatin1String(":/icons"));
|
|
|
|
}
|
|
|
|
return m_iconDirs;
|
|
|
|
}
|
|
|
|
|
|
|
|
QIconTheme::QIconTheme(const QString &themeName)
|
|
|
|
: m_valid(false)
|
|
|
|
{
|
|
|
|
QFile themeIndex;
|
|
|
|
|
|
|
|
QList <QIconDirInfo> keyList;
|
|
|
|
QStringList iconDirs = QIcon::themeSearchPaths();
|
|
|
|
for ( int i = 0 ; i < iconDirs.size() ; ++i) {
|
|
|
|
QDir iconDir(iconDirs[i]);
|
|
|
|
QString themeDir = iconDir.path() + QLatin1Char('/') + themeName;
|
|
|
|
themeIndex.setFileName(themeDir + QLatin1String("/index.theme"));
|
|
|
|
if (themeIndex.exists()) {
|
|
|
|
m_contentDir = themeDir;
|
|
|
|
m_valid = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#ifndef QT_NO_SETTINGS
|
|
|
|
if (themeIndex.exists()) {
|
|
|
|
const QSettings indexReader(themeIndex.fileName(), QSettings::IniFormat);
|
|
|
|
QStringListIterator keyIterator(indexReader.allKeys());
|
|
|
|
while (keyIterator.hasNext()) {
|
|
|
|
|
|
|
|
const QString key = keyIterator.next();
|
|
|
|
if (key.endsWith(QLatin1String("/Size"))) {
|
|
|
|
// Note the QSettings ini-format does not accept
|
|
|
|
// slashes in key names, hence we have to cheat
|
|
|
|
if (int size = indexReader.value(key).toInt()) {
|
|
|
|
QString directoryKey = key.left(key.size() - 5);
|
|
|
|
QIconDirInfo dirInfo(directoryKey);
|
|
|
|
dirInfo.size = size;
|
|
|
|
QString type = indexReader.value(directoryKey +
|
|
|
|
QLatin1String("/Type")
|
|
|
|
).toString();
|
|
|
|
|
|
|
|
if (type == QLatin1String("Fixed"))
|
|
|
|
dirInfo.type = QIconDirInfo::Fixed;
|
|
|
|
else if (type == QLatin1String("Scalable"))
|
|
|
|
dirInfo.type = QIconDirInfo::Scalable;
|
|
|
|
else
|
|
|
|
dirInfo.type = QIconDirInfo::Threshold;
|
|
|
|
|
|
|
|
dirInfo.threshold = indexReader.value(directoryKey +
|
|
|
|
QLatin1String("/Threshold"),
|
|
|
|
2).toInt();
|
|
|
|
|
|
|
|
dirInfo.minSize = indexReader.value(directoryKey +
|
|
|
|
QLatin1String("/MinSize"),
|
|
|
|
size).toInt();
|
|
|
|
|
|
|
|
dirInfo.maxSize = indexReader.value(directoryKey +
|
|
|
|
QLatin1String("/MaxSize"),
|
|
|
|
size).toInt();
|
|
|
|
m_keyList.append(dirInfo);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Parent themes provide fallbacks for missing icons
|
|
|
|
m_parents = indexReader.value(
|
|
|
|
QLatin1String("Icon Theme/Inherits")).toStringList();
|
|
|
|
|
|
|
|
// Ensure a default platform fallback for all themes
|
|
|
|
if (m_parents.isEmpty())
|
|
|
|
m_parents.append(fallbackTheme());
|
|
|
|
|
|
|
|
// Ensure that all themes fall back to hicolor
|
|
|
|
if (!m_parents.contains(QLatin1String("hicolor")))
|
|
|
|
m_parents.append(QLatin1String("hicolor"));
|
|
|
|
}
|
|
|
|
#endif //QT_NO_SETTINGS
|
|
|
|
}
|
|
|
|
|
|
|
|
QThemeIconEntries QIconLoader::findIconHelper(const QString &themeName,
|
|
|
|
const QString &iconName,
|
|
|
|
QStringList &visited) const
|
|
|
|
{
|
|
|
|
QThemeIconEntries entries;
|
|
|
|
Q_ASSERT(!themeName.isEmpty());
|
|
|
|
|
|
|
|
QPixmap pixmap;
|
|
|
|
|
|
|
|
// Used to protect against potential recursions
|
|
|
|
visited << themeName;
|
|
|
|
|
|
|
|
QIconTheme theme = themeList.value(themeName);
|
|
|
|
if (!theme.isValid()) {
|
|
|
|
theme = QIconTheme(themeName);
|
|
|
|
if (!theme.isValid())
|
|
|
|
theme = QIconTheme(fallbackTheme());
|
|
|
|
|
|
|
|
themeList.insert(themeName, theme);
|
|
|
|
}
|
|
|
|
|
|
|
|
QString contentDir = theme.contentDir() + QLatin1Char('/');
|
|
|
|
QList<QIconDirInfo> subDirs = theme.keyList();
|
|
|
|
|
|
|
|
// Add all relevant files
|
|
|
|
for (int i = 0; i < subDirs.size() ; ++i) {
|
|
|
|
const QIconDirInfo &dirInfo = subDirs.at(i);
|
2016-06-11 20:59:18 +00:00
|
|
|
const QString subDir = contentDir + dirInfo.path + QLatin1Char('/');
|
|
|
|
const QString pngPath = subDir + iconName + QLatin1String(".png");
|
|
|
|
if (QFile::exists(pngPath)) {
|
2015-12-10 05:06:13 +02:00
|
|
|
PixmapEntry *iconEntry = new PixmapEntry;
|
|
|
|
iconEntry->dir = dirInfo;
|
2016-06-11 20:59:18 +00:00
|
|
|
iconEntry->filename = pngPath;
|
2015-12-10 05:06:13 +02:00
|
|
|
// Notice we ensure that pixmap entries always come before
|
|
|
|
// scalable to preserve search order afterwards
|
|
|
|
entries.prepend(iconEntry);
|
2016-06-11 20:59:18 +00:00
|
|
|
} else if (m_supportsSvg) {
|
|
|
|
const QString svgPath = subDir + iconName + QLatin1String(".svg");
|
|
|
|
if (QFile::exists(svgPath)) {
|
|
|
|
ScalableEntry *iconEntry = new ScalableEntry;
|
|
|
|
iconEntry->dir = dirInfo;
|
|
|
|
iconEntry->filename = svgPath;
|
|
|
|
entries.append(iconEntry);
|
|
|
|
}
|
2015-12-10 05:06:13 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (entries.isEmpty()) {
|
|
|
|
const QStringList parents = theme.parents();
|
|
|
|
// Search recursively through inherited themes
|
|
|
|
for (int i = 0 ; i < parents.size() ; ++i) {
|
|
|
|
|
|
|
|
const QString parentTheme = parents.at(i).trimmed();
|
|
|
|
|
|
|
|
if (!visited.contains(parentTheme)) // guard against recursion
|
|
|
|
entries = findIconHelper(parentTheme, iconName, visited);
|
|
|
|
|
|
|
|
if (!entries.isEmpty()) // success
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return entries;
|
|
|
|
}
|
|
|
|
|
|
|
|
QThemeIconEntries QIconLoader::loadIcon(const QString &name) const
|
|
|
|
{
|
|
|
|
if (!themeName().isEmpty()) {
|
|
|
|
QStringList visited;
|
|
|
|
return findIconHelper(themeName(), name, visited);
|
|
|
|
}
|
|
|
|
|
|
|
|
return QThemeIconEntries();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// -------- Icon Loader Engine -------- //
|
|
|
|
|
|
|
|
|
|
|
|
QIconLoaderEngine::QIconLoaderEngine(const QString& iconName)
|
|
|
|
: m_iconName(iconName), m_key(0)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
QIconLoaderEngine::~QIconLoaderEngine()
|
|
|
|
{
|
|
|
|
while (!m_entries.isEmpty())
|
|
|
|
delete m_entries.takeLast();
|
|
|
|
Q_ASSERT(m_entries.size() == 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
QIconLoaderEngine::QIconLoaderEngine(const QIconLoaderEngine &other)
|
2016-06-25 14:33:40 +00:00
|
|
|
: QIconEngine(other),
|
2015-12-10 05:06:13 +02:00
|
|
|
m_iconName(other.m_iconName),
|
|
|
|
m_key(0)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2016-06-25 14:33:40 +00:00
|
|
|
QIconEngine *QIconLoaderEngine::clone() const
|
2015-12-10 05:06:13 +02:00
|
|
|
{
|
|
|
|
return new QIconLoaderEngine(*this);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool QIconLoaderEngine::read(QDataStream &in) {
|
|
|
|
in >> m_iconName;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool QIconLoaderEngine::write(QDataStream &out) const
|
|
|
|
{
|
|
|
|
out << m_iconName;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool QIconLoaderEngine::hasIcon() const
|
|
|
|
{
|
|
|
|
return !(m_entries.isEmpty());
|
|
|
|
}
|
|
|
|
|
|
|
|
// Lazily load the icon
|
|
|
|
void QIconLoaderEngine::ensureLoaded()
|
|
|
|
{
|
|
|
|
if (!(iconLoaderInstance()->themeKey() == m_key)) {
|
|
|
|
|
|
|
|
while (!m_entries.isEmpty())
|
|
|
|
delete m_entries.takeLast();
|
|
|
|
|
|
|
|
Q_ASSERT(m_entries.size() == 0);
|
|
|
|
m_entries = iconLoaderInstance()->loadIcon(m_iconName);
|
|
|
|
m_key = iconLoaderInstance()->themeKey();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void QIconLoaderEngine::paint(QPainter *painter, const QRect &rect,
|
|
|
|
QIcon::Mode mode, QIcon::State state)
|
|
|
|
{
|
2017-08-06 07:44:53 +00:00
|
|
|
painter->drawPixmap(rect, pixmap(rect.size(), mode, state));
|
2015-12-10 05:06:13 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* This algorithm is defined by the freedesktop spec:
|
|
|
|
* http://standards.freedesktop.org/icon-theme-spec/icon-theme-spec-latest.html
|
|
|
|
*/
|
|
|
|
static bool directoryMatchesSize(const QIconDirInfo &dir, int iconsize)
|
|
|
|
{
|
|
|
|
if (dir.type == QIconDirInfo::Fixed) {
|
|
|
|
return dir.size == iconsize;
|
|
|
|
|
|
|
|
} else if (dir.type == QIconDirInfo::Scalable) {
|
|
|
|
return dir.size <= dir.maxSize &&
|
|
|
|
iconsize >= dir.minSize;
|
|
|
|
|
|
|
|
} else if (dir.type == QIconDirInfo::Threshold) {
|
|
|
|
return iconsize >= dir.size - dir.threshold &&
|
|
|
|
iconsize <= dir.size + dir.threshold;
|
|
|
|
}
|
|
|
|
|
|
|
|
Q_ASSERT(1); // Not a valid value
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* This algorithm is defined by the freedesktop spec:
|
|
|
|
* http://standards.freedesktop.org/icon-theme-spec/icon-theme-spec-latest.html
|
|
|
|
*/
|
|
|
|
static int directorySizeDistance(const QIconDirInfo &dir, int iconsize)
|
|
|
|
{
|
|
|
|
if (dir.type == QIconDirInfo::Fixed) {
|
|
|
|
return qAbs(dir.size - iconsize);
|
|
|
|
|
|
|
|
} else if (dir.type == QIconDirInfo::Scalable) {
|
|
|
|
if (iconsize < dir.minSize)
|
|
|
|
return dir.minSize - iconsize;
|
|
|
|
else if (iconsize > dir.maxSize)
|
|
|
|
return iconsize - dir.maxSize;
|
|
|
|
else
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
} else if (dir.type == QIconDirInfo::Threshold) {
|
|
|
|
if (iconsize < dir.size - dir.threshold)
|
|
|
|
return dir.minSize - iconsize;
|
|
|
|
else if (iconsize > dir.size + dir.threshold)
|
|
|
|
return iconsize - dir.maxSize;
|
|
|
|
else return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
Q_ASSERT(1); // Not a valid value
|
|
|
|
return INT_MAX;
|
|
|
|
}
|
|
|
|
|
|
|
|
QIconLoaderEngineEntry *QIconLoaderEngine::entryForSize(const QSize &size)
|
|
|
|
{
|
|
|
|
int iconsize = qMin(size.width(), size.height());
|
|
|
|
|
|
|
|
// Note that m_entries are sorted so that png-files
|
|
|
|
// come first
|
|
|
|
|
|
|
|
// Search for exact matches first
|
|
|
|
for (int i = 0; i < m_entries.count(); ++i) {
|
|
|
|
QIconLoaderEngineEntry *entry = m_entries.at(i);
|
|
|
|
if (directoryMatchesSize(entry->dir, iconsize)) {
|
|
|
|
return entry;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Find the minimum distance icon
|
|
|
|
int minimalSize = INT_MAX;
|
|
|
|
QIconLoaderEngineEntry *closestMatch = 0;
|
|
|
|
for (int i = 0; i < m_entries.count(); ++i) {
|
|
|
|
QIconLoaderEngineEntry *entry = m_entries.at(i);
|
|
|
|
int distance = directorySizeDistance(entry->dir, iconsize);
|
|
|
|
if (distance < minimalSize) {
|
|
|
|
minimalSize = distance;
|
|
|
|
closestMatch = entry;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return closestMatch;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Returns the actual icon size. For scalable svg's this is equivalent
|
|
|
|
* to the requested size. Otherwise the closest match is returned but
|
|
|
|
* we can never return a bigger size than the requested size.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
QSize QIconLoaderEngine::actualSize(const QSize &size, QIcon::Mode mode,
|
|
|
|
QIcon::State state)
|
|
|
|
{
|
|
|
|
ensureLoaded();
|
|
|
|
|
|
|
|
QIconLoaderEngineEntry *entry = entryForSize(size);
|
|
|
|
if (entry) {
|
|
|
|
const QIconDirInfo &dir = entry->dir;
|
|
|
|
if (dir.type == QIconDirInfo::Scalable)
|
|
|
|
return size;
|
|
|
|
else {
|
|
|
|
int result = qMin<int>(dir.size, qMin(size.width(), size.height()));
|
|
|
|
return QSize(result, result);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return QIconEngineV2::actualSize(size, mode, state);
|
|
|
|
}
|
|
|
|
|
|
|
|
QPixmap PixmapEntry::pixmap(const QSize &size, QIcon::Mode mode, QIcon::State state)
|
|
|
|
{
|
|
|
|
Q_UNUSED(state);
|
|
|
|
|
|
|
|
// Ensure that basePixmap is lazily initialized before generating the
|
|
|
|
// key, otherwise the cache key is not unique
|
|
|
|
if (basePixmap.isNull())
|
|
|
|
basePixmap.load(filename);
|
|
|
|
|
|
|
|
int actualSize = qMin(size.width(), size.height());
|
|
|
|
|
2016-10-30 06:18:21 +00:00
|
|
|
QString key = QLatin1String("$qt_theme_")
|
2019-05-04 17:20:49 +00:00
|
|
|
+ HexString<qint64>(basePixmap.cacheKey())
|
|
|
|
+ HexString<int>(mode)
|
|
|
|
+ HexString<qint64>(qApp->palette().cacheKey())
|
|
|
|
+ HexString<int>(actualSize);
|
2015-12-10 05:06:13 +02:00
|
|
|
|
|
|
|
QPixmap cachedPixmap;
|
|
|
|
if (QPixmapCache::find(key, &cachedPixmap)) {
|
|
|
|
return cachedPixmap;
|
|
|
|
} else {
|
|
|
|
QStyleOption opt(0);
|
|
|
|
opt.palette = qApp->palette();
|
|
|
|
cachedPixmap = qApp->style()->generatedIconPixmap(mode, basePixmap, &opt);
|
|
|
|
QPixmapCache::insert(key, cachedPixmap);
|
|
|
|
}
|
|
|
|
return cachedPixmap;
|
|
|
|
}
|
|
|
|
|
|
|
|
QPixmap ScalableEntry::pixmap(const QSize &size, QIcon::Mode mode, QIcon::State state)
|
|
|
|
{
|
|
|
|
if (svgIcon.isNull())
|
|
|
|
svgIcon = QIcon(filename);
|
|
|
|
|
|
|
|
// Simply reuse svg icon engine
|
|
|
|
return svgIcon.pixmap(size, mode, state);
|
|
|
|
}
|
|
|
|
|
|
|
|
QPixmap QIconLoaderEngine::pixmap(const QSize &size, QIcon::Mode mode,
|
|
|
|
QIcon::State state)
|
|
|
|
{
|
|
|
|
ensureLoaded();
|
|
|
|
|
|
|
|
QIconLoaderEngineEntry *entry = entryForSize(size);
|
|
|
|
if (entry)
|
|
|
|
return entry->pixmap(size, mode, state);
|
|
|
|
|
|
|
|
return QPixmap();
|
|
|
|
}
|
|
|
|
|
|
|
|
QString QIconLoaderEngine::key() const
|
|
|
|
{
|
|
|
|
return QLatin1String("QIconLoaderEngine");
|
|
|
|
}
|
|
|
|
|
|
|
|
QT_END_NAMESPACE
|
|
|
|
|
|
|
|
#endif //QT_NO_ICON
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|