2008-11-04 00:08:39 +01:00
|
|
|
/*
|
|
|
|
* Copyright 2006-2007 Aaron Seigo <aseigo@kde.org>
|
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU Library General Public License as
|
|
|
|
* published by the Free Software Foundation; either version 2, 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 Library General Public
|
|
|
|
* License along with this program; if not, write to the
|
|
|
|
* Free Software Foundation, Inc.,
|
|
|
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "theme.h"
|
2013-03-11 19:24:07 +01:00
|
|
|
#include "private/theme_p.h"
|
2008-11-04 00:08:39 +01:00
|
|
|
|
|
|
|
#include <QApplication>
|
|
|
|
#include <QFile>
|
2013-03-13 10:47:10 +01:00
|
|
|
#include <QFont>
|
2008-11-09 21:41:27 +01:00
|
|
|
#include <QFileInfo>
|
2010-10-30 00:14:18 +02:00
|
|
|
#include <QMutableListIterator>
|
2009-07-20 22:30:07 +02:00
|
|
|
#include <QPair>
|
2010-09-22 09:46:09 +02:00
|
|
|
#include <QStringBuilder>
|
|
|
|
#include <QTimer>
|
2012-08-03 12:19:50 +02:00
|
|
|
|
2012-08-25 03:00:36 +02:00
|
|
|
#include "config-plasma.h"
|
2012-08-03 12:19:50 +02:00
|
|
|
|
2008-11-04 03:04:34 +01:00
|
|
|
#include <kcolorscheme.h>
|
|
|
|
#include <kconfiggroup.h>
|
2013-07-29 19:05:59 +02:00
|
|
|
#include <QDebug>
|
2009-01-26 07:22:44 +01:00
|
|
|
#include <kdirwatch.h>
|
2008-11-04 03:55:37 +01:00
|
|
|
#include <kmanagerselection.h>
|
2010-05-04 05:59:35 +02:00
|
|
|
#include <kimagecache.h>
|
2008-11-04 03:04:34 +01:00
|
|
|
#include <ksharedconfig.h>
|
2012-02-19 13:41:26 +01:00
|
|
|
#include <kwindoweffects.h>
|
2008-11-04 03:04:34 +01:00
|
|
|
#include <kwindowsystem.h>
|
2012-06-12 12:59:34 +02:00
|
|
|
#include <qstandardpaths.h>
|
2008-11-04 00:08:39 +01:00
|
|
|
|
2010-04-28 19:40:29 +02:00
|
|
|
#include "private/packages_p.h"
|
2008-11-04 00:08:39 +01:00
|
|
|
|
|
|
|
namespace Plasma
|
|
|
|
{
|
|
|
|
|
|
|
|
Theme::Theme(QObject *parent)
|
2013-03-12 13:03:06 +01:00
|
|
|
: QObject(parent)
|
2008-11-04 00:08:39 +01:00
|
|
|
{
|
2013-03-11 22:26:15 +01:00
|
|
|
if (!ThemePrivate::globalTheme) {
|
|
|
|
ThemePrivate::globalTheme = new ThemePrivate;
|
|
|
|
}
|
2013-03-12 16:29:04 +01:00
|
|
|
ThemePrivate::globalThemeRefCount.ref();
|
2013-03-11 22:26:15 +01:00
|
|
|
d = ThemePrivate::globalTheme;
|
|
|
|
|
2013-03-11 18:08:03 +01:00
|
|
|
d->settingsChanged();
|
2010-03-04 23:28:39 +01:00
|
|
|
if (QCoreApplication::instance()) {
|
|
|
|
connect(QCoreApplication::instance(), SIGNAL(aboutToQuit()),
|
2013-03-12 16:29:04 +01:00
|
|
|
d, SLOT(onAppExitCleanup()));
|
2010-03-04 23:28:39 +01:00
|
|
|
}
|
2013-03-11 20:20:50 +01:00
|
|
|
connect(d, &ThemePrivate::themeChanged, this, &Theme::themeChanged);
|
2009-05-18 14:28:19 +02:00
|
|
|
}
|
2008-11-04 00:08:39 +01:00
|
|
|
|
2009-05-18 14:28:19 +02:00
|
|
|
Theme::Theme(const QString &themeName, QObject *parent)
|
2013-03-11 22:26:15 +01:00
|
|
|
: QObject(parent)
|
2009-05-18 14:28:19 +02:00
|
|
|
{
|
2013-03-11 22:26:15 +01:00
|
|
|
if (!ThemePrivate::themes.contains(themeName)) {
|
|
|
|
ThemePrivate::themes[themeName] = new ThemePrivate;
|
2013-03-12 16:29:04 +01:00
|
|
|
ThemePrivate::themesRefCount[themeName] = QAtomicInt();
|
2013-03-11 22:26:15 +01:00
|
|
|
}
|
|
|
|
|
2013-03-12 16:29:04 +01:00
|
|
|
ThemePrivate::themesRefCount[themeName].ref();
|
2013-03-11 22:26:15 +01:00
|
|
|
d = ThemePrivate::themes[themeName];
|
|
|
|
|
2009-05-18 14:28:19 +02:00
|
|
|
// turn off caching so we don't accidently trigger unnecessary disk activity at this point
|
|
|
|
bool useCache = d->cacheTheme;
|
|
|
|
d->cacheTheme = false;
|
2013-03-11 22:26:15 +01:00
|
|
|
d->setThemeName(themeName, true);
|
2009-05-18 14:28:19 +02:00
|
|
|
d->cacheTheme = useCache;
|
2010-03-04 23:28:39 +01:00
|
|
|
if (QCoreApplication::instance()) {
|
|
|
|
connect(QCoreApplication::instance(), SIGNAL(aboutToQuit()),
|
2013-03-12 16:29:04 +01:00
|
|
|
d, SLOT(onAppExitCleanup()));
|
2010-03-04 23:28:39 +01:00
|
|
|
}
|
2013-03-11 22:39:15 +01:00
|
|
|
connect(d, &ThemePrivate::themeChanged, this, &Theme::themeChanged);
|
2008-11-04 00:08:39 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
Theme::~Theme()
|
|
|
|
{
|
2011-01-30 22:47:29 +01:00
|
|
|
if (d->svgElementsCache) {
|
|
|
|
QHashIterator<QString, QSet<QString> > it(d->invalidElements);
|
|
|
|
while (it.hasNext()) {
|
|
|
|
it.next();
|
|
|
|
KConfigGroup imageGroup(d->svgElementsCache, it.key());
|
|
|
|
imageGroup.writeEntry("invalidElements", it.value().toList()); //FIXME: add QSet support to KConfig
|
|
|
|
}
|
2008-12-11 00:17:12 +01:00
|
|
|
}
|
|
|
|
|
2011-01-30 22:47:29 +01:00
|
|
|
d->onAppExitCleanup();
|
2013-03-11 22:26:15 +01:00
|
|
|
|
|
|
|
if (d == ThemePrivate::globalTheme) {
|
2013-03-12 16:29:04 +01:00
|
|
|
if (!ThemePrivate::globalThemeRefCount.deref()) {
|
|
|
|
disconnect(ThemePrivate::globalTheme, 0, this, 0);
|
2013-03-11 22:26:15 +01:00
|
|
|
delete ThemePrivate::globalTheme;
|
2013-03-12 16:29:04 +01:00
|
|
|
ThemePrivate::globalTheme = 0;
|
|
|
|
d = 0;
|
2013-03-11 22:26:15 +01:00
|
|
|
}
|
|
|
|
} else {
|
2013-03-12 16:29:04 +01:00
|
|
|
if (!ThemePrivate::themesRefCount[d->themeName].deref()) {
|
2013-03-11 22:26:15 +01:00
|
|
|
ThemePrivate *themePrivate = ThemePrivate::themes[d->themeName];
|
|
|
|
ThemePrivate::themes.remove(d->themeName);
|
|
|
|
ThemePrivate::themesRefCount.remove(d->themeName);
|
|
|
|
delete themePrivate;
|
|
|
|
}
|
|
|
|
}
|
2008-11-04 00:08:39 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void Theme::setThemeName(const QString &themeName)
|
2009-02-24 21:49:46 +01:00
|
|
|
{
|
2013-03-11 22:26:15 +01:00
|
|
|
if (d->themeName == themeName) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (d != ThemePrivate::globalTheme) {
|
2013-03-12 16:29:04 +01:00
|
|
|
disconnect(QCoreApplication::instance(), 0, d, 0);
|
|
|
|
if (!ThemePrivate::themesRefCount[d->themeName].deref()) {
|
2013-03-11 22:26:15 +01:00
|
|
|
ThemePrivate *themePrivate = ThemePrivate::themes[d->themeName];
|
|
|
|
ThemePrivate::themes.remove(d->themeName);
|
|
|
|
ThemePrivate::themesRefCount.remove(d->themeName);
|
|
|
|
delete themePrivate;
|
|
|
|
}
|
2013-03-11 22:39:15 +01:00
|
|
|
if (!ThemePrivate::themes.contains(themeName)) {
|
|
|
|
ThemePrivate::themes[themeName] = new ThemePrivate;
|
2013-03-12 16:29:04 +01:00
|
|
|
ThemePrivate::themesRefCount[themeName] = QAtomicInt();
|
|
|
|
}
|
|
|
|
ThemePrivate::themesRefCount[themeName].ref();
|
|
|
|
d = ThemePrivate::themes[themeName];
|
|
|
|
if (QCoreApplication::instance()) {
|
|
|
|
connect(QCoreApplication::instance(), SIGNAL(aboutToQuit()),
|
|
|
|
d, SLOT(onAppExitCleanup()));
|
2013-03-11 22:39:15 +01:00
|
|
|
}
|
|
|
|
connect(d, &ThemePrivate::themeChanged, this, &Theme::themeChanged);
|
2013-03-11 22:26:15 +01:00
|
|
|
}
|
|
|
|
|
2009-02-24 21:49:46 +01:00
|
|
|
d->setThemeName(themeName, true);
|
|
|
|
}
|
|
|
|
|
2008-11-04 00:08:39 +01:00
|
|
|
QString Theme::themeName() const
|
|
|
|
{
|
|
|
|
return d->themeName;
|
|
|
|
}
|
|
|
|
|
|
|
|
QString Theme::imagePath(const QString &name) const
|
|
|
|
{
|
|
|
|
// look for a compressed svg file in the theme
|
2009-05-19 10:53:56 +02:00
|
|
|
if (name.contains("../") || name.isEmpty()) {
|
2008-11-04 00:08:39 +01:00
|
|
|
// we don't support relative paths
|
2013-07-29 19:05:59 +02:00
|
|
|
//qDebug() << "Theme says: bad image path " << name;
|
2008-11-04 00:08:39 +01:00
|
|
|
return QString();
|
|
|
|
}
|
|
|
|
|
2010-10-04 23:50:42 +02:00
|
|
|
const QString svgzName = name % QLatin1Literal(".svgz");
|
|
|
|
QString path = d->findInTheme(svgzName, d->themeName);
|
2008-11-04 00:08:39 +01:00
|
|
|
|
|
|
|
if (path.isEmpty()) {
|
|
|
|
// try for an uncompressed svg file
|
2010-10-04 23:50:42 +02:00
|
|
|
const QString svgName = name % QLatin1Literal(".svg");
|
|
|
|
path = d->findInTheme(svgName, d->themeName);
|
2008-11-04 00:08:39 +01:00
|
|
|
|
2009-06-26 12:02:32 +02:00
|
|
|
// search in fallback themes if necessary
|
|
|
|
for (int i = 0; path.isEmpty() && i < d->fallbackThemes.count(); ++i) {
|
|
|
|
if (d->themeName == d->fallbackThemes[i]) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// try a compressed svg file in the fallback theme
|
2010-10-07 15:51:31 +02:00
|
|
|
path = d->findInTheme(svgzName, d->fallbackThemes[i]);
|
2008-11-04 00:08:39 +01:00
|
|
|
|
|
|
|
if (path.isEmpty()) {
|
2009-06-26 12:02:32 +02:00
|
|
|
// try an uncompressed svg file in the fallback theme
|
2010-10-04 23:50:42 +02:00
|
|
|
path = d->findInTheme(svgName, d->fallbackThemes[i]);
|
2008-11-04 00:08:39 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-04-28 19:40:29 +02:00
|
|
|
/*
|
2008-11-04 00:08:39 +01:00
|
|
|
if (path.isEmpty()) {
|
2011-07-29 15:46:52 +02:00
|
|
|
#ifndef NDEBUG
|
2013-07-29 19:05:59 +02:00
|
|
|
// qDebug() << "Theme says: bad image path " << name;
|
2011-07-29 15:46:52 +02:00
|
|
|
#endif
|
2008-11-04 00:08:39 +01:00
|
|
|
}
|
2010-04-28 19:40:29 +02:00
|
|
|
*/
|
2008-11-04 00:08:39 +01:00
|
|
|
|
|
|
|
return path;
|
|
|
|
}
|
|
|
|
|
2010-05-11 10:23:41 +02:00
|
|
|
QString Theme::styleSheet(const QString &css) const
|
|
|
|
{
|
|
|
|
return d->processStyleSheet(css);
|
|
|
|
}
|
|
|
|
|
2008-11-04 00:08:39 +01:00
|
|
|
QString Theme::wallpaperPath(const QSize &size) const
|
|
|
|
{
|
|
|
|
QString fullPath;
|
|
|
|
QString image = d->defaultWallpaperTheme;
|
|
|
|
|
|
|
|
image.append("/contents/images/%1x%2").append(d->defaultWallpaperSuffix);
|
|
|
|
QString defaultImage = image.arg(d->defaultWallpaperWidth).arg(d->defaultWallpaperHeight);
|
|
|
|
|
|
|
|
if (size.isValid()) {
|
|
|
|
// try to customize the paper to the size requested
|
|
|
|
//TODO: this should do better than just fallback to the default size.
|
|
|
|
// a "best fit" matching would be far better, so we don't end
|
|
|
|
// up returning a 1920x1200 wallpaper for a 640x480 request ;)
|
|
|
|
image = image.arg(size.width()).arg(size.height());
|
|
|
|
} else {
|
|
|
|
image = defaultImage;
|
|
|
|
}
|
|
|
|
|
|
|
|
//TODO: the theme's wallpaper overrides regularly installed wallpapers.
|
|
|
|
// should it be possible for user installed (e.g. locateLocal) wallpapers
|
|
|
|
// to override the theme?
|
|
|
|
if (d->hasWallpapers) {
|
|
|
|
// check in the theme first
|
2010-10-02 22:06:47 +02:00
|
|
|
fullPath = d->findInTheme(QLatin1Literal("wallpapers/") % image, d->themeName);
|
2008-11-04 00:08:39 +01:00
|
|
|
|
|
|
|
if (fullPath.isEmpty()) {
|
2010-10-02 22:06:47 +02:00
|
|
|
fullPath = d->findInTheme(QLatin1Literal("wallpapers/") % defaultImage, d->themeName);
|
2008-11-04 00:08:39 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (fullPath.isEmpty()) {
|
|
|
|
// we failed to find it in the theme, so look in the standard directories
|
2013-07-29 19:05:59 +02:00
|
|
|
//qDebug() << "looking for" << image;
|
2012-06-12 12:59:34 +02:00
|
|
|
fullPath = QStandardPaths::locate(QStandardPaths::GenericDataLocation, QLatin1String("wallpapers/") + image);
|
2008-11-04 00:08:39 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if (fullPath.isEmpty()) {
|
|
|
|
// we still failed to find it in the theme, so look for the default in
|
|
|
|
// the standard directories
|
2013-07-29 19:05:59 +02:00
|
|
|
//qDebug() << "looking for" << defaultImage;
|
2012-06-12 12:59:34 +02:00
|
|
|
fullPath = QStandardPaths::locate(QStandardPaths::GenericDataLocation, QLatin1String("wallpapers/") + defaultImage);
|
2008-11-04 00:08:39 +01:00
|
|
|
|
|
|
|
if (fullPath.isEmpty()) {
|
2011-07-29 15:46:52 +02:00
|
|
|
#ifndef NDEBUG
|
2013-07-29 19:05:59 +02:00
|
|
|
// qDebug() << "exhausted every effort to find a wallpaper.";
|
2011-07-29 15:46:52 +02:00
|
|
|
#endif
|
2008-11-04 00:08:39 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return fullPath;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Theme::currentThemeHasImage(const QString &name) const
|
|
|
|
{
|
|
|
|
if (name.contains("../")) {
|
|
|
|
// we don't support relative paths
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2011-07-22 17:34:46 +02:00
|
|
|
return !(d->findInTheme(name % QLatin1Literal(".svgz"), d->themeName, false).isEmpty()) ||
|
|
|
|
!(d->findInTheme(name % QLatin1Literal(".svg"), d->themeName, false).isEmpty());
|
2008-11-04 00:08:39 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
KSharedConfigPtr Theme::colorScheme() const
|
|
|
|
{
|
|
|
|
return d->colors;
|
|
|
|
}
|
|
|
|
|
|
|
|
QColor Theme::color(ColorRole role) const
|
|
|
|
{
|
2013-03-11 20:20:50 +01:00
|
|
|
return d->color(role);
|
2008-11-04 00:08:39 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void Theme::setUseGlobalSettings(bool useGlobal)
|
|
|
|
{
|
|
|
|
if (d->useGlobal == useGlobal) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
d->useGlobal = useGlobal;
|
|
|
|
d->cfg = KConfigGroup();
|
|
|
|
d->themeName.clear();
|
2013-03-11 18:08:03 +01:00
|
|
|
d->settingsChanged();
|
2008-11-04 00:08:39 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
bool Theme::useGlobalSettings() const
|
|
|
|
{
|
|
|
|
return d->useGlobal;
|
|
|
|
}
|
|
|
|
|
2011-05-20 08:23:43 +02:00
|
|
|
bool Theme::findInCache(const QString &key, QPixmap &pix, unsigned int lastModified)
|
2008-11-04 00:08:39 +01:00
|
|
|
{
|
2013-06-05 19:41:58 +02:00
|
|
|
if (lastModified != 0 && d->useCache() && lastModified > uint(d->pixmapCache->lastModifiedTime().toTime_t())) {
|
2011-05-20 08:23:43 +02:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2009-05-30 05:08:33 +02:00
|
|
|
if (d->useCache()) {
|
2009-08-04 21:13:58 +02:00
|
|
|
const QString id = d->keysToCache.value(key);
|
2009-07-20 22:30:07 +02:00
|
|
|
if (d->pixmapsToCache.contains(id)) {
|
|
|
|
pix = d->pixmapsToCache.value(id);
|
2010-12-03 18:09:09 +01:00
|
|
|
return !pix.isNull();
|
2009-05-30 05:08:33 +02:00
|
|
|
}
|
|
|
|
|
2010-05-04 05:59:35 +02:00
|
|
|
QPixmap temp;
|
2010-12-03 18:09:09 +01:00
|
|
|
if (d->pixmapCache->findPixmap(key, &temp) && !temp.isNull()) {
|
2010-05-04 05:59:35 +02:00
|
|
|
pix = temp;
|
|
|
|
return true;
|
|
|
|
}
|
2009-05-30 05:08:33 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
2008-11-04 00:08:39 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void Theme::insertIntoCache(const QString& key, const QPixmap& pix)
|
|
|
|
{
|
2008-12-04 21:13:50 +01:00
|
|
|
if (d->useCache()) {
|
2010-05-04 05:59:35 +02:00
|
|
|
d->pixmapCache->insertPixmap(key, pix);
|
2009-07-20 22:30:07 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Theme::insertIntoCache(const QString& key, const QPixmap& pix, const QString& id)
|
|
|
|
{
|
|
|
|
if (d->useCache()) {
|
|
|
|
d->pixmapsToCache.insert(id, pix);
|
|
|
|
|
|
|
|
if (d->idsToCache.contains(id)) {
|
|
|
|
d->keysToCache.remove(d->idsToCache[id]);
|
|
|
|
}
|
|
|
|
|
|
|
|
d->keysToCache.insert(key, id);
|
|
|
|
d->idsToCache.insert(id, key);
|
2011-12-08 12:45:23 +01:00
|
|
|
d->saveTimer->start();
|
2008-11-04 00:08:39 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Theme::findInRectsCache(const QString &image, const QString &element, QRectF &rect) const
|
|
|
|
{
|
2011-01-30 22:47:29 +01:00
|
|
|
if (!d->svgElementsCache) {
|
2008-11-04 00:08:39 +01:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
KConfigGroup imageGroup(d->svgElementsCache, image);
|
2010-10-02 22:06:47 +02:00
|
|
|
rect = imageGroup.readEntry(element % QLatin1Literal("Size"), QRectF());
|
2008-11-04 00:08:39 +01:00
|
|
|
|
2008-12-11 00:17:12 +01:00
|
|
|
if (rect.isValid()) {
|
|
|
|
return true;
|
2008-11-04 00:08:39 +01:00
|
|
|
}
|
|
|
|
|
2009-07-31 20:47:20 +02:00
|
|
|
//Name starting by _ means the element is empty and we're asked for the size of
|
2009-01-22 20:59:25 +01:00
|
|
|
//the whole image, so the whole image is never invalid
|
2009-07-31 20:47:20 +02:00
|
|
|
if (element.indexOf('_') <= 0) {
|
2009-01-22 20:59:25 +01:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2008-12-11 00:17:12 +01:00
|
|
|
bool invalid = false;
|
|
|
|
|
|
|
|
QHash<QString, QSet<QString> >::iterator it = d->invalidElements.find(image);
|
|
|
|
if (it == d->invalidElements.end()) {
|
|
|
|
QSet<QString> elements = imageGroup.readEntry("invalidElements", QStringList()).toSet();
|
|
|
|
d->invalidElements.insert(image, elements);
|
|
|
|
invalid = elements.contains(element);
|
|
|
|
} else {
|
|
|
|
invalid = it.value().contains(element);
|
|
|
|
}
|
|
|
|
|
|
|
|
return invalid;
|
2008-11-04 00:08:39 +01:00
|
|
|
}
|
|
|
|
|
2010-10-30 00:14:18 +02:00
|
|
|
QStringList Theme::listCachedRectKeys(const QString &image) const
|
|
|
|
{
|
2011-01-30 22:47:29 +01:00
|
|
|
if (!d->svgElementsCache) {
|
|
|
|
return QStringList();
|
|
|
|
}
|
|
|
|
|
2010-10-30 00:14:18 +02:00
|
|
|
KConfigGroup imageGroup(d->svgElementsCache, image);
|
|
|
|
QStringList keys = imageGroup.keyList();
|
|
|
|
|
|
|
|
QMutableListIterator<QString> i(keys);
|
|
|
|
while (i.hasNext()) {
|
2010-10-30 00:30:10 +02:00
|
|
|
QString key = i.next();
|
2010-10-30 00:14:18 +02:00
|
|
|
if (key.endsWith("Size")) {
|
|
|
|
// The actual cache id used from outside doesn't end on "Size".
|
2010-10-30 00:30:10 +02:00
|
|
|
key.resize(key.size() - 4);
|
|
|
|
i.setValue(key);
|
2010-10-30 00:14:18 +02:00
|
|
|
} else {
|
|
|
|
i.remove();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return keys;
|
|
|
|
}
|
|
|
|
|
2008-11-04 00:08:39 +01:00
|
|
|
void Theme::insertIntoRectsCache(const QString& image, const QString &element, const QRectF &rect)
|
|
|
|
{
|
2011-01-30 22:47:29 +01:00
|
|
|
if (!d->svgElementsCache) {
|
2008-11-04 00:08:39 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (rect.isValid()) {
|
2008-12-11 00:17:12 +01:00
|
|
|
KConfigGroup imageGroup(d->svgElementsCache, image);
|
2010-10-02 22:06:47 +02:00
|
|
|
imageGroup.writeEntry(element % QLatin1Literal("Size"), rect);
|
2008-12-11 00:17:12 +01:00
|
|
|
} else {
|
|
|
|
QHash<QString, QSet<QString> >::iterator it = d->invalidElements.find(image);
|
|
|
|
if (it == d->invalidElements.end()) {
|
|
|
|
d->invalidElements[image].insert(element);
|
|
|
|
} else if (!it.value().contains(element)) {
|
|
|
|
if (it.value().count() > 1000) {
|
|
|
|
it.value().erase(it.value().begin());
|
|
|
|
}
|
|
|
|
|
|
|
|
it.value().insert(element);
|
2008-11-04 00:08:39 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Theme::invalidateRectsCache(const QString& image)
|
|
|
|
{
|
2011-01-30 22:47:29 +01:00
|
|
|
if (d->svgElementsCache) {
|
|
|
|
KConfigGroup imageGroup(d->svgElementsCache, image);
|
|
|
|
imageGroup.deleteGroup();
|
|
|
|
}
|
2008-12-11 03:45:45 +01:00
|
|
|
|
2011-01-30 22:47:29 +01:00
|
|
|
d->invalidElements.remove(image);
|
2008-12-11 03:45:45 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void Theme::releaseRectsCache(const QString &image)
|
|
|
|
{
|
|
|
|
QHash<QString, QSet<QString> >::iterator it = d->invalidElements.find(image);
|
|
|
|
if (it != d->invalidElements.end()) {
|
2011-01-30 22:47:29 +01:00
|
|
|
if (!d->svgElementsCache) {
|
|
|
|
KConfigGroup imageGroup(d->svgElementsCache, it.key());
|
|
|
|
imageGroup.writeEntry("invalidElements", it.value().toList());
|
|
|
|
}
|
|
|
|
|
2008-12-11 03:45:45 +01:00
|
|
|
d->invalidElements.erase(it);
|
|
|
|
}
|
2008-11-04 00:08:39 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void Theme::setCacheLimit(int kbytes)
|
|
|
|
{
|
2013-03-11 14:34:48 +01:00
|
|
|
d->cacheSize = kbytes;
|
|
|
|
delete d->pixmapCache;
|
|
|
|
d->pixmapCache = 0;
|
2008-11-04 00:08:39 +01:00
|
|
|
}
|
|
|
|
|
2013-03-11 18:01:22 +01:00
|
|
|
KPluginInfo Theme::pluginInfo() const
|
2010-12-27 01:46:17 +01:00
|
|
|
{
|
2013-03-11 18:01:22 +01:00
|
|
|
return d->pluginInfo;
|
2010-12-27 01:46:17 +01:00
|
|
|
}
|
|
|
|
|
2008-11-04 00:08:39 +01:00
|
|
|
}
|
|
|
|
|
2012-02-08 23:33:03 +01:00
|
|
|
#include "moc_theme.cpp"
|