Qt
Internal/Contributor docs for the Qt SDK. Note: These are NOT official API docs; those are found at https://doc.qt.io/
Loading...
Searching...
No Matches
qwasmclipboard.cpp
Go to the documentation of this file.
1// Copyright (C) 2018 The Qt Company Ltd.
2// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR GPL-3.0-only
3// Qt-Security score:significant reason:default
4
7#include "qwasmdom.h"
8#include "qwasmevent.h"
9#include "qwasmwindow.h"
10
11#include <private/qstdweb_p.h>
12
13#include <QCoreApplication>
14#include <qpa/qwindowsysteminterface.h>
15#include <QBuffer>
16#include <QString>
17
18#include <emscripten/val.h>
19
21using namespace emscripten;
22
23static void commonCopyEvent(val event)
24{
25 QMimeData *_mimes = QWasmIntegration::get()->getWasmClipboard()->mimeData(QClipboard::Clipboard);
26 if (!_mimes)
27 return;
28
29 // doing it this way seems to sanitize the text better that calling data() like down below
30 if (_mimes->hasText()) {
31 event["clipboardData"].call<void>("setData", val("text/plain"),
32 _mimes->text().toEcmaString());
33 }
34 if (_mimes->hasHtml()) {
35 event["clipboardData"].call<void>("setData", val("text/html"), _mimes->html().toEcmaString());
36 }
37
38 for (auto mimetype : _mimes->formats()) {
39 if (mimetype.contains("text/"))
40 continue;
41 QByteArray ba = _mimes->data(mimetype);
42 if (!ba.isEmpty())
43 event["clipboardData"].call<void>("setData", mimetype.toEcmaString(),
44 val(ba.constData()));
45 }
46
47 event.call<void>("preventDefault");
48}
49
50void QWasmClipboard::cut(val event)
51{
53 // Send synthetic Ctrl+X to make the app cut data to Qt's clipboard
54 QWindowSystemInterface::handleKeyEvent(
55 0, QEvent::KeyPress, Qt::Key_X, Qt::ControlModifier, "X");
56 }
57
58 commonCopyEvent(event);
59}
60
61void QWasmClipboard::copy(val event)
62{
64 // Send synthetic Ctrl+C to make the app copy data to Qt's clipboard
65 QWindowSystemInterface::handleKeyEvent(
66 0, QEvent::KeyPress, Qt::Key_C, Qt::ControlModifier, "C");
67 }
68 commonCopyEvent(event);
69}
70
71void QWasmClipboard::paste(val event)
72{
73 event.call<void>("preventDefault"); // prevent browser from handling drop event
74
75 QWasmIntegration::get()->getWasmClipboard()->sendClipboardData(event);
76}
77
79{
80 val clipboard = val::global("navigator")["clipboard"];
81
82 const bool hasPermissionsApi = !val::global("navigator")["permissions"].isUndefined();
83 m_hasClipboardApi = !clipboard.isUndefined() && !clipboard["readText"].isUndefined();
84
85 if (m_hasClipboardApi && hasPermissionsApi)
86 initClipboardPermissions();
87
89 val document = val::global("document");
90 m_documentCut = QWasmEventHandler(document, "cut", QWasmClipboard::cut);
91 m_documentCopy = QWasmEventHandler(document, "copy", QWasmClipboard::copy);
92 m_documentPaste = QWasmEventHandler(document, "paste", QWasmClipboard::paste);
93 }
94}
95
99
100QMimeData *QWasmClipboard::mimeData(QClipboard::Mode mode)
101{
102 if (mode != QClipboard::Clipboard)
103 return nullptr;
104
105 return QPlatformClipboard::mimeData(mode);
106}
107
108void QWasmClipboard::setMimeData(QMimeData *mimeData, QClipboard::Mode mode)
109{
110 // handle setText/ setData programmatically
111 QPlatformClipboard::setMimeData(mimeData, mode);
112 if (m_hasClipboardApi)
113 writeToClipboardApi();
114 else
115 writeToClipboard();
116}
117
119{
120 if (event.type != EventType::KeyDown || !event.modifiers.testFlag(Qt::ControlModifier))
122
123 if (event.key != Qt::Key_C && event.key != Qt::Key_V && event.key != Qt::Key_X)
125
126 const bool isPaste = event.key == Qt::Key_V;
127
128 return m_hasClipboardApi && !isPaste
131}
132
133bool QWasmClipboard::supportsMode(QClipboard::Mode mode) const
134{
135 return mode == QClipboard::Clipboard;
136}
137
138bool QWasmClipboard::ownsMode(QClipboard::Mode mode) const
139{
140 Q_UNUSED(mode);
141 return false;
142}
143
144void QWasmClipboard::initClipboardPermissions()
145{
146 val permissions = val::global("navigator")["permissions"];
147
148 qstdweb::Promise::make(permissions, "query", { .catchFunc = [](emscripten::val) {} }, ([]() {
149 val readPermissionsMap = val::object();
150 readPermissionsMap.set("name", val("clipboard-read"));
151 return readPermissionsMap;
152 })());
153 qstdweb::Promise::make(permissions, "query", { .catchFunc = [](emscripten::val) {} }, ([]() {
154 val readPermissionsMap = val::object();
155 readPermissionsMap.set("name", val("clipboard-write"));
156 return readPermissionsMap;
157 })());
158}
159
161{
162 return m_hasClipboardApi;
163}
164
166{
167 // Chrome uses global handlers
168 return val::global("window")["chrome"].isUndefined();
169}
170
171void QWasmClipboard::writeToClipboardApi()
172{
173 Q_ASSERT(m_hasClipboardApi);
174
175 QMimeData *mimeData = this->mimeData(QClipboard::Clipboard);
176 if (!mimeData)
177 return;
178
179 // Support for plain text, html and images (png) are standardized,
180 // copy those to the clipboard data object.
181 emscripten::val clipboardData = emscripten::val::object();
182 for (const QString &mimetype: mimeData->formats()) {
183 if (mimetype == QLatin1String("text/plain")) {
184 emscripten::val text = mimeData->text().toEcmaString();
185 clipboardData.set(mimetype.toEcmaString(), text);
186 } else if (mimetype == QLatin1String("text/html")) {
187 emscripten::val html = mimeData->html().toEcmaString();
188 clipboardData.set(mimetype.toEcmaString(), html);
189 } else if (mimetype.contains(QLatin1String("image"))) {
190 // Serialize the Qt image data to browser supported png
191 QImage img = qvariant_cast<QImage>(mimeData->imageData());
192 QByteArray ba;
193 QBuffer buffer(&ba);
194 buffer.open(QIODevice::WriteOnly);
195 img.save(&buffer, "PNG");
196
197 qstdweb::Blob blob = qstdweb::Blob::fromArrayBuffer(qstdweb::Uint8Array::copyFrom(ba).buffer());
198 clipboardData.set(std::string("image/png"), blob.val());
199 }
200 }
201
202 // Return if there is no data (creating an empty ClipboardItem is an error)
203 if (val::global("Object").call<val>("keys", clipboardData)["length"].as<int>() == 0)
204 return;
205
206 // Write a single clipboard item containing the data formats to the clipboard
207 emscripten::val clipboardItem = val::global("ClipboardItem").new_(clipboardData);
208 emscripten::val clipboardItemArray = emscripten::val::array();
209 clipboardItemArray.call<void>("push", clipboardItem);
210 val navigator = val::global("navigator");
211 qstdweb::Promise::make(
212 navigator["clipboard"], "write",
213 {
214 .catchFunc = [](emscripten::val error) {
215 qWarning() << "clipboard error"
216 << QString::fromStdString(error["name"].as<std::string>())
217 << QString::fromStdString(error["message"].as<std::string>());
218 }
219 },
220 clipboardItemArray);
221}
222
223void QWasmClipboard::writeToClipboard()
224{
225 // this works for firefox, chrome by generating
226 // copy event, but not safari
227 // execCommand has been deemed deprecated in the docs, but browsers do not seem
228 // interested in removing it. There is no replacement, so we use it here.
229 val document = val::global("document");
230 document.call<val>("execCommand", val("copy"));
231}
232
233void QWasmClipboard::sendClipboardData(emscripten::val event)
234{
235 dom::DataTransfer *transfer = new dom::DataTransfer(event["clipboardData"]);
236 const auto mimeCallback = std::function([transfer](QMimeData *data) {
237
238 // Persist clipboard data so that the app can read it when handling the CTRL+V
239 QWasmIntegration::get()->clipboard()->QPlatformClipboard::setMimeData(data, QClipboard::Clipboard);
240 QWindowSystemInterface::handleKeyEvent(0, QEvent::KeyPress, Qt::Key_V,
241 Qt::ControlModifier, "V");
242 QWindowSystemInterface::handleKeyEvent(0, QEvent::KeyRelease, Qt::Key_V,
243 Qt::ControlModifier, "V");
244 delete transfer;
245 });
246
247 transfer->toMimeDataWithFile(mimeCallback);
248}
249QT_END_NAMESPACE
virtual ~QWasmClipboard()
static bool shouldInstallWindowEventHandlers()
void setMimeData(QMimeData *data, QClipboard::Mode mode=QClipboard::Clipboard) override
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.
Definition qwasmdom.h:30
static void commonCopyEvent(val event)
EventType
Definition qwasmevent.h:23