2020-05-18 19:37:41 +00:00
|
|
|
/*
|
2020-08-02 22:22:19 +00:00
|
|
|
SPDX-FileCopyrightText: 2020 Vlad Zahorodnii <vlad.zahorodnii@kde.org>
|
|
|
|
|
|
|
|
SPDX-License-Identifier: GPL-2.0-or-later
|
|
|
|
*/
|
2020-05-18 19:37:41 +00:00
|
|
|
|
|
|
|
#include "xcursortheme.h"
|
|
|
|
#include "3rdparty/xcursor.h"
|
|
|
|
|
2020-07-13 16:02:45 +00:00
|
|
|
#include <QMap>
|
|
|
|
#include <QSharedData>
|
|
|
|
|
2020-05-18 19:37:41 +00:00
|
|
|
namespace KWin
|
|
|
|
{
|
|
|
|
|
2020-07-13 16:02:45 +00:00
|
|
|
class KXcursorSpritePrivate : public QSharedData
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
QImage data;
|
|
|
|
QPoint hotspot;
|
|
|
|
std::chrono::milliseconds delay;
|
|
|
|
};
|
|
|
|
|
|
|
|
class KXcursorThemePrivate : public QSharedData
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
QMap<QByteArray, QVector<KXcursorSprite>> registry;
|
|
|
|
};
|
|
|
|
|
2020-05-18 19:37:41 +00:00
|
|
|
KXcursorSprite::KXcursorSprite()
|
2020-07-13 16:02:45 +00:00
|
|
|
: d(new KXcursorSpritePrivate)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
KXcursorSprite::KXcursorSprite(const KXcursorSprite &other)
|
|
|
|
: d(other.d)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
KXcursorSprite::~KXcursorSprite()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
KXcursorSprite &KXcursorSprite::operator=(const KXcursorSprite &other)
|
2020-05-18 19:37:41 +00:00
|
|
|
{
|
2020-07-13 16:02:45 +00:00
|
|
|
d = other.d;
|
|
|
|
return *this;
|
2020-05-18 19:37:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
KXcursorSprite::KXcursorSprite(const QImage &data, const QPoint &hotspot,
|
|
|
|
const std::chrono::milliseconds &delay)
|
2020-07-13 16:02:45 +00:00
|
|
|
: d(new KXcursorSpritePrivate)
|
2020-05-18 19:37:41 +00:00
|
|
|
{
|
2020-07-13 16:02:45 +00:00
|
|
|
d->data = data;
|
|
|
|
d->hotspot = hotspot;
|
|
|
|
d->delay = delay;
|
2020-05-18 19:37:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
QImage KXcursorSprite::data() const
|
|
|
|
{
|
2020-07-13 16:02:45 +00:00
|
|
|
return d->data;
|
2020-05-18 19:37:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
QPoint KXcursorSprite::hotspot() const
|
|
|
|
{
|
2020-07-13 16:02:45 +00:00
|
|
|
return d->hotspot;
|
2020-05-18 19:37:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
std::chrono::milliseconds KXcursorSprite::delay() const
|
|
|
|
{
|
2020-07-13 16:02:45 +00:00
|
|
|
return d->delay;
|
2020-05-18 19:37:41 +00:00
|
|
|
}
|
|
|
|
|
2021-05-02 10:37:45 +00:00
|
|
|
struct XcursorThemeClosure
|
|
|
|
{
|
|
|
|
QMap<QByteArray, QVector<KXcursorSprite>> registry;
|
|
|
|
int desiredSize;
|
|
|
|
};
|
|
|
|
|
2020-05-18 19:37:41 +00:00
|
|
|
static void load_callback(XcursorImages *images, void *data)
|
|
|
|
{
|
2021-05-02 10:37:45 +00:00
|
|
|
XcursorThemeClosure *closure = static_cast<XcursorThemeClosure *>(data);
|
2020-05-18 19:37:41 +00:00
|
|
|
QVector<KXcursorSprite> sprites;
|
|
|
|
|
|
|
|
for (int i = 0; i < images->nimage; ++i) {
|
|
|
|
const XcursorImage *nativeCursorImage = images->images[i];
|
2021-05-02 10:37:45 +00:00
|
|
|
const qreal scale = std::max(qreal(1), qreal(nativeCursorImage->size) / closure->desiredSize);
|
2020-05-18 19:37:41 +00:00
|
|
|
const QPoint hotspot(nativeCursorImage->xhot, nativeCursorImage->yhot);
|
|
|
|
const std::chrono::milliseconds delay(nativeCursorImage->delay);
|
|
|
|
|
2021-02-22 13:28:50 +00:00
|
|
|
QImage data(nativeCursorImage->width, nativeCursorImage->height, QImage::Format_ARGB32_Premultiplied);
|
2021-05-02 10:37:45 +00:00
|
|
|
data.setDevicePixelRatio(scale);
|
2020-05-18 19:37:41 +00:00
|
|
|
memcpy(data.bits(), nativeCursorImage->pixels, data.sizeInBytes());
|
|
|
|
|
2021-05-02 10:37:45 +00:00
|
|
|
sprites.append(KXcursorSprite(data, hotspot / scale, delay));
|
2020-05-18 19:37:41 +00:00
|
|
|
}
|
|
|
|
|
2021-05-02 10:37:45 +00:00
|
|
|
if (!sprites.isEmpty()) {
|
|
|
|
closure->registry.insert(images->name, sprites);
|
|
|
|
}
|
2020-05-18 19:37:41 +00:00
|
|
|
XcursorImagesDestroy(images);
|
|
|
|
}
|
|
|
|
|
2020-07-13 16:02:45 +00:00
|
|
|
KXcursorTheme::KXcursorTheme()
|
|
|
|
: d(new KXcursorThemePrivate)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2021-05-02 10:37:45 +00:00
|
|
|
KXcursorTheme::KXcursorTheme(const QMap<QByteArray, QVector<KXcursorSprite>> ®istry)
|
|
|
|
: KXcursorTheme()
|
|
|
|
{
|
|
|
|
d->registry = registry;
|
|
|
|
}
|
|
|
|
|
2020-07-13 16:02:45 +00:00
|
|
|
KXcursorTheme::KXcursorTheme(const KXcursorTheme &other)
|
|
|
|
: d(other.d)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
KXcursorTheme::~KXcursorTheme()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
KXcursorTheme &KXcursorTheme::operator=(const KXcursorTheme &other)
|
|
|
|
{
|
|
|
|
d = other.d;
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
2020-05-18 19:37:41 +00:00
|
|
|
bool KXcursorTheme::isEmpty() const
|
|
|
|
{
|
2020-07-13 16:02:45 +00:00
|
|
|
return d->registry.isEmpty();
|
2020-05-18 19:37:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
QVector<KXcursorSprite> KXcursorTheme::shape(const QByteArray &name) const
|
|
|
|
{
|
2020-07-13 16:02:45 +00:00
|
|
|
return d->registry.value(name);
|
2020-05-18 19:37:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
KXcursorTheme KXcursorTheme::fromTheme(const QString &themeName, int size, qreal dpr)
|
|
|
|
{
|
2021-05-02 10:37:45 +00:00
|
|
|
// Xcursors don't support HiDPI natively so we fake it by scaling the desired cursor
|
|
|
|
// size. The device pixel ratio argument acts only as a hint. The real scale factor
|
|
|
|
// of every cursor sprite will be computed in the loading closure.
|
|
|
|
XcursorThemeClosure closure;
|
|
|
|
closure.desiredSize = size;
|
|
|
|
xcursor_load_theme(themeName.toUtf8().constData(), size * dpr, load_callback, &closure);
|
2020-05-18 19:37:41 +00:00
|
|
|
|
2021-05-02 10:37:45 +00:00
|
|
|
if (closure.registry.isEmpty()) {
|
|
|
|
return KXcursorTheme();
|
|
|
|
}
|
2020-05-18 19:37:41 +00:00
|
|
|
|
2021-05-02 10:37:45 +00:00
|
|
|
return KXcursorTheme(closure.registry);
|
2020-05-18 19:37:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace KWin
|