moment/src/clipboard.h
miruka 606fe3d022 Use a thread lock for Clipboard.qimage()
Prevents this: user paste image (image provider retrieves the clipboard
image to show it as a thumbnail),
and immediatly presses enter (retrieves the image to convert it and send
to Python); the two calls would run at the same time instead of one
waiting for the other to set the cache.
2020-07-16 16:24:16 -04:00

118 lines
3.0 KiB
C++

// SPDX-License-Identifier: LGPL-3.0-or-later
// The Clipboard class exposes system clipboard management and retrieval
// to QML.
#ifndef CLIPBOARD_H
#define CLIPBOARD_H
#include <QBuffer>
#include <QByteArray>
#include <QClipboard>
#include <QGuiApplication>
#include <QIODevice>
#include <QImage>
#include <QMimeData>
#include <QMutex>
#include <QMutexLocker>
#include <QObject>
class Clipboard : public QObject {
Q_OBJECT
Q_PROPERTY(QString text READ text WRITE setText NOTIFY textChanged)
Q_PROPERTY(QByteArray image READ image WRITE setImage NOTIFY imageChanged)
Q_PROPERTY(bool hasImage READ hasImage NOTIFY hasImageChanged)
Q_PROPERTY(QString selection READ selection WRITE setSelection
NOTIFY selectionChanged)
Q_PROPERTY(bool supportsSelection READ supportsSelection CONSTANT)
public:
explicit Clipboard(QObject *parent = nullptr) : QObject(parent) {
connect(this->clipboard, &QClipboard::dataChanged,
this, &Clipboard::mainClipboardChanged);
connect(this->clipboard, &QClipboard::selectionChanged,
this, &Clipboard::selectionChanged);
}
// Normal primary clipboard
QString text() const {
return this->clipboard->text(QClipboard::Clipboard);
}
void setText(const QString &text) const {
this->clipboard->setText(text, QClipboard::Clipboard);
}
QImage *qimage() {
QMutexLocker locker(&(this->imageRetrievalLock));
if (this->cachedImage.isNull())
this->cachedImage = this->clipboard->image();
return &(this->cachedImage);
}
QByteArray image() {
QByteArray byteArray;
QBuffer buffer(&byteArray);
buffer.open(QIODevice::WriteOnly);
QImage *image = this->qimage();
// minimum compression, fastest to not freeze the UI
image->save(&buffer, "PNG", 100);
buffer.close();
return byteArray;
}
void setImage(const QByteArray &image) const { // TODO
Q_UNUSED(image)
}
bool hasImage() const {
return this->clipboard->mimeData()->hasImage();
}
// X11 select-middle-click-paste clipboard
QString selection() const {
return this->clipboard->text(QClipboard::Selection);
}
void setSelection(const QString &text) const {
if (this->clipboard->supportsSelection()) {
this->clipboard->setText(text, QClipboard::Selection);
}
}
// Info
bool supportsSelection() const {
return this->clipboard->supportsSelection();
}
signals:
void contentChanged();
void textChanged();
void imageChanged();
void hasImageChanged();
void selectionChanged();
private:
QClipboard *clipboard = QGuiApplication::clipboard();
QImage cachedImage = QImage();
QMutex imageRetrievalLock;
void mainClipboardChanged() {
this->contentChanged();
this->cachedImage = QImage();
this->hasImage() ? this->imageChanged() : this->textChanged();
this->hasImageChanged();
};
};
#endif