11#include <private/qstdweb_p.h>
13#include <QCoreApplication>
14#include <qpa/qwindowsysteminterface.h>
18#include <emscripten/val.h>
21using namespace emscripten;
30 if (_mimes->hasText()) {
31 event[
"clipboardData"].call<
void>(
"setData", val(
"text/plain"),
32 _mimes->text().toEcmaString());
34 if (_mimes->hasHtml()) {
35 event[
"clipboardData"].call<
void>(
"setData", val(
"text/html"), _mimes->html().toEcmaString());
38 for (
auto mimetype : _mimes->formats()) {
39 if (mimetype.contains(
"text/"))
41 QByteArray ba = _mimes->data(mimetype);
43 event[
"clipboardData"].call<
void>(
"setData", mimetype.toEcmaString(),
47 event.call<
void>(
"preventDefault");
54 QWindowSystemInterface::handleKeyEvent(
55 0, QEvent::KeyPress, Qt::Key_X, Qt::ControlModifier,
"X");
58 commonCopyEvent(event);
65 QWindowSystemInterface::handleKeyEvent(
66 0, QEvent::KeyPress, Qt::Key_C, Qt::ControlModifier,
"C");
68 commonCopyEvent(event);
78 val clipboard = val::global(
"navigator")[
"clipboard"];
80 const bool hasPermissionsApi = !val::global(
"navigator")[
"permissions"].isUndefined();
81 m_hasClipboardApi = !clipboard.isUndefined() && !clipboard[
"readText"].isUndefined();
83 if (m_hasClipboardApi && hasPermissionsApi)
84 initClipboardPermissions();
87 val document = val::global(
"document");
88 m_documentCut = QWasmEventHandler(document,
"cut", QWasmClipboard::cut);
89 m_documentCopy = QWasmEventHandler(document,
"copy", QWasmClipboard::copy);
90 m_documentPaste = QWasmEventHandler(document,
"paste", QWasmClipboard::paste);
100 if (mode != QClipboard::Clipboard)
103 return QPlatformClipboard::mimeData(mode);
109 QPlatformClipboard::setMimeData(mimeData, mode);
110 if (m_hasClipboardApi)
111 writeToClipboardApi();
118 if (event.type !=
EventType::KeyDown || !event.modifiers.testFlag(Qt::ControlModifier))
121 if (event.key != Qt::Key_C && event.key != Qt::Key_V && event.key != Qt::Key_X)
124 const bool isPaste = event.key == Qt::Key_V;
126 return m_hasClipboardApi && !isPaste
133 return mode == QClipboard::Clipboard;
144 val permissions = val::global(
"navigator")[
"permissions"];
146 qstdweb::Promise::make(permissions,
"query", { .catchFunc = [](emscripten::val) {} }, ([]() {
147 val readPermissionsMap = val::object();
148 readPermissionsMap.set(
"name", val(
"clipboard-read"));
149 return readPermissionsMap;
151 qstdweb::Promise::make(permissions,
"query", { .catchFunc = [](emscripten::val) {} }, ([]() {
152 val readPermissionsMap = val::object();
153 readPermissionsMap.set(
"name", val(
"clipboard-write"));
154 return readPermissionsMap;
160 return m_hasClipboardApi;
166 return val::global(
"window")[
"chrome"].isUndefined();
171 Q_ASSERT(m_hasClipboardApi);
173 QMimeData *mimeData =
this->mimeData(QClipboard::Clipboard);
179 emscripten::val clipboardData = emscripten::val::object();
180 for (
const QString &mimetype: mimeData->formats()) {
181 if (mimetype == QLatin1String(
"text/plain")) {
182 emscripten::val text = mimeData->text().toEcmaString();
183 clipboardData.set(mimetype.toEcmaString(), text);
184 }
else if (mimetype == QLatin1String(
"text/html")) {
185 emscripten::val html = mimeData->html().toEcmaString();
186 clipboardData.set(mimetype.toEcmaString(), html);
187 }
else if (mimetype.contains(QLatin1String(
"image"))) {
189 QImage img = qvariant_cast<QImage>(mimeData->imageData());
192 buffer.open(QIODevice::WriteOnly);
193 img.save(&buffer,
"PNG");
195 qstdweb::Blob blob = qstdweb::Blob::fromArrayBuffer(qstdweb::Uint8Array::copyFrom(ba).buffer());
196 clipboardData.set(std::string(
"image/png"), blob.val());
201 if (val::global(
"Object").call<val>(
"keys", clipboardData)[
"length"].as<
int>() == 0)
205 emscripten::val clipboardItem = val::global(
"ClipboardItem").new_(clipboardData);
206 emscripten::val clipboardItemArray = emscripten::val::array();
207 clipboardItemArray.call<
void>(
"push", clipboardItem);
208 val navigator = val::global(
"navigator");
209 qstdweb::Promise::make(
210 navigator[
"clipboard"],
"write",
212 .catchFunc = [](emscripten::val error) {
213 qWarning() <<
"clipboard error"
214 << QString::fromStdString(error[
"name"].as<std::string>())
215 << QString::fromStdString(error[
"message"].as<std::string>());
227 val document = val::global(
"document");
228 document.call<val>(
"execCommand", val(
"copy"));
234 const auto mimeCallback = std::function([transfer](QMimeData *data) {
238 QWindowSystemInterface::handleKeyEvent(0, QEvent::KeyPress, Qt::Key_V,
239 Qt::ControlModifier,
"V");
240 QWindowSystemInterface::handleKeyEvent(0, QEvent::KeyRelease, Qt::Key_V,
241 Qt::ControlModifier,
"V");
245 transfer->toMimeDataWithFile(mimeCallback);
virtual ~QWasmClipboard()
static bool shouldInstallWindowEventHandlers()
void setMimeData(QMimeData *data, QClipboard::Mode mode=QClipboard::Clipboard) override
@ NativeClipboardEventNeeded
@ NativeClipboardEventAndCopiedDataNeeded
bool supportsMode(QClipboard::Mode mode) const override
QMimeData * mimeData(QClipboard::Mode mode=QClipboard::Clipboard) override
ProcessKeyboardResult processKeyboard(const KeyEvent &event)
bool ownsMode(QClipboard::Mode mode) const override
static QWasmIntegration * get()
QWasmClipboard * getWasmClipboard()
Combined button and popup list for selecting options.
static void commonCopyEvent(val event)