2008-11-04 00:08:39 +01:00
|
|
|
/*
|
2010-10-14 14:09:23 +02:00
|
|
|
* Copyright 2008-2010 by Aaron Seigo <aseigo@kde.org>
|
|
|
|
* Copyright 2008-2010 Marco Martin <notmart@gmail.com>
|
2008-11-04 00:08:39 +01:00
|
|
|
*
|
|
|
|
* 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 "framesvg.h"
|
2009-01-13 22:46:07 +01:00
|
|
|
#include "private/framesvg_p.h"
|
2008-11-04 00:08:39 +01:00
|
|
|
|
2010-09-27 19:19:06 +02:00
|
|
|
#include <QAtomicInt>
|
2008-11-04 00:08:39 +01:00
|
|
|
#include <QBitmap>
|
2010-09-27 19:19:06 +02:00
|
|
|
#include <QCryptographicHash>
|
|
|
|
#include <QPainter>
|
2008-12-15 23:17:59 +01:00
|
|
|
#include <QRegion>
|
2010-09-27 19:19:06 +02:00
|
|
|
#include <QSize>
|
|
|
|
#include <QStringBuilder>
|
2008-11-29 20:24:40 +01:00
|
|
|
#include <QTimer>
|
2008-11-04 00:08:39 +01:00
|
|
|
|
2013-07-29 19:05:59 +02:00
|
|
|
#include <QDebug>
|
2008-11-04 00:08:39 +01:00
|
|
|
|
2010-10-14 14:09:23 +02:00
|
|
|
#include <theme.h>
|
|
|
|
#include <private/svg_p.h>
|
2008-11-04 00:08:39 +01:00
|
|
|
|
|
|
|
namespace Plasma
|
|
|
|
{
|
|
|
|
|
2014-05-28 20:25:19 +02:00
|
|
|
QHash<ThemePrivate *, QHash<QString, FrameData *> > FrameSvgPrivate::s_sharedFrames;
|
2010-09-27 19:19:06 +02:00
|
|
|
|
2009-12-22 18:28:22 +01:00
|
|
|
// Any attempt to generate a frame whose width or height is larger than this
|
|
|
|
// will be rejected
|
|
|
|
static const int MAX_FRAME_SIZE = 100000;
|
|
|
|
|
2014-05-28 20:25:19 +02:00
|
|
|
FrameData::~FrameData()
|
|
|
|
{
|
|
|
|
foreach (FrameSvg *frame, references.keys()) {
|
|
|
|
frame->d->frames.remove(prefix);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-11-04 00:08:39 +01:00
|
|
|
FrameSvg::FrameSvg(QObject *parent)
|
|
|
|
: Svg(parent),
|
|
|
|
d(new FrameSvgPrivate(this))
|
|
|
|
{
|
|
|
|
connect(this, SIGNAL(repaintNeeded()), this, SLOT(updateNeeded()));
|
2014-05-28 20:25:19 +02:00
|
|
|
d->frames.insert(QString(), new FrameData(this, QString()));
|
2008-11-04 00:08:39 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
FrameSvg::~FrameSvg()
|
|
|
|
{
|
|
|
|
delete d;
|
|
|
|
}
|
|
|
|
|
|
|
|
void FrameSvg::setImagePath(const QString &path)
|
|
|
|
{
|
|
|
|
if (path == imagePath()) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2010-10-14 14:09:23 +02:00
|
|
|
bool updateNeeded = true;
|
|
|
|
clearCache();
|
|
|
|
|
|
|
|
FrameData *fd = d->frames[d->prefix];
|
|
|
|
if (fd->refcount() == 1) {
|
|
|
|
// we're the only user of it, let's remove it from the shared keys
|
|
|
|
// we don't want to deref it, however, as we'll still be using it
|
2014-05-28 20:25:19 +02:00
|
|
|
const QString oldKey = d->cacheId(fd, d->prefix);
|
|
|
|
FrameSvgPrivate::s_sharedFrames[theme()->d].remove(oldKey);
|
2010-10-14 14:09:23 +02:00
|
|
|
} else {
|
|
|
|
// others are using this frame, so deref it for ourselves
|
|
|
|
fd->deref(this);
|
|
|
|
fd = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
Svg::d->setImagePath(path);
|
|
|
|
|
|
|
|
if (!fd) {
|
|
|
|
// we need to replace our frame, start by looking in the frame cache
|
2010-10-14 15:04:28 +02:00
|
|
|
FrameData *oldFd = d->frames[d->prefix];
|
2014-05-28 20:25:19 +02:00
|
|
|
const QString key = d->cacheId(oldFd, d->prefix);
|
|
|
|
fd = FrameSvgPrivate::s_sharedFrames[theme()->d].value(key);
|
2010-10-14 14:09:23 +02:00
|
|
|
|
|
|
|
if (fd) {
|
|
|
|
// we found one, so ref it and use it; we also don't need to (or want to!)
|
|
|
|
// trigger a full update of the frame since it is already the one we want
|
|
|
|
// and likely already rendered just fine
|
|
|
|
fd->ref(this);
|
|
|
|
updateNeeded = false;
|
|
|
|
} else {
|
|
|
|
// nothing exists for us in the cache, so create a new FrameData based
|
|
|
|
// on the old one
|
2010-10-14 15:04:28 +02:00
|
|
|
fd = new FrameData(*oldFd, this);
|
2010-10-14 14:09:23 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
d->frames.insert(d->prefix, fd);
|
|
|
|
}
|
|
|
|
|
2008-11-04 00:08:39 +01:00
|
|
|
setContainsMultipleImages(true);
|
2010-10-14 14:09:23 +02:00
|
|
|
if (updateNeeded) {
|
|
|
|
// ensure our frame is in the cache
|
2014-05-28 20:25:19 +02:00
|
|
|
const QString key = d->cacheId(fd, d->prefix);
|
|
|
|
FrameSvgPrivate::s_sharedFrames[theme()->d].insert(key, fd);
|
2008-11-04 00:08:39 +01:00
|
|
|
|
2010-10-14 14:09:23 +02:00
|
|
|
// this will emit repaintNeeded() as well when it is done
|
|
|
|
d->updateAndSignalSizes();
|
|
|
|
} else {
|
|
|
|
emit repaintNeeded();
|
|
|
|
}
|
2008-11-04 00:08:39 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void FrameSvg::setEnabledBorders(const EnabledBorders borders)
|
|
|
|
{
|
|
|
|
if (borders == d->frames[d->prefix]->enabledBorders) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2010-10-09 20:07:17 +02:00
|
|
|
FrameData *fd = d->frames[d->prefix];
|
|
|
|
|
|
|
|
const QString oldKey = d->cacheId(fd, d->prefix);
|
|
|
|
const EnabledBorders oldBorders = fd->enabledBorders;
|
|
|
|
fd->enabledBorders = borders;
|
|
|
|
const QString newKey = d->cacheId(fd, d->prefix);
|
|
|
|
fd->enabledBorders = oldBorders;
|
|
|
|
|
2013-07-29 19:05:59 +02:00
|
|
|
//qDebug() << "looking for" << newKey;
|
2014-05-28 20:25:19 +02:00
|
|
|
FrameData *newFd = FrameSvgPrivate::s_sharedFrames[theme()->d].value(newKey);
|
2010-10-09 20:07:17 +02:00
|
|
|
if (newFd) {
|
2013-07-29 19:05:59 +02:00
|
|
|
//qDebug() << "FOUND IT!" << newFd->refcount;
|
2010-10-09 20:07:17 +02:00
|
|
|
// we've found a math, so insert that new one and ref it ..
|
|
|
|
newFd->ref(this);
|
|
|
|
d->frames.insert(d->prefix, newFd);
|
|
|
|
|
|
|
|
//.. then deref the old one and if it's no longer used, get rid of it
|
|
|
|
if (fd->deref(this)) {
|
|
|
|
//const QString oldKey = d->cacheId(fd, d->prefix);
|
2013-07-29 19:05:59 +02:00
|
|
|
//qDebug() << "1. Removing it" << oldKey << fd->refcount;
|
2014-05-28 20:25:19 +02:00
|
|
|
FrameSvgPrivate::s_sharedFrames[theme()->d].remove(oldKey);
|
2010-10-09 20:07:17 +02:00
|
|
|
delete fd;
|
|
|
|
}
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (fd->refcount() == 1) {
|
|
|
|
// we're the only user of it, let's remove it from the shared keys
|
|
|
|
// we don't want to deref it, however, as we'll still be using it
|
2014-05-28 20:25:19 +02:00
|
|
|
FrameSvgPrivate::s_sharedFrames[theme()->d].remove(oldKey);
|
2010-10-09 20:07:17 +02:00
|
|
|
} else {
|
|
|
|
// others are using it, but we wish to change its size. so deref it,
|
|
|
|
// then create a copy of it (we're automatically ref'd via the ctor),
|
|
|
|
// then insert it into our frames.
|
|
|
|
fd->deref(this);
|
|
|
|
fd = new FrameData(*fd, this);
|
|
|
|
d->frames.insert(d->prefix, fd);
|
|
|
|
}
|
|
|
|
|
|
|
|
fd->enabledBorders = borders;
|
2008-11-04 00:08:39 +01:00
|
|
|
d->updateAndSignalSizes();
|
|
|
|
}
|
|
|
|
|
|
|
|
FrameSvg::EnabledBorders FrameSvg::enabledBorders() const
|
|
|
|
{
|
2010-05-13 12:08:44 +02:00
|
|
|
if (d->frames.isEmpty()) {
|
|
|
|
return NoBorder;
|
|
|
|
}
|
|
|
|
|
2014-04-26 01:45:47 +02:00
|
|
|
QHash<QString, FrameData *>::const_iterator it = d->frames.constFind(d->prefix);
|
2008-11-04 00:08:39 +01:00
|
|
|
|
|
|
|
if (it != d->frames.constEnd()) {
|
|
|
|
return it.value()->enabledBorders;
|
|
|
|
} else {
|
|
|
|
return NoBorder;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-05-14 18:27:27 +02:00
|
|
|
void FrameSvg::setElementPrefix(Plasma::Types::Location location)
|
2008-11-04 00:08:39 +01:00
|
|
|
{
|
|
|
|
switch (location) {
|
2014-04-26 01:45:47 +02:00
|
|
|
case Types::TopEdge:
|
|
|
|
setElementPrefix("north");
|
|
|
|
break;
|
|
|
|
case Types::BottomEdge:
|
|
|
|
setElementPrefix("south");
|
|
|
|
break;
|
|
|
|
case Types::LeftEdge:
|
|
|
|
setElementPrefix("west");
|
|
|
|
break;
|
|
|
|
case Types::RightEdge:
|
|
|
|
setElementPrefix("east");
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
setElementPrefix(QString());
|
|
|
|
break;
|
2008-11-04 00:08:39 +01:00
|
|
|
}
|
2009-05-31 07:38:42 +02:00
|
|
|
|
2008-11-04 00:08:39 +01:00
|
|
|
d->location = location;
|
|
|
|
}
|
|
|
|
|
2009-05-31 07:38:42 +02:00
|
|
|
void FrameSvg::setElementPrefix(const QString &prefix)
|
2008-11-04 00:08:39 +01:00
|
|
|
{
|
|
|
|
const QString oldPrefix(d->prefix);
|
|
|
|
|
2010-09-27 19:19:06 +02:00
|
|
|
if (!hasElement(prefix % "-center")) {
|
2008-11-04 00:08:39 +01:00
|
|
|
d->prefix.clear();
|
|
|
|
} else {
|
|
|
|
d->prefix = prefix;
|
|
|
|
if (!d->prefix.isEmpty()) {
|
|
|
|
d->prefix += '-';
|
|
|
|
}
|
|
|
|
}
|
2014-07-11 16:42:55 +02:00
|
|
|
d->requestedPrefix = prefix;
|
2008-11-04 00:08:39 +01:00
|
|
|
|
2009-05-31 07:38:42 +02:00
|
|
|
FrameData *oldFrameData = d->frames.value(oldPrefix);
|
|
|
|
if (oldPrefix == d->prefix && oldFrameData) {
|
2008-11-04 00:08:39 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!d->frames.contains(d->prefix)) {
|
2009-05-31 07:38:42 +02:00
|
|
|
if (oldFrameData) {
|
2010-09-27 19:19:06 +02:00
|
|
|
FrameData *newFd = 0;
|
|
|
|
if (!oldFrameData->frameSize.isEmpty()) {
|
2014-05-28 20:25:19 +02:00
|
|
|
const QString key = d->cacheId(oldFrameData, d->prefix);
|
|
|
|
newFd = FrameSvgPrivate::s_sharedFrames[theme()->d].value(key);
|
2010-09-27 19:19:06 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// we need to put this in the cache if we didn't find it in the shared frames
|
|
|
|
// and we have a size; if we don't have a size, we'll catch it later
|
|
|
|
const bool cache = !newFd && !oldFrameData->frameSize.isEmpty();
|
|
|
|
if (newFd) {
|
|
|
|
newFd->ref(this);
|
|
|
|
} else {
|
|
|
|
newFd = new FrameData(*oldFrameData, this);
|
|
|
|
}
|
|
|
|
|
|
|
|
d->frames.insert(d->prefix, newFd);
|
|
|
|
|
|
|
|
if (cache) {
|
|
|
|
// we have to cache after inserting the frame since the cacheId requires the
|
|
|
|
// frame to be in the frames collection already
|
2014-05-28 20:25:19 +02:00
|
|
|
const QString key = d->cacheId(oldFrameData, d->prefix);
|
2013-07-29 19:05:59 +02:00
|
|
|
//qDebug() << this << " 1. inserting as" << key;
|
2010-10-05 21:26:00 +02:00
|
|
|
|
2014-05-28 20:25:19 +02:00
|
|
|
FrameSvgPrivate::s_sharedFrames[theme()->d].insert(key, newFd);
|
2010-09-27 19:19:06 +02:00
|
|
|
}
|
2009-05-31 07:38:42 +02:00
|
|
|
} else {
|
2010-09-27 19:19:06 +02:00
|
|
|
// couldn't find anything useful, so we just create something here
|
|
|
|
// we don't have a size for it yet, so don't bother trying to share it just yet
|
2014-05-28 20:25:19 +02:00
|
|
|
FrameData *newFd = new FrameData(this, d->prefix);
|
2010-09-27 19:19:06 +02:00
|
|
|
d->frames.insert(d->prefix, newFd);
|
2009-05-31 07:38:42 +02:00
|
|
|
}
|
|
|
|
|
2008-11-04 00:08:39 +01:00
|
|
|
d->updateSizes();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!d->cacheAll) {
|
|
|
|
d->frames.remove(oldPrefix);
|
2010-09-27 19:19:06 +02:00
|
|
|
if (oldFrameData) {
|
|
|
|
if (oldFrameData->deref(this)) {
|
2014-05-28 20:25:19 +02:00
|
|
|
const QString oldKey = d->cacheId(oldFrameData, oldPrefix);
|
|
|
|
FrameSvgPrivate::s_sharedFrames[theme()->d].remove(oldKey);
|
2010-09-27 19:19:06 +02:00
|
|
|
delete oldFrameData;
|
|
|
|
}
|
|
|
|
}
|
2008-11-04 00:08:39 +01:00
|
|
|
}
|
|
|
|
|
2013-05-10 19:29:13 +02:00
|
|
|
d->location = Types::Floating;
|
2008-11-04 00:08:39 +01:00
|
|
|
}
|
|
|
|
|
2014-04-26 01:45:47 +02:00
|
|
|
bool FrameSvg::hasElementPrefix(const QString &prefix) const
|
2008-11-04 00:08:39 +01:00
|
|
|
{
|
|
|
|
//for now it simply checks if a center element exists,
|
|
|
|
//because it could make sense for certain themes to not have all the elements
|
|
|
|
if (prefix.isEmpty()) {
|
|
|
|
return hasElement("center");
|
|
|
|
} else {
|
2010-09-27 19:19:06 +02:00
|
|
|
return hasElement(prefix % "-center");
|
2008-11-04 00:08:39 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-05-14 18:27:27 +02:00
|
|
|
bool FrameSvg::hasElementPrefix(Plasma::Types::Location location) const
|
2008-11-04 00:08:39 +01:00
|
|
|
{
|
|
|
|
switch (location) {
|
2014-04-26 01:45:47 +02:00
|
|
|
case Types::TopEdge:
|
|
|
|
return hasElementPrefix("north");
|
|
|
|
break;
|
|
|
|
case Types::BottomEdge:
|
|
|
|
return hasElementPrefix("south");
|
|
|
|
break;
|
|
|
|
case Types::LeftEdge:
|
|
|
|
return hasElementPrefix("west");
|
|
|
|
break;
|
|
|
|
case Types::RightEdge:
|
|
|
|
return hasElementPrefix("east");
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return hasElementPrefix(QString());
|
|
|
|
break;
|
2008-11-04 00:08:39 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
QString FrameSvg::prefix()
|
|
|
|
{
|
2014-07-11 16:55:59 +02:00
|
|
|
return d->requestedPrefix;
|
2008-11-04 00:08:39 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void FrameSvg::resizeFrame(const QSizeF &size)
|
|
|
|
{
|
2010-10-11 23:08:38 +02:00
|
|
|
if (imagePath().isEmpty()) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2008-11-04 00:08:39 +01:00
|
|
|
if (size.isEmpty()) {
|
2011-07-29 15:46:52 +02:00
|
|
|
#ifndef NDEBUG
|
2013-07-29 19:05:59 +02:00
|
|
|
// qDebug() << "Invalid size" << size;
|
2011-07-29 15:46:52 +02:00
|
|
|
#endif
|
2008-11-04 00:08:39 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2010-09-27 19:19:06 +02:00
|
|
|
FrameData *fd = d->frames[d->prefix];
|
|
|
|
if (size == fd->frameSize) {
|
2008-11-04 00:08:39 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2010-10-05 21:26:00 +02:00
|
|
|
const QString oldKey = d->cacheId(fd, d->prefix);
|
2010-09-27 19:19:06 +02:00
|
|
|
const QSize currentSize = fd->frameSize;
|
|
|
|
fd->frameSize = size.toSize();
|
|
|
|
const QString newKey = d->cacheId(fd, d->prefix);
|
|
|
|
fd->frameSize = currentSize;
|
|
|
|
|
2013-07-29 19:05:59 +02:00
|
|
|
//qDebug() << "looking for" << newKey;
|
2014-05-28 20:25:19 +02:00
|
|
|
FrameData *newFd = FrameSvgPrivate::s_sharedFrames[theme()->d].value(newKey);
|
2010-09-27 19:19:06 +02:00
|
|
|
if (newFd) {
|
2013-07-29 19:05:59 +02:00
|
|
|
//qDebug() << "FOUND IT!" << newFd->refcount;
|
2010-09-27 19:19:06 +02:00
|
|
|
// we've found a math, so insert that new one and ref it ..
|
|
|
|
newFd->ref(this);
|
|
|
|
d->frames.insert(d->prefix, newFd);
|
|
|
|
|
|
|
|
//.. then deref the old one and if it's no longer used, get rid of it
|
|
|
|
if (fd->deref(this)) {
|
|
|
|
//const QString oldKey = d->cacheId(fd, d->prefix);
|
2013-07-29 19:05:59 +02:00
|
|
|
//qDebug() << "1. Removing it" << oldKey << fd->refcount;
|
2014-05-28 20:25:19 +02:00
|
|
|
FrameSvgPrivate::s_sharedFrames[theme()->d].remove(oldKey);
|
2010-09-27 19:19:06 +02:00
|
|
|
delete fd;
|
|
|
|
}
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (fd->refcount() == 1) {
|
|
|
|
// we're the only user of it, let's remove it from the shared keys
|
|
|
|
// we don't want to deref it, however, as we'll still be using it
|
2014-05-28 20:25:19 +02:00
|
|
|
FrameSvgPrivate::s_sharedFrames[theme()->d].remove(oldKey);
|
2010-09-27 19:19:06 +02:00
|
|
|
} else {
|
|
|
|
// others are using it, but we wish to change its size. so deref it,
|
|
|
|
// then create a copy of it (we're automatically ref'd via the ctor),
|
|
|
|
// then insert it into our frames.
|
|
|
|
fd->deref(this);
|
|
|
|
fd = new FrameData(*fd, this);
|
|
|
|
d->frames.insert(d->prefix, fd);
|
|
|
|
}
|
|
|
|
|
2008-11-04 00:08:39 +01:00
|
|
|
d->updateSizes();
|
2010-09-27 19:19:06 +02:00
|
|
|
fd->frameSize = size.toSize();
|
2010-10-13 16:27:08 +02:00
|
|
|
// we know it isn't in s_sharedFrames due to the check above, so insert it now
|
2014-05-28 20:25:19 +02:00
|
|
|
FrameSvgPrivate::s_sharedFrames[theme()->d].insert(newKey, fd);
|
2008-11-04 00:08:39 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
QSizeF FrameSvg::frameSize() const
|
|
|
|
{
|
2014-04-26 01:45:47 +02:00
|
|
|
QHash<QString, FrameData *>::const_iterator it = d->frames.constFind(d->prefix);
|
2008-11-04 00:08:39 +01:00
|
|
|
|
2009-06-10 19:11:02 +02:00
|
|
|
if (it == d->frames.constEnd()) {
|
2008-11-04 00:08:39 +01:00
|
|
|
return QSize(-1, -1);
|
2009-06-10 19:11:02 +02:00
|
|
|
} else {
|
|
|
|
return d->frameSize(it.value());
|
2008-11-04 00:08:39 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-05-14 20:37:16 +02:00
|
|
|
qreal FrameSvg::marginSize(const Plasma::Types::MarginEdge edge) const
|
2008-11-04 00:08:39 +01:00
|
|
|
{
|
|
|
|
if (d->frames[d->prefix]->noBorderPadding) {
|
|
|
|
return .0;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (edge) {
|
2013-05-10 19:29:13 +02:00
|
|
|
case Plasma::Types::TopMargin:
|
2008-11-04 00:08:39 +01:00
|
|
|
return d->frames[d->prefix]->topMargin;
|
2014-04-26 01:45:47 +02:00
|
|
|
break;
|
2008-11-04 00:08:39 +01:00
|
|
|
|
2013-05-10 19:29:13 +02:00
|
|
|
case Plasma::Types::LeftMargin:
|
2008-11-04 00:08:39 +01:00
|
|
|
return d->frames[d->prefix]->leftMargin;
|
2014-04-26 01:45:47 +02:00
|
|
|
break;
|
2008-11-04 00:08:39 +01:00
|
|
|
|
2013-05-10 19:29:13 +02:00
|
|
|
case Plasma::Types::RightMargin:
|
2008-11-04 00:08:39 +01:00
|
|
|
return d->frames[d->prefix]->rightMargin;
|
2014-04-26 01:45:47 +02:00
|
|
|
break;
|
2008-11-04 00:08:39 +01:00
|
|
|
|
|
|
|
//Plasma::BottomMargin
|
|
|
|
default:
|
|
|
|
return d->frames[d->prefix]->bottomMargin;
|
2014-04-26 01:45:47 +02:00
|
|
|
break;
|
2008-11-04 00:08:39 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-02-21 21:13:12 +01:00
|
|
|
qreal FrameSvg::fixedMarginSize(const Plasma::Types::MarginEdge edge) const
|
|
|
|
{
|
|
|
|
if (d->frames[d->prefix]->noBorderPadding) {
|
|
|
|
return .0;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (edge) {
|
|
|
|
case Plasma::Types::TopMargin:
|
|
|
|
return d->frames[d->prefix]->fixedTopMargin;
|
2014-04-26 01:45:47 +02:00
|
|
|
break;
|
2014-02-21 21:13:12 +01:00
|
|
|
|
|
|
|
case Plasma::Types::LeftMargin:
|
|
|
|
return d->frames[d->prefix]->fixedLeftMargin;
|
2014-04-26 01:45:47 +02:00
|
|
|
break;
|
2014-02-21 21:13:12 +01:00
|
|
|
|
|
|
|
case Plasma::Types::RightMargin:
|
|
|
|
return d->frames[d->prefix]->fixedRightMargin;
|
2014-04-26 01:45:47 +02:00
|
|
|
break;
|
2014-02-21 21:13:12 +01:00
|
|
|
|
|
|
|
//Plasma::BottomMargin
|
|
|
|
default:
|
|
|
|
return d->frames[d->prefix]->fixedBottomMargin;
|
2014-04-26 01:45:47 +02:00
|
|
|
break;
|
2014-02-21 21:13:12 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-11-04 00:08:39 +01:00
|
|
|
void FrameSvg::getMargins(qreal &left, qreal &top, qreal &right, qreal &bottom) const
|
|
|
|
{
|
|
|
|
FrameData *frame = d->frames[d->prefix];
|
|
|
|
|
2009-11-04 01:12:59 +01:00
|
|
|
if (frame->noBorderPadding) {
|
2008-11-04 00:08:39 +01:00
|
|
|
left = top = right = bottom = 0;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
top = frame->topMargin;
|
|
|
|
left = frame->leftMargin;
|
|
|
|
right = frame->rightMargin;
|
|
|
|
bottom = frame->bottomMargin;
|
|
|
|
}
|
|
|
|
|
2014-02-21 21:13:12 +01:00
|
|
|
void FrameSvg::getFixedMargins(qreal &left, qreal &top, qreal &right, qreal &bottom) const
|
|
|
|
{
|
|
|
|
FrameData *frame = d->frames[d->prefix];
|
|
|
|
|
|
|
|
if (frame->noBorderPadding) {
|
|
|
|
left = top = right = bottom = 0;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
top = frame->fixedTopMargin;
|
|
|
|
left = frame->fixedLeftMargin;
|
|
|
|
right = frame->fixedRightMargin;
|
|
|
|
bottom = frame->fixedBottomMargin;
|
|
|
|
}
|
|
|
|
|
2008-11-04 00:08:39 +01:00
|
|
|
QRectF FrameSvg::contentsRect() const
|
|
|
|
{
|
2014-07-21 15:44:25 +02:00
|
|
|
QHash<QString, FrameData *>::const_iterator it = d->frames.constFind(d->prefix);
|
|
|
|
if (it != d->frames.constEnd()) {
|
|
|
|
return d->contentGeometry(*it, (*it)->frameSize);
|
2008-11-04 00:08:39 +01:00
|
|
|
} else {
|
|
|
|
return QRectF();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-01-13 22:46:07 +01:00
|
|
|
QPixmap FrameSvg::alphaMask() const
|
2008-11-04 00:08:39 +01:00
|
|
|
{
|
2014-04-26 01:45:47 +02:00
|
|
|
//FIXME: the distinction between overlay and
|
2010-03-05 23:37:44 +01:00
|
|
|
return d->alphaMask();
|
2009-01-13 22:46:07 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
QRegion FrameSvg::mask() const
|
|
|
|
{
|
|
|
|
FrameData *frame = d->frames[d->prefix];
|
2010-08-05 23:15:18 +02:00
|
|
|
QString id = d->cacheId(frame, QString());
|
2014-07-11 18:37:43 +02:00
|
|
|
|
|
|
|
QRegion* obj = frame->cachedMasks.object(id);
|
|
|
|
|
|
|
|
if (!obj) {
|
|
|
|
obj = new QRegion(QBitmap(d->alphaMask().alphaChannel().createMaskFromColor(Qt::black)));
|
|
|
|
frame->cachedMasks.insert(id, obj);
|
2010-08-05 23:15:18 +02:00
|
|
|
}
|
2014-07-11 18:37:43 +02:00
|
|
|
return *obj;
|
2008-11-04 00:08:39 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void FrameSvg::setCacheAllRenderedFrames(bool cache)
|
|
|
|
{
|
|
|
|
if (d->cacheAll && !cache) {
|
|
|
|
clearCache();
|
|
|
|
}
|
|
|
|
|
|
|
|
d->cacheAll = cache;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool FrameSvg::cacheAllRenderedFrames() const
|
|
|
|
{
|
|
|
|
return d->cacheAll;
|
|
|
|
}
|
|
|
|
|
|
|
|
void FrameSvg::clearCache()
|
|
|
|
{
|
|
|
|
FrameData *frame = d->frames[d->prefix];
|
|
|
|
|
|
|
|
// delete all the frames that aren't this one
|
2014-04-26 01:45:47 +02:00
|
|
|
QMutableHashIterator<QString, FrameData *> it(d->frames);
|
2008-11-04 00:08:39 +01:00
|
|
|
while (it.hasNext()) {
|
|
|
|
FrameData *p = it.next().value();
|
|
|
|
if (frame != p) {
|
2010-09-27 19:19:06 +02:00
|
|
|
//TODO: should we clear from the Theme pixmap cache as well?
|
|
|
|
if (p->deref(this)) {
|
2014-05-28 20:25:19 +02:00
|
|
|
const QString key = d->cacheId(p, it.key());
|
|
|
|
FrameSvgPrivate::s_sharedFrames[theme()->d].remove(key);
|
2011-01-20 12:23:56 +01:00
|
|
|
p->cachedBackground = QPixmap();
|
2010-09-27 19:19:06 +02:00
|
|
|
}
|
|
|
|
|
2008-11-04 00:08:39 +01:00
|
|
|
it.remove();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
QPixmap FrameSvg::framePixmap()
|
|
|
|
{
|
|
|
|
FrameData *frame = d->frames[d->prefix];
|
|
|
|
if (frame->cachedBackground.isNull()) {
|
|
|
|
d->generateBackground(frame);
|
|
|
|
}
|
|
|
|
|
|
|
|
return frame->cachedBackground;
|
|
|
|
}
|
|
|
|
|
|
|
|
void FrameSvg::paintFrame(QPainter *painter, const QRectF &target, const QRectF &source)
|
|
|
|
{
|
|
|
|
FrameData *frame = d->frames[d->prefix];
|
|
|
|
if (frame->cachedBackground.isNull()) {
|
|
|
|
d->generateBackground(frame);
|
|
|
|
if (frame->cachedBackground.isNull()) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-01-22 14:05:34 +01:00
|
|
|
painter->drawPixmap(target, frame->cachedBackground, source.isValid() ? source : target);
|
2008-11-04 00:08:39 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void FrameSvg::paintFrame(QPainter *painter, const QPointF &pos)
|
|
|
|
{
|
|
|
|
FrameData *frame = d->frames[d->prefix];
|
|
|
|
if (frame->cachedBackground.isNull()) {
|
|
|
|
d->generateBackground(frame);
|
|
|
|
if (frame->cachedBackground.isNull()) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
painter->drawPixmap(pos, frame->cachedBackground);
|
|
|
|
}
|
|
|
|
|
2010-09-27 19:19:06 +02:00
|
|
|
//#define DEBUG_FRAMESVG_CACHE
|
|
|
|
FrameSvgPrivate::~FrameSvgPrivate()
|
|
|
|
{
|
|
|
|
#ifdef DEBUG_FRAMESVG_CACHE
|
2011-07-29 15:46:52 +02:00
|
|
|
#ifndef NDEBUG
|
2013-07-29 19:05:59 +02:00
|
|
|
// qDebug() << "*************" << q << q->imagePath() << "****************";
|
2011-07-29 15:46:52 +02:00
|
|
|
#endif
|
2010-09-27 19:19:06 +02:00
|
|
|
#endif
|
|
|
|
|
|
|
|
QHashIterator<QString, FrameData *> it(frames);
|
|
|
|
while (it.hasNext()) {
|
|
|
|
it.next();
|
|
|
|
if (it.value()) {
|
|
|
|
// we remove all references from this widget to the frame, and delete it if we're the
|
|
|
|
// last user
|
|
|
|
if (it.value()->removeRefs(q)) {
|
|
|
|
const QString key = cacheId(it.value(), it.key());
|
|
|
|
#ifdef DEBUG_FRAMESVG_CACHE
|
2011-07-29 15:46:52 +02:00
|
|
|
#ifndef NDEBUG
|
2014-05-28 20:25:19 +02:00
|
|
|
// qDebug() << "2. Removing it" << key << it.value() << it.value()->refcount() << s_sharedFrames[theme()->d].contains(key);
|
2011-07-29 15:46:52 +02:00
|
|
|
#endif
|
2010-09-27 19:19:06 +02:00
|
|
|
#endif
|
2014-05-28 20:25:19 +02:00
|
|
|
s_sharedFrames[q->theme()->d].remove(key);
|
2010-09-27 19:19:06 +02:00
|
|
|
delete it.value();
|
|
|
|
}
|
|
|
|
#ifdef DEBUG_FRAMESVG_CACHE
|
|
|
|
else {
|
2011-07-29 15:46:52 +02:00
|
|
|
#ifndef NDEBUG
|
2013-07-29 19:05:59 +02:00
|
|
|
// qDebug() << "still shared:" << cacheId(it.value(), it.key()) << it.value() << it.value()->refcount() << it.value()->isUsed();
|
2011-07-29 15:46:52 +02:00
|
|
|
#endif
|
2010-09-27 19:19:06 +02:00
|
|
|
}
|
|
|
|
} else {
|
2011-07-29 15:46:52 +02:00
|
|
|
#ifndef NDEBUG
|
2013-07-29 19:05:59 +02:00
|
|
|
// qDebug() << "lost our value for" << it.key();
|
2011-07-29 15:46:52 +02:00
|
|
|
#endif
|
2010-09-27 19:19:06 +02:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef DEBUG_FRAMESVG_CACHE
|
2014-05-28 20:25:19 +02:00
|
|
|
QHashIterator<QString, FrameData *> it2(s_sharedFrames[theme()->d]);
|
2010-09-27 19:19:06 +02:00
|
|
|
int shares = 0;
|
|
|
|
while (it2.hasNext()) {
|
|
|
|
it2.next();
|
|
|
|
const int rc = it2.value()->refcount();
|
|
|
|
if (rc == 0) {
|
2011-07-29 15:46:52 +02:00
|
|
|
#ifndef NDEBUG
|
2013-07-29 19:05:59 +02:00
|
|
|
// qDebug() << " LOST!" << it2.key() << rc << it2.value();// << it2.value()->references;
|
2011-07-29 15:46:52 +02:00
|
|
|
#endif
|
2010-09-27 19:19:06 +02:00
|
|
|
} else {
|
2011-07-29 15:46:52 +02:00
|
|
|
#ifndef NDEBUG
|
2013-07-29 19:05:59 +02:00
|
|
|
// qDebug() << " " << it2.key() << rc << it2.value();
|
2011-07-29 15:46:52 +02:00
|
|
|
#endif
|
2010-09-27 19:19:06 +02:00
|
|
|
foreach (FrameSvg *data, it2.value()->references.keys()) {
|
2011-07-29 15:46:52 +02:00
|
|
|
#ifndef NDEBUG
|
2014-04-26 01:45:47 +02:00
|
|
|
qDebug() << " " << (void *)data << it2.value()->references[data];
|
2011-07-29 15:46:52 +02:00
|
|
|
#endif
|
2010-09-27 19:19:06 +02:00
|
|
|
}
|
|
|
|
shares += rc - 1;
|
|
|
|
}
|
|
|
|
}
|
2011-07-29 15:46:52 +02:00
|
|
|
#ifndef NDEBUG
|
2014-05-28 20:25:19 +02:00
|
|
|
// qDebug() << "#####################################" << s_sharedFrames[theme()->d].count() << ", pixmaps saved:" << shares;
|
2011-07-29 15:46:52 +02:00
|
|
|
#endif
|
2010-09-27 19:19:06 +02:00
|
|
|
#endif
|
|
|
|
|
|
|
|
frames.clear();
|
|
|
|
}
|
|
|
|
|
2010-03-05 23:37:44 +01:00
|
|
|
QPixmap FrameSvgPrivate::alphaMask()
|
2009-10-10 23:34:51 +02:00
|
|
|
{
|
|
|
|
FrameData *frame = frames[prefix];
|
|
|
|
QString maskPrefix;
|
|
|
|
|
2010-09-27 19:19:06 +02:00
|
|
|
if (q->hasElement("mask-" % prefix % "center")) {
|
2009-10-10 23:34:51 +02:00
|
|
|
maskPrefix = "mask-";
|
|
|
|
}
|
|
|
|
|
2010-09-27 19:19:06 +02:00
|
|
|
if (maskPrefix.isNull()) {
|
|
|
|
if (frame->cachedBackground.isNull()) {
|
|
|
|
generateBackground(frame);
|
|
|
|
if (frame->cachedBackground.isNull()) {
|
|
|
|
return QPixmap();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return frame->cachedBackground;
|
|
|
|
} else {
|
2009-10-10 23:34:51 +02:00
|
|
|
QString oldPrefix = prefix;
|
|
|
|
|
|
|
|
// We are setting the prefix only temporary to generate
|
|
|
|
// the needed mask image
|
2010-09-27 19:19:06 +02:00
|
|
|
prefix = maskPrefix % oldPrefix;
|
2009-10-10 23:34:51 +02:00
|
|
|
|
|
|
|
if (!frames.contains(prefix)) {
|
2014-05-28 20:25:19 +02:00
|
|
|
const QString key = cacheId(frame, prefix);
|
2010-09-27 19:19:06 +02:00
|
|
|
// see if we can find a suitable candidate in the shared frames
|
|
|
|
// if successful, ref and insert, otherwise create a new one
|
|
|
|
// and insert that into both the shared frames and our frames.
|
2014-05-28 20:25:19 +02:00
|
|
|
FrameData *maskFrame = s_sharedFrames[q->theme()->d].value(key);
|
2010-10-05 21:26:00 +02:00
|
|
|
|
2010-09-27 19:19:06 +02:00
|
|
|
if (maskFrame) {
|
|
|
|
maskFrame->ref(q);
|
|
|
|
} else {
|
|
|
|
maskFrame = new FrameData(*frame, q);
|
2014-05-28 20:25:19 +02:00
|
|
|
s_sharedFrames[q->theme()->d].insert(key, maskFrame);
|
2010-09-27 19:19:06 +02:00
|
|
|
}
|
2014-06-13 17:14:30 +02:00
|
|
|
maskFrame->enabledBorders = frame->enabledBorders;
|
2010-09-27 19:19:06 +02:00
|
|
|
|
|
|
|
frames.insert(prefix, maskFrame);
|
2009-10-10 23:34:51 +02:00
|
|
|
updateSizes();
|
|
|
|
}
|
|
|
|
|
|
|
|
FrameData *maskFrame = frames[prefix];
|
2014-06-13 17:14:30 +02:00
|
|
|
maskFrame->enabledBorders = frame->enabledBorders;
|
2009-10-10 23:34:51 +02:00
|
|
|
if (maskFrame->cachedBackground.isNull() || maskFrame->frameSize != frameSize(frame)) {
|
2010-10-05 21:26:00 +02:00
|
|
|
const QString oldKey = cacheId(maskFrame, prefix);
|
2009-10-10 23:34:51 +02:00
|
|
|
maskFrame->frameSize = frameSize(frame).toSize();
|
2010-10-05 21:26:00 +02:00
|
|
|
const QString newKey = cacheId(maskFrame, prefix);
|
2014-05-28 20:25:19 +02:00
|
|
|
if (s_sharedFrames[q->theme()->d].contains(oldKey)) {
|
|
|
|
s_sharedFrames[q->theme()->d].remove(oldKey);
|
|
|
|
s_sharedFrames[q->theme()->d].insert(newKey, maskFrame);
|
2010-10-05 21:26:00 +02:00
|
|
|
}
|
|
|
|
|
2009-10-10 23:34:51 +02:00
|
|
|
maskFrame->cachedBackground = QPixmap();
|
|
|
|
|
|
|
|
generateBackground(maskFrame);
|
|
|
|
if (maskFrame->cachedBackground.isNull()) {
|
|
|
|
return QPixmap();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
prefix = oldPrefix;
|
|
|
|
return maskFrame->cachedBackground;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-11-04 00:08:39 +01:00
|
|
|
void FrameSvgPrivate::generateBackground(FrameData *frame)
|
|
|
|
{
|
2010-11-03 22:55:50 +01:00
|
|
|
if (!frame->cachedBackground.isNull() || !q->hasElementPrefix(q->prefix())) {
|
2008-11-04 00:08:39 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2009-09-28 23:54:00 +02:00
|
|
|
const QString id = cacheId(frame, prefix);
|
2009-08-04 23:19:20 +02:00
|
|
|
|
2009-09-28 23:54:00 +02:00
|
|
|
bool frameCached = !frame->cachedBackground.isNull();
|
|
|
|
bool overlayCached = false;
|
2012-06-15 00:11:42 +02:00
|
|
|
const bool overlayAvailable = !prefix.startsWith(QLatin1String("mask-")) && q->hasElement(prefix % "overlay");
|
2009-09-28 23:54:00 +02:00
|
|
|
QPixmap overlay;
|
2009-05-13 18:37:13 +02:00
|
|
|
if (q->isUsingRenderingCache()) {
|
2014-05-19 14:58:19 +02:00
|
|
|
frameCached = q->theme()->findInCache(id, frame->cachedBackground) && !frame->cachedBackground.isNull();
|
2009-09-28 23:54:00 +02:00
|
|
|
|
|
|
|
if (overlayAvailable) {
|
2014-05-19 14:58:19 +02:00
|
|
|
overlayCached = q->theme()->findInCache("overlay_" % id, overlay) && !overlay.isNull();
|
2009-05-13 18:37:13 +02:00
|
|
|
}
|
2009-05-01 18:28:10 +02:00
|
|
|
}
|
2008-11-04 00:08:39 +01:00
|
|
|
|
2009-09-28 23:54:00 +02:00
|
|
|
if (!frameCached) {
|
|
|
|
generateFrameBackground(frame);
|
|
|
|
}
|
|
|
|
|
|
|
|
//Overlays
|
|
|
|
QSize overlaySize;
|
2009-12-13 21:13:55 +01:00
|
|
|
QPoint actualOverlayPos = QPoint(0, 0);
|
2009-09-28 23:54:00 +02:00
|
|
|
if (overlayAvailable && !overlayCached) {
|
2010-09-27 19:19:06 +02:00
|
|
|
overlaySize = q->elementSize(prefix % "overlay");
|
2009-09-28 23:54:00 +02:00
|
|
|
|
2012-02-28 18:08:17 +01:00
|
|
|
if (q->hasElement(prefix % "hint-overlay-pos-right")) {
|
2009-12-13 21:13:55 +01:00
|
|
|
actualOverlayPos.setX(frame->frameSize.width() - overlaySize.width());
|
2010-09-27 19:19:06 +02:00
|
|
|
} else if (q->hasElement(prefix % "hint-overlay-pos-bottom")) {
|
2009-12-13 21:13:55 +01:00
|
|
|
actualOverlayPos.setY(frame->frameSize.height() - overlaySize.height());
|
2014-04-26 01:45:47 +02:00
|
|
|
//Stretched or Tiled?
|
2010-09-27 19:19:06 +02:00
|
|
|
} else if (q->hasElement(prefix % "hint-overlay-stretch")) {
|
2009-09-28 23:54:00 +02:00
|
|
|
overlaySize = frameSize(frame).toSize();
|
|
|
|
} else {
|
2010-09-27 19:19:06 +02:00
|
|
|
if (q->hasElement(prefix % "hint-overlay-tile-horizontal")) {
|
2009-09-28 23:54:00 +02:00
|
|
|
overlaySize.setWidth(frameSize(frame).width());
|
|
|
|
}
|
2010-09-27 19:19:06 +02:00
|
|
|
if (q->hasElement(prefix % "hint-overlay-tile-vertical")) {
|
2009-09-28 23:54:00 +02:00
|
|
|
overlaySize.setHeight(frameSize(frame).height());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-03-05 23:37:44 +01:00
|
|
|
overlay = alphaMask();
|
2009-09-28 23:54:00 +02:00
|
|
|
QPainter overlayPainter(&overlay);
|
|
|
|
overlayPainter.setCompositionMode(QPainter::CompositionMode_SourceIn);
|
|
|
|
//Tiling?
|
2010-09-27 19:19:06 +02:00
|
|
|
if (q->hasElement(prefix % "hint-overlay-tile-horizontal") ||
|
2014-04-26 01:45:47 +02:00
|
|
|
q->hasElement(prefix % "hint-overlay-tile-vertical")) {
|
2009-09-28 23:54:00 +02:00
|
|
|
|
|
|
|
QSize s = q->size();
|
2010-09-27 19:19:06 +02:00
|
|
|
q->resize(q->elementSize(prefix % "overlay"));
|
2009-09-28 23:54:00 +02:00
|
|
|
|
2014-04-26 01:45:47 +02:00
|
|
|
overlayPainter.drawTiledPixmap(QRect(QPoint(0, 0), overlaySize), q->pixmap(prefix % "overlay"));
|
2009-09-28 23:54:00 +02:00
|
|
|
q->resize(s);
|
|
|
|
} else {
|
2010-09-27 19:19:06 +02:00
|
|
|
q->paint(&overlayPainter, QRect(actualOverlayPos, overlaySize), prefix % "overlay");
|
2009-09-28 23:54:00 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
overlayPainter.end();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!frameCached) {
|
|
|
|
cacheFrame(prefix, frame->cachedBackground, overlayCached ? overlay : QPixmap());
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!overlay.isNull()) {
|
|
|
|
QPainter p(&frame->cachedBackground);
|
|
|
|
p.setCompositionMode(QPainter::CompositionMode_SourceOver);
|
2009-12-13 21:13:55 +01:00
|
|
|
p.drawPixmap(actualOverlayPos, overlay, QRect(actualOverlayPos, overlaySize));
|
2009-09-28 23:54:00 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void FrameSvgPrivate::generateFrameBackground(FrameData *frame)
|
|
|
|
{
|
2013-07-29 19:05:59 +02:00
|
|
|
//qDebug() << "generating background";
|
2014-07-14 16:32:10 +02:00
|
|
|
const QSize size = frameSize(frame).toSize();
|
2008-11-04 00:08:39 +01:00
|
|
|
|
2009-06-10 19:11:02 +02:00
|
|
|
if (!size.isValid()) {
|
2011-07-29 15:46:52 +02:00
|
|
|
#ifndef NDEBUG
|
2013-07-29 19:05:59 +02:00
|
|
|
// qDebug() << "Invalid frame size" << size;
|
2011-07-29 15:46:52 +02:00
|
|
|
#endif
|
2008-11-04 00:08:39 +01:00
|
|
|
return;
|
|
|
|
}
|
2009-12-22 18:28:22 +01:00
|
|
|
if (size.width() >= MAX_FRAME_SIZE || size.height() >= MAX_FRAME_SIZE) {
|
2013-07-29 19:05:59 +02:00
|
|
|
qWarning() << "Not generating frame background for a size whose width or height is more than" << MAX_FRAME_SIZE << size;
|
2009-12-22 18:28:22 +01:00
|
|
|
return;
|
|
|
|
}
|
2008-11-04 00:08:39 +01:00
|
|
|
|
2014-07-14 16:32:10 +02:00
|
|
|
frame->cachedBackground = QPixmap(size);
|
2008-11-04 00:08:39 +01:00
|
|
|
frame->cachedBackground.fill(Qt::transparent);
|
|
|
|
QPainter p(&frame->cachedBackground);
|
|
|
|
p.setCompositionMode(QPainter::CompositionMode_Source);
|
|
|
|
p.setRenderHint(QPainter::SmoothPixmapTransform);
|
|
|
|
|
2014-07-14 18:13:40 +02:00
|
|
|
QRect contentRect = contentGeometry(frame, size);
|
2014-07-21 15:44:25 +02:00
|
|
|
paintCenter(p, frame, contentRect, size);
|
2008-11-04 00:08:39 +01:00
|
|
|
|
2014-07-14 20:02:47 +02:00
|
|
|
paintCorner(p, frame, FrameSvg::LeftBorder|FrameSvg::TopBorder, contentRect);
|
|
|
|
paintCorner(p, frame, FrameSvg::RightBorder|FrameSvg::TopBorder, contentRect);
|
|
|
|
paintCorner(p, frame, FrameSvg::LeftBorder|FrameSvg::BottomBorder, contentRect);
|
|
|
|
paintCorner(p, frame, FrameSvg::RightBorder|FrameSvg::BottomBorder, contentRect);
|
2014-07-14 16:47:56 +02:00
|
|
|
|
2008-11-04 00:08:39 +01:00
|
|
|
// Sides
|
2014-07-14 14:16:16 +02:00
|
|
|
const int leftHeight = q->elementSize(prefix % "left").height();
|
2014-07-14 20:02:47 +02:00
|
|
|
paintBorder(p, frame, FrameSvg::LeftBorder, QSize(frame->leftWidth, leftHeight), contentRect);
|
|
|
|
paintBorder(p, frame, FrameSvg::RightBorder, QSize(frame->rightWidth, leftHeight), contentRect);
|
2008-11-04 00:08:39 +01:00
|
|
|
|
2014-07-14 14:16:16 +02:00
|
|
|
const int topWidth = q->elementSize(prefix % "top").width();
|
2014-07-14 20:02:47 +02:00
|
|
|
paintBorder(p, frame, FrameSvg::TopBorder, QSize(topWidth, frame->topHeight), contentRect);
|
|
|
|
paintBorder(p, frame, FrameSvg::BottomBorder, QSize(topWidth, frame->bottomHeight), contentRect);
|
|
|
|
}
|
|
|
|
|
2014-07-21 15:44:25 +02:00
|
|
|
QRect FrameSvg::sectionRect(Plasma::FrameSvg::EnabledBorders borders, const QRect& contentRect, const QSize& fullSize)
|
2014-07-14 20:02:47 +02:00
|
|
|
{
|
2014-07-18 02:38:26 +02:00
|
|
|
//don't use QRect corner methods here, they have semantics that might come as unexpected.
|
|
|
|
//prefer constructing the points explicitly. e.g. from QRect::topRight docs:
|
|
|
|
//Note that for historical reasons this function returns QPoint(left() + width() -1, top()).
|
|
|
|
|
2014-07-14 20:02:47 +02:00
|
|
|
switch(borders) {
|
|
|
|
case FrameSvg::NoBorder:
|
2014-07-21 15:44:25 +02:00
|
|
|
return contentRect;
|
2014-07-14 20:02:47 +02:00
|
|
|
case FrameSvg::TopBorder:
|
2014-07-21 15:44:25 +02:00
|
|
|
return QRect(QPoint(contentRect.left(), 0), QSize(contentRect.width(), contentRect.top()));
|
2014-07-14 20:02:47 +02:00
|
|
|
case FrameSvg::BottomBorder:
|
2014-07-21 15:44:25 +02:00
|
|
|
return QRect(QPoint(contentRect.left(), contentRect.bottom()+1), QSize(contentRect.width(), fullSize.height()-contentRect.bottom()-1));
|
2014-07-14 20:02:47 +02:00
|
|
|
case FrameSvg::LeftBorder:
|
2014-07-21 15:44:25 +02:00
|
|
|
return QRect(QPoint(0, contentRect.top()), QSize(contentRect.left(), contentRect.height()));
|
2014-07-14 20:02:47 +02:00
|
|
|
case FrameSvg::RightBorder:
|
2014-07-21 15:44:25 +02:00
|
|
|
return QRect(QPoint(contentRect.right()+1, contentRect.top()), QSize(fullSize.width()-contentRect.right()-1, contentRect.height()));
|
2014-07-14 20:02:47 +02:00
|
|
|
case FrameSvg::TopBorder | FrameSvg::LeftBorder:
|
2014-07-21 15:44:25 +02:00
|
|
|
return QRect(QPoint(0, 0), QSize(contentRect.left(), contentRect.top()));
|
2014-07-14 20:02:47 +02:00
|
|
|
case FrameSvg::TopBorder | FrameSvg::RightBorder:
|
2014-07-21 15:44:25 +02:00
|
|
|
return QRect(QPoint(contentRect.right()+1, 0), QSize(fullSize.width()-contentRect.right()-1, contentRect.top()));
|
2014-07-14 20:02:47 +02:00
|
|
|
case FrameSvg::BottomBorder | FrameSvg::LeftBorder:
|
2014-07-21 15:44:25 +02:00
|
|
|
return QRect(QPoint(0, contentRect.bottom()+1), QSize(contentRect.left(), fullSize.height()-contentRect.bottom()-1));
|
2014-07-14 20:02:47 +02:00
|
|
|
case FrameSvg::BottomBorder | FrameSvg::RightBorder:
|
2014-07-21 15:44:25 +02:00
|
|
|
return QRect(QPoint(contentRect.right()+1, contentRect.bottom()+1), QSize(fullSize.width()-contentRect.right()-1, fullSize.height()-contentRect.bottom()-1));
|
2014-07-14 20:02:47 +02:00
|
|
|
default:
|
|
|
|
qWarning() << "unrecognized border" << borders;
|
|
|
|
return QRect();
|
|
|
|
}
|
|
|
|
return QRect();
|
2014-07-14 18:13:40 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
QRect FrameSvgPrivate::contentGeometry(FrameData* frame, const QSize& size) const
|
|
|
|
{
|
|
|
|
const QSize contentSize(size.width() - frame->leftWidth - frame->rightWidth, size.height() - frame->topHeight - frame->bottomHeight);
|
|
|
|
QRect contentRect(QPoint(0,0), contentSize);
|
|
|
|
if (frame->enabledBorders & FrameSvg::LeftBorder && q->hasElement(prefix % "left")) {
|
|
|
|
contentRect.translate(frame->leftWidth, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Corners
|
|
|
|
if (frame->enabledBorders & FrameSvg::TopBorder && q->hasElement(prefix % "top")) {
|
|
|
|
contentRect.translate(0, frame->topHeight);
|
|
|
|
}
|
|
|
|
return contentRect;
|
2014-07-14 14:16:16 +02:00
|
|
|
}
|
2008-11-04 00:08:39 +01:00
|
|
|
|
2014-07-21 15:44:25 +02:00
|
|
|
void FrameSvgPrivate::paintCenter(QPainter& p, FrameData* frame, const QRect& contentRect, const QSize& fullSize)
|
2014-07-14 17:00:39 +02:00
|
|
|
{
|
2014-07-21 15:44:25 +02:00
|
|
|
if (!contentRect.isEmpty()) {
|
2014-07-14 17:00:39 +02:00
|
|
|
const QString centerElementId = prefix % "center";
|
|
|
|
if (frame->tileCenter) {
|
|
|
|
QSize centerTileSize = q->elementSize(centerElementId);
|
|
|
|
QPixmap center(centerTileSize);
|
|
|
|
center.fill(Qt::transparent);
|
|
|
|
|
|
|
|
QPainter centerPainter(¢er);
|
|
|
|
centerPainter.setCompositionMode(QPainter::CompositionMode_Source);
|
|
|
|
q->paint(¢erPainter, QRect(QPoint(0, 0), centerTileSize),centerElementId);
|
|
|
|
|
|
|
|
if (frame->composeOverBorder) {
|
|
|
|
p.drawTiledPixmap(QRect(QPoint(0, 0), fullSize), center);
|
|
|
|
} else {
|
2014-07-21 15:44:25 +02:00
|
|
|
p.drawTiledPixmap(FrameSvg::sectionRect(FrameSvg::NoBorder, contentRect, fullSize), center);
|
2014-07-14 17:00:39 +02:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (frame->composeOverBorder) {
|
|
|
|
q->paint(&p, QRect(QPoint(0, 0), fullSize),
|
|
|
|
centerElementId);
|
|
|
|
} else {
|
2014-07-21 15:44:25 +02:00
|
|
|
q->paint(&p, FrameSvg::sectionRect(FrameSvg::NoBorder, contentRect, fullSize), centerElementId);
|
2014-07-14 17:00:39 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (frame->composeOverBorder) {
|
|
|
|
p.setCompositionMode(QPainter::CompositionMode_DestinationIn);
|
|
|
|
p.drawPixmap(QRect(QPoint(0, 0), fullSize), alphaMask());
|
|
|
|
p.setCompositionMode(QPainter::CompositionMode_SourceOver);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-07-14 20:02:47 +02:00
|
|
|
void FrameSvgPrivate::paintBorder(QPainter& p, FrameData* frame, const FrameSvg::EnabledBorders borders, const QSize& size, const QRect& contentRect) const
|
2014-07-14 14:16:16 +02:00
|
|
|
{
|
2014-07-21 15:44:25 +02:00
|
|
|
QString side = prefix % FrameSvg::borderToElementId(borders);
|
2014-07-14 16:37:00 +02:00
|
|
|
if (frame->enabledBorders & borders && q->hasElement(side) && !size.isEmpty()) {
|
2014-07-14 14:16:16 +02:00
|
|
|
if (frame->stretchBorders) {
|
2014-07-21 15:44:25 +02:00
|
|
|
q->paint(&p, FrameSvg::sectionRect(borders, contentRect, frame->frameSize), side);
|
2014-07-14 14:16:16 +02:00
|
|
|
} else {
|
2014-07-14 16:37:00 +02:00
|
|
|
QPixmap px(size);
|
2014-07-14 14:16:16 +02:00
|
|
|
px.fill(Qt::transparent);
|
2008-11-04 00:08:39 +01:00
|
|
|
|
2014-07-14 14:16:16 +02:00
|
|
|
QPainter sidePainter(&px);
|
2008-11-04 00:08:39 +01:00
|
|
|
sidePainter.setCompositionMode(QPainter::CompositionMode_Source);
|
2014-07-14 16:37:00 +02:00
|
|
|
q->paint(&sidePainter, QRect(QPoint(0, 0), size), side);
|
2008-11-04 00:08:39 +01:00
|
|
|
|
2014-07-21 15:44:25 +02:00
|
|
|
p.drawTiledPixmap(FrameSvg::sectionRect(borders, contentRect, frame->frameSize), px);
|
2008-11-04 00:08:39 +01:00
|
|
|
}
|
|
|
|
}
|
2014-07-14 14:16:16 +02:00
|
|
|
}
|
2008-11-04 00:08:39 +01:00
|
|
|
|
2014-07-14 20:02:47 +02:00
|
|
|
void FrameSvgPrivate::paintCorner(QPainter& p, FrameData* frame, Plasma::FrameSvg::EnabledBorders border, const QRect& contentRect) const
|
2014-07-14 16:47:56 +02:00
|
|
|
{
|
2014-07-21 15:44:25 +02:00
|
|
|
QString corner = prefix % FrameSvg::borderToElementId(border);
|
2014-07-14 16:47:56 +02:00
|
|
|
if (frame->enabledBorders & border && q->hasElement(corner)) {
|
2014-07-21 15:44:25 +02:00
|
|
|
q->paint(&p, FrameSvg::sectionRect(border, contentRect, frame->frameSize), corner);
|
2014-07-14 16:47:56 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-07-21 15:44:25 +02:00
|
|
|
QString FrameSvg::borderToElementId(FrameSvg::EnabledBorders borders)
|
2014-07-14 14:16:16 +02:00
|
|
|
{
|
|
|
|
switch(borders) {
|
|
|
|
case FrameSvg::NoBorder:
|
2014-07-14 20:02:47 +02:00
|
|
|
return QStringLiteral("center");
|
2014-07-14 14:16:16 +02:00
|
|
|
case FrameSvg::TopBorder:
|
|
|
|
return QStringLiteral("top");
|
|
|
|
case FrameSvg::BottomBorder:
|
|
|
|
return QStringLiteral("bottom");
|
|
|
|
case FrameSvg::LeftBorder:
|
|
|
|
return QStringLiteral("left");
|
|
|
|
case FrameSvg::RightBorder:
|
|
|
|
return QStringLiteral("right");
|
|
|
|
case FrameSvg::TopBorder | FrameSvg::LeftBorder:
|
|
|
|
return QStringLiteral("topleft");
|
|
|
|
case FrameSvg::TopBorder | FrameSvg::RightBorder:
|
|
|
|
return QStringLiteral("topright");
|
|
|
|
case FrameSvg::BottomBorder | FrameSvg::LeftBorder:
|
|
|
|
return QStringLiteral("bottomleft");
|
|
|
|
case FrameSvg::BottomBorder | FrameSvg::RightBorder:
|
|
|
|
return QStringLiteral("bottomright");
|
|
|
|
default:
|
|
|
|
qWarning() << "unrecognized border" << borders;
|
|
|
|
}
|
|
|
|
return QString();
|
2008-11-29 20:24:40 +01:00
|
|
|
}
|
|
|
|
|
2009-09-28 23:54:00 +02:00
|
|
|
QString FrameSvgPrivate::cacheId(FrameData *frame, const QString &prefixToSave) const
|
|
|
|
{
|
2010-09-27 19:19:06 +02:00
|
|
|
const QSize size = frameSize(frame).toSize();
|
2010-10-13 16:27:08 +02:00
|
|
|
const QLatin1Char s('_');
|
2014-04-26 01:45:47 +02:00
|
|
|
return QString::number(frame->enabledBorders) % s % QString::number(size.width()) % s % QString::number(size.height()) % s % prefixToSave % s % q->imagePath();
|
2009-09-28 23:54:00 +02:00
|
|
|
}
|
2009-01-13 22:46:07 +01:00
|
|
|
|
2009-09-28 23:02:24 +02:00
|
|
|
void FrameSvgPrivate::cacheFrame(const QString &prefixToSave, const QPixmap &background, const QPixmap &overlay)
|
2008-11-29 20:24:40 +01:00
|
|
|
{
|
2009-05-13 18:37:13 +02:00
|
|
|
if (!q->isUsingRenderingCache()) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2009-05-31 07:38:42 +02:00
|
|
|
//insert background
|
|
|
|
FrameData *frame = frames.value(prefixToSave);
|
|
|
|
|
|
|
|
if (!frame) {
|
|
|
|
return;
|
|
|
|
}
|
2008-11-29 20:24:40 +01:00
|
|
|
|
2009-09-28 23:54:00 +02:00
|
|
|
const QString id = cacheId(frame, prefixToSave);
|
2008-11-29 20:24:40 +01:00
|
|
|
|
2013-07-29 19:05:59 +02:00
|
|
|
//qDebug()<<"Saving to cache frame"<<id;
|
2009-01-13 22:46:07 +01:00
|
|
|
|
2014-05-19 14:58:19 +02:00
|
|
|
q->theme()->insertIntoCache(id, background, QString::number((qint64)q, 16) % prefixToSave);
|
2009-01-13 22:46:07 +01:00
|
|
|
|
2009-09-28 23:02:24 +02:00
|
|
|
if (!overlay.isNull()) {
|
|
|
|
//insert overlay
|
2014-05-19 14:58:19 +02:00
|
|
|
q->theme()->insertIntoCache("overlay_" % id, overlay, QString::number((qint64)q, 16) % prefixToSave % "overlay");
|
2009-09-28 23:02:24 +02:00
|
|
|
}
|
2008-11-04 00:08:39 +01:00
|
|
|
}
|
|
|
|
|
2009-09-28 23:54:00 +02:00
|
|
|
void FrameSvgPrivate::updateSizes() const
|
2008-11-04 00:08:39 +01:00
|
|
|
{
|
2013-07-29 19:05:59 +02:00
|
|
|
//qDebug() << "!!!!!!!!!!!!!!!!!!!!!! updating sizes" << prefix;
|
2008-11-04 00:08:39 +01:00
|
|
|
FrameData *frame = frames[prefix];
|
|
|
|
Q_ASSERT(frame);
|
|
|
|
|
2009-01-04 20:07:46 +01:00
|
|
|
QSize s = q->size();
|
|
|
|
q->resize();
|
2008-11-04 00:08:39 +01:00
|
|
|
frame->cachedBackground = QPixmap();
|
|
|
|
|
2014-02-21 21:13:12 +01:00
|
|
|
//This has the same size regardless the border is enabled or not
|
|
|
|
frame->fixedTopHeight = q->elementSize(prefix % "top").height();
|
|
|
|
|
|
|
|
if (q->hasElement(prefix % "hint-top-margin")) {
|
|
|
|
frame->fixedTopMargin = q->elementSize(prefix % "hint-top-margin").height();
|
|
|
|
} else {
|
|
|
|
frame->fixedTopMargin = frame->fixedTopHeight;
|
|
|
|
}
|
|
|
|
|
|
|
|
//The same, but its size depends from the margin being enabled
|
2008-11-04 00:08:39 +01:00
|
|
|
if (frame->enabledBorders & FrameSvg::TopBorder) {
|
2010-09-27 19:19:06 +02:00
|
|
|
frame->topHeight = q->elementSize(prefix % "top").height();
|
2008-11-04 00:08:39 +01:00
|
|
|
|
2010-09-27 19:19:06 +02:00
|
|
|
if (q->hasElement(prefix % "hint-top-margin")) {
|
|
|
|
frame->topMargin = q->elementSize(prefix % "hint-top-margin").height();
|
2008-11-04 00:08:39 +01:00
|
|
|
} else {
|
|
|
|
frame->topMargin = frame->topHeight;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
frame->topMargin = frame->topHeight = 0;
|
|
|
|
}
|
|
|
|
|
2014-02-21 21:13:12 +01:00
|
|
|
frame->fixedLeftWidth = q->elementSize(prefix % "left").width();
|
|
|
|
|
|
|
|
if (q->hasElement(prefix % "hint-left-margin")) {
|
|
|
|
frame->fixedLeftMargin = q->elementSize(prefix % "hint-left-margin").width();
|
|
|
|
} else {
|
|
|
|
frame->fixedLeftMargin = frame->fixedLeftWidth;
|
|
|
|
}
|
|
|
|
|
2008-11-04 00:08:39 +01:00
|
|
|
if (frame->enabledBorders & FrameSvg::LeftBorder) {
|
2010-09-27 19:19:06 +02:00
|
|
|
frame->leftWidth = q->elementSize(prefix % "left").width();
|
2008-11-04 00:08:39 +01:00
|
|
|
|
2010-09-27 19:19:06 +02:00
|
|
|
if (q->hasElement(prefix % "hint-left-margin")) {
|
|
|
|
frame->leftMargin = q->elementSize(prefix % "hint-left-margin").width();
|
2008-11-04 00:08:39 +01:00
|
|
|
} else {
|
|
|
|
frame->leftMargin = frame->leftWidth;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
frame->leftMargin = frame->leftWidth = 0;
|
|
|
|
}
|
|
|
|
|
2014-02-21 21:13:12 +01:00
|
|
|
frame->fixedRightWidth = q->elementSize(prefix % "right").width();
|
|
|
|
|
|
|
|
if (q->hasElement(prefix % "hint-right-margin")) {
|
|
|
|
frame->fixedRightMargin = q->elementSize(prefix % "hint-right-margin").width();
|
|
|
|
} else {
|
|
|
|
frame->fixedRightMargin = frame->fixedRightWidth;
|
|
|
|
}
|
|
|
|
|
2008-11-04 00:08:39 +01:00
|
|
|
if (frame->enabledBorders & FrameSvg::RightBorder) {
|
2010-09-27 19:19:06 +02:00
|
|
|
frame->rightWidth = q->elementSize(prefix % "right").width();
|
2008-11-04 00:08:39 +01:00
|
|
|
|
2010-09-27 19:19:06 +02:00
|
|
|
if (q->hasElement(prefix % "hint-right-margin")) {
|
|
|
|
frame->rightMargin = q->elementSize(prefix % "hint-right-margin").width();
|
2008-11-04 00:08:39 +01:00
|
|
|
} else {
|
|
|
|
frame->rightMargin = frame->rightWidth;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
frame->rightMargin = frame->rightWidth = 0;
|
|
|
|
}
|
|
|
|
|
2014-02-21 21:13:12 +01:00
|
|
|
frame->fixedBottomHeight = q->elementSize(prefix % "bottom").height();
|
|
|
|
|
|
|
|
if (q->hasElement(prefix % "hint-bottom-margin")) {
|
|
|
|
frame->fixedBottomMargin = q->elementSize(prefix % "hint-bottom-margin").height();
|
|
|
|
} else {
|
|
|
|
frame->fixedBottomMargin = frame->fixedBottomHeight;
|
|
|
|
}
|
|
|
|
|
2008-11-04 00:08:39 +01:00
|
|
|
if (frame->enabledBorders & FrameSvg::BottomBorder) {
|
2010-09-27 19:19:06 +02:00
|
|
|
frame->bottomHeight = q->elementSize(prefix % "bottom").height();
|
2008-11-04 00:08:39 +01:00
|
|
|
|
2010-09-27 19:19:06 +02:00
|
|
|
if (q->hasElement(prefix % "hint-bottom-margin")) {
|
|
|
|
frame->bottomMargin = q->elementSize(prefix % "hint-bottom-margin").height();
|
2008-11-04 00:08:39 +01:00
|
|
|
} else {
|
|
|
|
frame->bottomMargin = frame->bottomHeight;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
frame->bottomMargin = frame->bottomHeight = 0;
|
|
|
|
}
|
|
|
|
|
2010-09-27 19:19:06 +02:00
|
|
|
frame->composeOverBorder = (q->hasElement(prefix % "hint-compose-over-border") &&
|
2011-05-22 18:03:12 +02:00
|
|
|
q->hasElement("mask-" % prefix % "center"));
|
2010-05-06 16:16:03 +02:00
|
|
|
|
2008-11-04 00:08:39 +01:00
|
|
|
//since it's rectangular, topWidth and bottomWidth must be the same
|
2010-05-06 16:16:03 +02:00
|
|
|
//the ones that don't have a prefix is for retrocompatibility
|
2011-05-22 19:23:48 +02:00
|
|
|
frame->tileCenter = (q->hasElement("hint-tile-center") || q->hasElement(prefix % "hint-tile-center"));
|
|
|
|
frame->noBorderPadding = (q->hasElement("hint-no-border-padding") || q->hasElement(prefix % "hint-no-border-padding"));
|
|
|
|
frame->stretchBorders = (q->hasElement("hint-stretch-borders") || q->hasElement(prefix % "hint-stretch-borders"));
|
2009-01-04 20:07:46 +01:00
|
|
|
q->resize(s);
|
2008-11-04 00:08:39 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void FrameSvgPrivate::updateNeeded()
|
|
|
|
{
|
2014-07-11 16:42:55 +02:00
|
|
|
q->setElementPrefix(requestedPrefix);
|
2008-11-04 00:08:39 +01:00
|
|
|
q->clearCache();
|
|
|
|
updateSizes();
|
|
|
|
}
|
|
|
|
|
|
|
|
void FrameSvgPrivate::updateAndSignalSizes()
|
|
|
|
{
|
|
|
|
updateSizes();
|
|
|
|
emit q->repaintNeeded();
|
|
|
|
}
|
|
|
|
|
2009-09-28 23:54:00 +02:00
|
|
|
QSizeF FrameSvgPrivate::frameSize(FrameData *frame) const
|
2009-06-10 19:11:02 +02:00
|
|
|
{
|
|
|
|
if (!frame->frameSize.isValid()) {
|
|
|
|
updateSizes();
|
|
|
|
frame->frameSize = q->size();
|
|
|
|
}
|
|
|
|
|
|
|
|
return frame->frameSize;
|
|
|
|
}
|
|
|
|
|
2010-09-27 19:19:06 +02:00
|
|
|
void FrameData::ref(FrameSvg *svg)
|
|
|
|
{
|
|
|
|
references[svg] = references[svg] + 1;
|
2013-07-29 19:05:59 +02:00
|
|
|
//qDebug() << this << svg << references[svg];
|
2010-09-27 19:19:06 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
bool FrameData::deref(FrameSvg *svg)
|
|
|
|
{
|
|
|
|
references[svg] = references[svg] - 1;
|
2013-07-29 19:05:59 +02:00
|
|
|
//qDebug() << this << svg << references[svg];
|
2010-09-27 19:19:06 +02:00
|
|
|
if (references[svg] < 1) {
|
|
|
|
references.remove(svg);
|
|
|
|
}
|
|
|
|
|
|
|
|
return references.isEmpty();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool FrameData::removeRefs(FrameSvg *svg)
|
|
|
|
{
|
|
|
|
references.remove(svg);
|
|
|
|
return references.isEmpty();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool FrameData::isUsed() const
|
|
|
|
{
|
|
|
|
return !references.isEmpty();
|
|
|
|
}
|
|
|
|
|
|
|
|
int FrameData::refcount() const
|
|
|
|
{
|
|
|
|
return references.count();
|
|
|
|
}
|
2009-06-10 19:11:02 +02:00
|
|
|
|
2014-07-21 15:44:25 +02:00
|
|
|
QString FrameSvg::actualPrefix() const
|
|
|
|
{
|
|
|
|
return d->prefix;
|
|
|
|
}
|
|
|
|
|
2008-11-04 00:08:39 +01:00
|
|
|
} // Plasma namespace
|
|
|
|
|
2012-02-08 23:33:03 +01:00
|
|
|
#include "moc_framesvg.cpp"
|