You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
866 lines
29 KiB
866 lines
29 KiB
10 years ago
|
/****************************************************************************
|
||
|
**
|
||
|
** Copyright (C) 2014 Digia Plc and/or its subsidiary(-ies).
|
||
|
** Contact: http://www.qt-project.org/legal
|
||
|
**
|
||
|
** This file is part of the QtGui module of the Qt Toolkit.
|
||
|
**
|
||
|
** $QT_BEGIN_LICENSE:LGPL21$
|
||
|
** 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 Digia. For licensing terms and
|
||
|
** conditions see http://qt.digia.com/licensing. For further information
|
||
|
** use the contact form at http://qt.digia.com/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.
|
||
|
**
|
||
|
** In addition, as a special exception, Digia gives you certain additional
|
||
|
** rights. These rights are described in the Digia Qt LGPL Exception
|
||
|
** version 1.1, included in the file LGPL_EXCEPTION.txt in this package.
|
||
|
**
|
||
|
** $QT_END_LICENSE$
|
||
|
**
|
||
|
****************************************************************************/
|
||
|
#ifndef QT_NO_ICON
|
||
8 years ago
|
#include "xdgiconloader_p.h"
|
||
10 years ago
|
|
||
|
#include <private/qguiapplication_p.h>
|
||
|
#include <private/qicon_p.h>
|
||
|
|
||
|
#include <QtGui/QIconEnginePlugin>
|
||
|
#include <QtGui/QPixmapCache>
|
||
|
#include <qpa/qplatformtheme.h>
|
||
|
#include <QtGui/QIconEngine>
|
||
|
#include <QtGui/QPalette>
|
||
|
#include <QtCore/QList>
|
||
|
#include <QtCore/QDir>
|
||
|
#include <QtCore/QSettings>
|
||
|
#include <QtGui/QPainter>
|
||
|
|
||
8 years ago
|
#ifdef Q_DEAD_CODE_FROM_QT4_MAC
|
||
10 years ago
|
#include <private/qt_cocoa_helpers_mac_p.h>
|
||
|
#endif
|
||
|
|
||
|
#include <private/qhexstring_p.h>
|
||
|
|
||
|
//QT_BEGIN_NAMESPACE
|
||
|
|
||
|
|
||
8 years ago
|
Q_GLOBAL_STATIC(XdgIconLoader, iconLoaderInstance)
|
||
10 years ago
|
|
||
|
/* Theme to use in last resort, if the theme does not have the icon, neither the parents */
|
||
|
static QString fallbackTheme()
|
||
|
{
|
||
|
if (const QPlatformTheme *theme = QGuiApplicationPrivate::platformTheme()) {
|
||
|
const QVariant themeHint = theme->themeHint(QPlatformTheme::SystemIconFallbackThemeName);
|
||
|
if (themeHint.isValid())
|
||
|
return themeHint.toString();
|
||
|
}
|
||
8 years ago
|
return QLatin1String("hicolor");
|
||
10 years ago
|
}
|
||
|
|
||
8 years ago
|
XdgIconLoader::XdgIconLoader() :
|
||
10 years ago
|
m_themeKey(1), m_supportsSvg(false), m_initialized(false)
|
||
|
{
|
||
|
}
|
||
|
|
||
|
static inline QString systemThemeName()
|
||
|
{
|
||
|
if (const QPlatformTheme *theme = QGuiApplicationPrivate::platformTheme()) {
|
||
|
const QVariant themeHint = theme->themeHint(QPlatformTheme::SystemIconThemeName);
|
||
|
if (themeHint.isValid())
|
||
|
return themeHint.toString();
|
||
|
}
|
||
|
return QIcon::themeName();
|
||
|
}
|
||
|
|
||
|
static inline QStringList systemIconSearchPaths()
|
||
|
{
|
||
|
if (const QPlatformTheme *theme = QGuiApplicationPrivate::platformTheme()) {
|
||
|
const QVariant themeHint = theme->themeHint(QPlatformTheme::IconThemeSearchPaths);
|
||
|
if (themeHint.isValid())
|
||
|
return themeHint.toStringList();
|
||
|
}
|
||
|
return QIcon::themeSearchPaths();
|
||
|
}
|
||
|
|
||
|
#ifndef QT_NO_LIBRARY
|
||
|
//extern QFactoryLoader *qt_iconEngineFactoryLoader(); // qicon.cpp
|
||
|
#endif
|
||
|
|
||
8 years ago
|
void XdgIconLoader::ensureInitialized()
|
||
10 years ago
|
{
|
||
|
if (!m_initialized) {
|
||
|
m_initialized = true;
|
||
|
|
||
|
Q_ASSERT(qApp);
|
||
|
|
||
|
m_systemTheme = systemThemeName();
|
||
|
|
||
|
if (m_systemTheme.isEmpty())
|
||
|
m_systemTheme = fallbackTheme();
|
||
|
#ifndef QT_NO_LIBRARY
|
||
|
// if (qt_iconEngineFactoryLoader()->keyMap().key(QLatin1String("svg"), -1) != -1)
|
||
|
m_supportsSvg = true;
|
||
|
#endif //QT_NO_LIBRARY
|
||
|
}
|
||
|
}
|
||
|
|
||
8 years ago
|
XdgIconLoader *XdgIconLoader::instance()
|
||
10 years ago
|
{
|
||
|
iconLoaderInstance()->ensureInitialized();
|
||
|
return iconLoaderInstance();
|
||
|
}
|
||
|
|
||
|
// Queries the system theme and invalidates existing
|
||
|
// icons if the theme has changed.
|
||
8 years ago
|
void XdgIconLoader::updateSystemTheme()
|
||
10 years ago
|
{
|
||
|
// Only change if this is not explicitly set by the user
|
||
|
if (m_userTheme.isEmpty()) {
|
||
|
QString theme = systemThemeName();
|
||
|
if (theme.isEmpty())
|
||
|
theme = fallbackTheme();
|
||
|
if (theme != m_systemTheme) {
|
||
|
m_systemTheme = theme;
|
||
|
invalidateKey();
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
8 years ago
|
void XdgIconLoader::setThemeName(const QString &themeName)
|
||
10 years ago
|
{
|
||
|
m_userTheme = themeName;
|
||
|
invalidateKey();
|
||
|
}
|
||
|
|
||
8 years ago
|
void XdgIconLoader::setThemeSearchPath(const QStringList &searchPaths)
|
||
10 years ago
|
{
|
||
|
m_iconDirs = searchPaths;
|
||
|
themeList.clear();
|
||
|
invalidateKey();
|
||
|
}
|
||
|
|
||
8 years ago
|
QStringList XdgIconLoader::themeSearchPaths() const
|
||
10 years ago
|
{
|
||
|
if (m_iconDirs.isEmpty()) {
|
||
|
m_iconDirs = systemIconSearchPaths();
|
||
|
// Always add resource directory as search path
|
||
|
m_iconDirs.append(QLatin1String(":/icons"));
|
||
|
}
|
||
|
return m_iconDirs;
|
||
|
}
|
||
|
|
||
8 years ago
|
/*!
|
||
|
\class QIconCacheGtkReader
|
||
|
\internal
|
||
|
Helper class that reads and looks up into the icon-theme.cache generated with
|
||
|
gtk-update-icon-cache. If at any point we detect a corruption in the file
|
||
|
(because the offsets point at wrong locations for example), the reader
|
||
|
is marked as invalid.
|
||
|
*/
|
||
|
class QIconCacheGtkReader
|
||
|
{
|
||
|
public:
|
||
|
explicit QIconCacheGtkReader(const QString &themeDir);
|
||
|
QVector<const char *> lookup(const QString &);
|
||
|
bool isValid() const { return m_isValid; }
|
||
|
private:
|
||
|
QFile m_file;
|
||
|
const unsigned char *m_data;
|
||
|
quint64 m_size;
|
||
|
bool m_isValid;
|
||
|
|
||
|
quint16 read16(uint offset)
|
||
|
{
|
||
|
if (offset > m_size - 2 || (offset & 0x1)) {
|
||
|
m_isValid = false;
|
||
|
return 0;
|
||
|
}
|
||
|
return m_data[offset+1] | m_data[offset] << 8;
|
||
|
}
|
||
|
quint32 read32(uint offset)
|
||
|
{
|
||
|
if (offset > m_size - 4 || (offset & 0x3)) {
|
||
|
m_isValid = false;
|
||
|
return 0;
|
||
|
}
|
||
|
return m_data[offset+3] | m_data[offset+2] << 8
|
||
|
| m_data[offset+1] << 16 | m_data[offset] << 24;
|
||
|
}
|
||
|
};
|
||
|
|
||
|
|
||
|
QIconCacheGtkReader::QIconCacheGtkReader(const QString &dirName)
|
||
|
: m_isValid(false)
|
||
|
{
|
||
|
QFileInfo info(dirName + QLatin1Literal("/icon-theme.cache"));
|
||
|
if (!info.exists() || info.lastModified() < QFileInfo(dirName).lastModified())
|
||
|
return;
|
||
|
m_file.setFileName(info.absoluteFilePath());
|
||
|
if (!m_file.open(QFile::ReadOnly))
|
||
|
return;
|
||
|
m_size = m_file.size();
|
||
|
m_data = m_file.map(0, m_size);
|
||
|
if (!m_data)
|
||
|
return;
|
||
|
if (read16(0) != 1) // VERSION_MAJOR
|
||
|
return;
|
||
|
|
||
|
m_isValid = true;
|
||
|
|
||
|
// Check that all the directories are older than the cache
|
||
|
auto lastModified = info.lastModified();
|
||
|
quint32 dirListOffset = read32(8);
|
||
|
quint32 dirListLen = read32(dirListOffset);
|
||
|
for (uint i = 0; i < dirListLen; ++i) {
|
||
|
quint32 offset = read32(dirListOffset + 4 + 4 * i);
|
||
|
if (!m_isValid || offset >= m_size || lastModified < QFileInfo(dirName + QLatin1Char('/')
|
||
|
+ QString::fromUtf8(reinterpret_cast<const char*>(m_data + offset))).lastModified()) {
|
||
|
m_isValid = false;
|
||
|
return;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static quint32 icon_name_hash(const char *p)
|
||
|
{
|
||
|
quint32 h = static_cast<signed char>(*p);
|
||
|
for (p += 1; *p != '\0'; p++)
|
||
|
h = (h << 5) - h + *p;
|
||
|
return h;
|
||
|
}
|
||
|
|
||
|
/*! \internal
|
||
|
lookup the icon name and return the list of subdirectories in which an icon
|
||
|
with this name is present. The char* are pointers to the mapped data.
|
||
|
For example, this would return { "32x32/apps", "24x24/apps" , ... }
|
||
|
*/
|
||
|
QVector<const char *> QIconCacheGtkReader::lookup(const QString &name)
|
||
|
{
|
||
|
QVector<const char *> ret;
|
||
|
if (!isValid())
|
||
|
return ret;
|
||
|
|
||
|
QByteArray nameUtf8 = name.toUtf8();
|
||
|
quint32 hash = icon_name_hash(nameUtf8.data());
|
||
|
|
||
|
quint32 hashOffset = read32(4);
|
||
|
quint32 hashBucketCount = read32(hashOffset);
|
||
|
|
||
|
if (!isValid() || hashBucketCount == 0) {
|
||
|
m_isValid = false;
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
quint32 bucketIndex = hash % hashBucketCount;
|
||
|
quint32 bucketOffset = read32(hashOffset + 4 + bucketIndex * 4);
|
||
|
while (bucketOffset > 0 && bucketOffset <= m_size - 12) {
|
||
|
quint32 nameOff = read32(bucketOffset + 4);
|
||
|
if (nameOff < m_size && strcmp(reinterpret_cast<const char*>(m_data + nameOff), nameUtf8.constData()) == 0) {
|
||
|
quint32 dirListOffset = read32(8);
|
||
|
quint32 dirListLen = read32(dirListOffset);
|
||
|
|
||
|
quint32 listOffset = read32(bucketOffset+8);
|
||
|
quint32 listLen = read32(listOffset);
|
||
|
|
||
|
if (!m_isValid || listOffset + 4 + 8 * listLen > m_size) {
|
||
|
m_isValid = false;
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
ret.reserve(listLen);
|
||
|
for (uint j = 0; j < listLen && m_isValid; ++j) {
|
||
|
quint32 dirIndex = read16(listOffset + 4 + 8 * j);
|
||
|
quint32 o = read32(dirListOffset + 4 + dirIndex*4);
|
||
|
if (!m_isValid || dirIndex >= dirListLen || o >= m_size) {
|
||
|
m_isValid = false;
|
||
|
return ret;
|
||
|
}
|
||
|
ret.append(reinterpret_cast<const char*>(m_data) + o);
|
||
|
}
|
||
|
return ret;
|
||
|
}
|
||
|
bucketOffset = read32(bucketOffset);
|
||
|
}
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
10 years ago
|
QIconTheme::QIconTheme(const QString &themeName)
|
||
|
: m_valid(false)
|
||
|
{
|
||
|
QFile themeIndex;
|
||
|
|
||
|
QStringList iconDirs = QIcon::themeSearchPaths();
|
||
|
for ( int i = 0 ; i < iconDirs.size() ; ++i) {
|
||
|
QDir iconDir(iconDirs[i]);
|
||
|
QString themeDir = iconDir.path() + QLatin1Char('/') + themeName;
|
||
8 years ago
|
QFileInfo themeDirInfo(themeDir);
|
||
10 years ago
|
|
||
8 years ago
|
if (themeDirInfo.isDir()) {
|
||
|
m_contentDirs << themeDir;
|
||
|
m_gtkCaches << QSharedPointer<QIconCacheGtkReader>::create(themeDir);
|
||
|
}
|
||
10 years ago
|
|
||
8 years ago
|
if (!m_valid) {
|
||
|
themeIndex.setFileName(themeDir + QLatin1String("/index.theme"));
|
||
|
if (themeIndex.exists())
|
||
|
m_valid = true;
|
||
10 years ago
|
}
|
||
|
}
|
||
|
#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);
|
||
8 years ago
|
XdgIconDirInfo dirInfo(directoryKey);
|
||
10 years ago
|
dirInfo.size = size;
|
||
|
QString type = indexReader.value(directoryKey +
|
||
|
QLatin1String("/Type")
|
||
|
).toString();
|
||
|
|
||
|
if (type == QLatin1String("Fixed"))
|
||
8 years ago
|
dirInfo.type = XdgIconDirInfo::Fixed;
|
||
10 years ago
|
else if (type == QLatin1String("Scalable"))
|
||
8 years ago
|
dirInfo.type = XdgIconDirInfo::Scalable;
|
||
10 years ago
|
else
|
||
8 years ago
|
dirInfo.type = XdgIconDirInfo::Threshold;
|
||
10 years ago
|
|
||
|
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();
|
||
|
m_parents.removeAll(QString());
|
||
|
|
||
|
// Ensure a default platform fallback for all themes
|
||
|
if (m_parents.isEmpty()) {
|
||
|
const QString fallback = fallbackTheme();
|
||
|
if (!fallback.isEmpty())
|
||
|
m_parents.append(fallback);
|
||
|
}
|
||
|
|
||
|
// Ensure that all themes fall back to hicolor
|
||
|
if (!m_parents.contains(QLatin1String("hicolor")))
|
||
|
m_parents.append(QLatin1String("hicolor"));
|
||
|
}
|
||
|
#endif //QT_NO_SETTINGS
|
||
|
}
|
||
|
|
||
8 years ago
|
QThemeIconInfo XdgIconLoader::findIconHelper(const QString &themeName,
|
||
10 years ago
|
const QString &iconName,
|
||
|
QStringList &visited) const
|
||
|
{
|
||
10 years ago
|
QThemeIconInfo info;
|
||
10 years ago
|
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);
|
||
|
}
|
||
|
|
||
10 years ago
|
const QStringList contentDirs = theme.contentDirs();
|
||
10 years ago
|
|
||
|
const QString svgext(QLatin1String(".svg"));
|
||
|
const QString pngext(QLatin1String(".png"));
|
||
|
const QString xpmext(QLatin1String(".xpm"));
|
||
|
|
||
|
|
||
10 years ago
|
QString iconNameFallback = iconName;
|
||
|
|
||
|
// Iterate through all icon's fallbacks in current theme
|
||
|
while (info.entries.isEmpty()) {
|
||
|
const QString svgIconName = iconNameFallback + svgext;
|
||
|
const QString pngIconName = iconNameFallback + pngext;
|
||
|
const QString xpmIconName = iconNameFallback + xpmext;
|
||
|
|
||
|
// Add all relevant files
|
||
|
for (int i = 0; i < contentDirs.size(); ++i) {
|
||
8 years ago
|
QVector<XdgIconDirInfo> subDirs = theme.keyList();
|
||
|
|
||
|
// Try to reduce the amount of subDirs by looking in the GTK+ cache in order to save
|
||
|
// a massive amount of file stat (especially if the icon is not there)
|
||
|
auto cache = theme.m_gtkCaches.at(i);
|
||
|
if (cache->isValid()) {
|
||
|
auto result = cache->lookup(iconNameFallback);
|
||
|
if (cache->isValid()) {
|
||
|
const QVector<XdgIconDirInfo> subDirsCopy = subDirs;
|
||
|
subDirs.clear();
|
||
|
subDirs.reserve(result.count());
|
||
|
foreach (const char *s, result) {
|
||
|
QString path = QString::fromUtf8(s);
|
||
|
auto it = std::find_if(subDirsCopy.cbegin(), subDirsCopy.cend(),
|
||
|
[&](const XdgIconDirInfo &info) {
|
||
|
return info.path == path; } );
|
||
|
if (it != subDirsCopy.cend()) {
|
||
|
subDirs.append(*it);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
10 years ago
|
QString contentDir = contentDirs.at(i) + QLatin1Char('/');
|
||
|
for (int j = 0; j < subDirs.size() ; ++j) {
|
||
8 years ago
|
const XdgIconDirInfo &dirInfo = subDirs.at(j);
|
||
10 years ago
|
QString subdir = dirInfo.path;
|
||
|
QDir currentDir(contentDir + subdir);
|
||
|
if (currentDir.exists(pngIconName)) {
|
||
|
PixmapEntry *iconEntry = new PixmapEntry;
|
||
|
iconEntry->dir = dirInfo;
|
||
|
iconEntry->filename = currentDir.filePath(pngIconName);
|
||
|
// Notice we ensure that pixmap entries always come before
|
||
|
// scalable to preserve search order afterwards
|
||
|
info.entries.prepend(iconEntry);
|
||
|
} else if (m_supportsSvg &&
|
||
|
currentDir.exists(svgIconName)) {
|
||
|
ScalableEntry *iconEntry = new ScalableEntry;
|
||
|
iconEntry->dir = dirInfo;
|
||
|
iconEntry->filename = currentDir.filePath(svgIconName);
|
||
|
info.entries.append(iconEntry);
|
||
|
} else if(currentDir.exists(iconName + xpmext)) {
|
||
|
PixmapEntry *iconEntry = new PixmapEntry;
|
||
|
iconEntry->dir = dirInfo;
|
||
|
iconEntry->filename = currentDir.filePath(iconName + xpmext);
|
||
|
// Notice we ensure that pixmap entries always come before
|
||
|
// scalable to preserve search order afterwards
|
||
|
info.entries.append(iconEntry);
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if (!info.entries.isEmpty()) {
|
||
|
info.iconName = iconNameFallback;
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
// If it's possible - find next fallback for the icon
|
||
|
const int indexOfDash = iconNameFallback.lastIndexOf(QLatin1Char('-'));
|
||
|
if (indexOfDash == -1)
|
||
|
break;
|
||
|
|
||
|
iconNameFallback.truncate(indexOfDash);
|
||
|
}
|
||
|
|
||
|
if (info.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
|
||
|
info = findIconHelper(parentTheme, iconName, visited);
|
||
|
|
||
|
if (!info.entries.isEmpty()) // success
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if (info.entries.isEmpty()) {
|
||
|
// Search for unthemed icons in main dir of search paths
|
||
|
QStringList themeSearchPaths = QIcon::themeSearchPaths();
|
||
|
foreach (QString contentDir, themeSearchPaths) {
|
||
|
QDir currentDir(contentDir);
|
||
10 years ago
|
|
||
|
if (currentDir.exists(iconName + pngext)) {
|
||
|
PixmapEntry *iconEntry = new PixmapEntry;
|
||
|
iconEntry->filename = currentDir.filePath(iconName + pngext);
|
||
|
// Notice we ensure that pixmap entries always come before
|
||
|
// scalable to preserve search order afterwards
|
||
10 years ago
|
info.entries.prepend(iconEntry);
|
||
10 years ago
|
} else if (m_supportsSvg &&
|
||
|
currentDir.exists(iconName + svgext)) {
|
||
|
ScalableEntry *iconEntry = new ScalableEntry;
|
||
|
iconEntry->filename = currentDir.filePath(iconName + svgext);
|
||
10 years ago
|
info.entries.append(iconEntry);
|
||
10 years ago
|
break;
|
||
|
} else if (currentDir.exists(iconName + xpmext)) {
|
||
|
PixmapEntry *iconEntry = new PixmapEntry;
|
||
|
iconEntry->filename = currentDir.filePath(iconName + xpmext);
|
||
|
// Notice we ensure that pixmap entries always come before
|
||
|
// scalable to preserve search order afterwards
|
||
10 years ago
|
info.entries.append(iconEntry);
|
||
10 years ago
|
break;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
/*********************************************************************
|
||
|
Author: Kaitlin Rupert <kaitlin.rupert@intel.com>
|
||
|
Date: Aug 12, 2010
|
||
|
Description: Make it so that the QIcon loader honors /usr/share/pixmaps
|
||
|
directory. This is a valid directory per the Freedesktop.org
|
||
|
icon theme specification.
|
||
|
Bug: https://bugreports.qt.nokia.com/browse/QTBUG-12874
|
||
|
*********************************************************************/
|
||
|
#ifdef Q_OS_LINUX
|
||
|
/* Freedesktop standard says to look in /usr/share/pixmaps last */
|
||
10 years ago
|
if (info.entries.isEmpty()) {
|
||
10 years ago
|
const QString pixmaps(QLatin1String("/usr/share/pixmaps"));
|
||
|
|
||
10 years ago
|
const QDir currentDir(pixmaps);
|
||
8 years ago
|
const XdgIconDirInfo dirInfo(pixmaps);
|
||
10 years ago
|
if (currentDir.exists(iconName + pngext)) {
|
||
|
PixmapEntry *iconEntry = new PixmapEntry;
|
||
|
iconEntry->dir = dirInfo;
|
||
|
iconEntry->filename = currentDir.filePath(iconName + pngext);
|
||
|
// Notice we ensure that pixmap entries always come before
|
||
|
// scalable to preserve search order afterwards
|
||
10 years ago
|
info.entries.prepend(iconEntry);
|
||
10 years ago
|
} else if (m_supportsSvg &&
|
||
|
currentDir.exists(iconName + svgext)) {
|
||
|
ScalableEntry *iconEntry = new ScalableEntry;
|
||
|
iconEntry->dir = dirInfo;
|
||
|
iconEntry->filename = currentDir.filePath(iconName + svgext);
|
||
10 years ago
|
info.entries.append(iconEntry);
|
||
10 years ago
|
} else if (currentDir.exists(iconName + xpmext)) {
|
||
|
PixmapEntry *iconEntry = new PixmapEntry;
|
||
|
iconEntry->dir = dirInfo;
|
||
|
iconEntry->filename = currentDir.filePath(iconName + xpmext);
|
||
|
// Notice we ensure that pixmap entries always come before
|
||
|
// scalable to preserve search order afterwards
|
||
10 years ago
|
info.entries.append(iconEntry);
|
||
10 years ago
|
}
|
||
|
}
|
||
|
#endif
|
||
|
|
||
10 years ago
|
return info;
|
||
10 years ago
|
}
|
||
|
|
||
8 years ago
|
QThemeIconInfo XdgIconLoader::loadIcon(const QString &name) const
|
||
10 years ago
|
{
|
||
|
if (!themeName().isEmpty()) {
|
||
|
QStringList visited;
|
||
|
return findIconHelper(themeName(), name, visited);
|
||
|
}
|
||
|
|
||
10 years ago
|
return QThemeIconInfo();
|
||
10 years ago
|
}
|
||
|
|
||
|
|
||
|
// -------- Icon Loader Engine -------- //
|
||
|
|
||
|
|
||
8 years ago
|
XdgIconLoaderEngine::XdgIconLoaderEngine(const QString& iconName)
|
||
10 years ago
|
: m_iconName(iconName), m_key(0)
|
||
|
{
|
||
|
}
|
||
|
|
||
8 years ago
|
XdgIconLoaderEngine::~XdgIconLoaderEngine()
|
||
10 years ago
|
{
|
||
10 years ago
|
qDeleteAll(m_info.entries);
|
||
10 years ago
|
}
|
||
|
|
||
8 years ago
|
XdgIconLoaderEngine::XdgIconLoaderEngine(const XdgIconLoaderEngine &other)
|
||
10 years ago
|
: QIconEngine(other),
|
||
|
m_iconName(other.m_iconName),
|
||
|
m_key(0)
|
||
|
{
|
||
|
}
|
||
|
|
||
8 years ago
|
QIconEngine *XdgIconLoaderEngine::clone() const
|
||
10 years ago
|
{
|
||
8 years ago
|
return new XdgIconLoaderEngine(*this);
|
||
10 years ago
|
}
|
||
|
|
||
8 years ago
|
bool XdgIconLoaderEngine::read(QDataStream &in) {
|
||
10 years ago
|
in >> m_iconName;
|
||
|
return true;
|
||
|
}
|
||
|
|
||
8 years ago
|
bool XdgIconLoaderEngine::write(QDataStream &out) const
|
||
10 years ago
|
{
|
||
|
out << m_iconName;
|
||
|
return true;
|
||
|
}
|
||
|
|
||
8 years ago
|
bool XdgIconLoaderEngine::hasIcon() const
|
||
10 years ago
|
{
|
||
10 years ago
|
return !(m_info.entries.isEmpty());
|
||
10 years ago
|
}
|
||
|
|
||
|
// Lazily load the icon
|
||
8 years ago
|
void XdgIconLoaderEngine::ensureLoaded()
|
||
10 years ago
|
{
|
||
8 years ago
|
if (!(XdgIconLoader::instance()->themeKey() == m_key)) {
|
||
10 years ago
|
qDeleteAll(m_info.entries);
|
||
|
m_info.entries.clear();
|
||
|
m_info.iconName.clear();
|
||
10 years ago
|
|
||
10 years ago
|
Q_ASSERT(m_info.entries.size() == 0);
|
||
8 years ago
|
m_info = XdgIconLoader::instance()->loadIcon(m_iconName);
|
||
|
m_key = XdgIconLoader::instance()->themeKey();
|
||
10 years ago
|
}
|
||
|
}
|
||
|
|
||
8 years ago
|
void XdgIconLoaderEngine::paint(QPainter *painter, const QRect &rect,
|
||
10 years ago
|
QIcon::Mode mode, QIcon::State state)
|
||
|
{
|
||
|
QSize pixmapSize = rect.size();
|
||
8 years ago
|
#if defined(Q_DEAD_CODE_FROM_QT4_MAC)
|
||
10 years ago
|
pixmapSize *= qt_mac_get_scalefactor();
|
||
|
#endif
|
||
|
painter->drawPixmap(rect, pixmap(pixmapSize, mode, state));
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
* This algorithm is defined by the freedesktop spec:
|
||
|
* http://standards.freedesktop.org/icon-theme-spec/icon-theme-spec-latest.html
|
||
|
*/
|
||
8 years ago
|
static bool directoryMatchesSize(const XdgIconDirInfo &dir, int iconsize)
|
||
10 years ago
|
{
|
||
8 years ago
|
if (dir.type == XdgIconDirInfo::Fixed) {
|
||
10 years ago
|
return dir.size == iconsize;
|
||
|
|
||
8 years ago
|
} else if (dir.type == XdgIconDirInfo::Scalable) {
|
||
|
return iconsize <= dir.maxSize &&
|
||
10 years ago
|
iconsize >= dir.minSize;
|
||
|
|
||
8 years ago
|
} else if (dir.type == XdgIconDirInfo::Threshold) {
|
||
10 years ago
|
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
|
||
|
*/
|
||
8 years ago
|
static int directorySizeDistance(const XdgIconDirInfo &dir, int iconsize)
|
||
10 years ago
|
{
|
||
8 years ago
|
if (dir.type == XdgIconDirInfo::Fixed) {
|
||
10 years ago
|
return qAbs(dir.size - iconsize);
|
||
|
|
||
8 years ago
|
} else if (dir.type == XdgIconDirInfo::Scalable) {
|
||
10 years ago
|
if (iconsize < dir.minSize)
|
||
|
return dir.minSize - iconsize;
|
||
|
else if (iconsize > dir.maxSize)
|
||
|
return iconsize - dir.maxSize;
|
||
|
else
|
||
|
return 0;
|
||
|
|
||
8 years ago
|
} else if (dir.type == XdgIconDirInfo::Threshold) {
|
||
10 years ago
|
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;
|
||
|
}
|
||
|
|
||
8 years ago
|
XdgIconLoaderEngineEntry *XdgIconLoaderEngine::entryForSize(const QSize &size)
|
||
10 years ago
|
{
|
||
|
int iconsize = qMin(size.width(), size.height());
|
||
|
|
||
10 years ago
|
// Note that m_info.entries are sorted so that png-files
|
||
10 years ago
|
// come first
|
||
|
|
||
10 years ago
|
const int numEntries = m_info.entries.size();
|
||
10 years ago
|
|
||
|
// Search for exact matches first
|
||
|
for (int i = 0; i < numEntries; ++i) {
|
||
8 years ago
|
XdgIconLoaderEngineEntry *entry = m_info.entries.at(i);
|
||
10 years ago
|
if (directoryMatchesSize(entry->dir, iconsize)) {
|
||
|
return entry;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Find the minimum distance icon
|
||
|
int minimalSize = INT_MAX;
|
||
8 years ago
|
XdgIconLoaderEngineEntry *closestMatch = 0;
|
||
10 years ago
|
for (int i = 0; i < numEntries; ++i) {
|
||
8 years ago
|
XdgIconLoaderEngineEntry *entry = m_info.entries.at(i);
|
||
10 years ago
|
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.
|
||
|
*
|
||
|
*/
|
||
8 years ago
|
QSize XdgIconLoaderEngine::actualSize(const QSize &size, QIcon::Mode mode,
|
||
10 years ago
|
QIcon::State state)
|
||
|
{
|
||
|
ensureLoaded();
|
||
|
|
||
8 years ago
|
XdgIconLoaderEngineEntry *entry = entryForSize(size);
|
||
10 years ago
|
if (entry) {
|
||
8 years ago
|
const XdgIconDirInfo &dir = entry->dir;
|
||
|
if (dir.type == XdgIconDirInfo::Scalable || dynamic_cast<ScalableEntry *>(entry))
|
||
10 years ago
|
return size;
|
||
|
else {
|
||
8 years ago
|
int dir_size = dir.size;
|
||
|
//Note: fallback for directories that don't have its content size defined
|
||
|
// -> get the actual size based on the image if possible
|
||
|
PixmapEntry * pix_e;
|
||
|
if (0 == dir_size && nullptr != (pix_e = dynamic_cast<PixmapEntry *>(entry)))
|
||
|
{
|
||
|
QSize pix_size = pix_e->basePixmap.size();
|
||
|
dir_size = qMin(pix_size.width(), pix_size.height());
|
||
|
}
|
||
|
int result = qMin(dir_size, qMin(size.width(), size.height()));
|
||
10 years ago
|
return QSize(result, result);
|
||
|
}
|
||
|
}
|
||
|
return QIconEngine::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);
|
||
|
|
||
|
QSize actualSize = basePixmap.size();
|
||
|
if (!actualSize.isNull() && (actualSize.width() > size.width() || actualSize.height() > size.height()))
|
||
|
actualSize.scale(size, Qt::KeepAspectRatio);
|
||
|
|
||
|
QString key = QLatin1String("$qt_theme_")
|
||
|
% HexString<qint64>(basePixmap.cacheKey())
|
||
|
% HexString<int>(mode)
|
||
|
% HexString<qint64>(QGuiApplication::palette().cacheKey())
|
||
|
% HexString<int>(actualSize.width())
|
||
|
% HexString<int>(actualSize.height());
|
||
|
|
||
|
QPixmap cachedPixmap;
|
||
|
if (QPixmapCache::find(key, &cachedPixmap)) {
|
||
|
return cachedPixmap;
|
||
|
} else {
|
||
|
if (basePixmap.size() != actualSize)
|
||
|
cachedPixmap = basePixmap.scaled(actualSize, Qt::IgnoreAspectRatio, Qt::SmoothTransformation);
|
||
|
else
|
||
|
cachedPixmap = basePixmap;
|
||
|
if (QGuiApplication *guiApp = qobject_cast<QGuiApplication *>(qApp))
|
||
|
cachedPixmap = static_cast<QGuiApplicationPrivate*>(QObjectPrivate::get(guiApp))->applyQIconStyleHelper(mode, cachedPixmap);
|
||
|
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);
|
||
|
}
|
||
|
|
||
8 years ago
|
QPixmap XdgIconLoaderEngine::pixmap(const QSize &size, QIcon::Mode mode,
|
||
10 years ago
|
QIcon::State state)
|
||
|
{
|
||
|
ensureLoaded();
|
||
|
|
||
8 years ago
|
XdgIconLoaderEngineEntry *entry = entryForSize(size);
|
||
10 years ago
|
if (entry)
|
||
|
return entry->pixmap(size, mode, state);
|
||
|
|
||
|
return QPixmap();
|
||
|
}
|
||
|
|
||
8 years ago
|
QString XdgIconLoaderEngine::key() const
|
||
10 years ago
|
{
|
||
8 years ago
|
return QLatin1String("XdgIconLoaderEngine");
|
||
10 years ago
|
}
|
||
|
|
||
8 years ago
|
void XdgIconLoaderEngine::virtual_hook(int id, void *data)
|
||
10 years ago
|
{
|
||
|
ensureLoaded();
|
||
|
|
||
|
switch (id) {
|
||
|
case QIconEngine::AvailableSizesHook:
|
||
|
{
|
||
|
QIconEngine::AvailableSizesArgument &arg
|
||
|
= *reinterpret_cast<QIconEngine::AvailableSizesArgument*>(data);
|
||
10 years ago
|
const int N = m_info.entries.size();
|
||
10 years ago
|
QList<QSize> sizes;
|
||
|
sizes.reserve(N);
|
||
|
|
||
|
// Gets all sizes from the DirectoryInfo entries
|
||
|
for (int i = 0; i < N; ++i) {
|
||
10 years ago
|
int size = m_info.entries.at(i)->dir.size;
|
||
10 years ago
|
sizes.append(QSize(size, size));
|
||
|
}
|
||
|
arg.sizes.swap(sizes); // commit
|
||
|
}
|
||
|
break;
|
||
|
case QIconEngine::IconNameHook:
|
||
|
{
|
||
|
QString &name = *reinterpret_cast<QString*>(data);
|
||
8 years ago
|
name = m_info.iconName;
|
||
|
}
|
||
|
break;
|
||
|
#if QT_VERSION >= QT_VERSION_CHECK(5, 7, 0)
|
||
|
case QIconEngine::IsNullHook:
|
||
|
{
|
||
|
*reinterpret_cast<bool*>(data) = m_info.entries.isEmpty();
|
||
10 years ago
|
}
|
||
|
break;
|
||
8 years ago
|
#endif
|
||
10 years ago
|
default:
|
||
|
QIconEngine::virtual_hook(id, data);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
//QT_END_NAMESPACE
|
||
|
|
||
|
#endif //QT_NO_ICON
|