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 QWasmIntegration::get()->getWasmClipboard()->sendClipboardData(event);
74}
75
77{
78 val clipboard = val::global("navigator")["clipboard"];
79
80 const bool hasPermissionsApi = !val::global("navigator")["permissions"].isUndefined();
81 m_hasClipboardApi = !clipboard.isUndefined() && !clipboard["readText"].isUndefined();
82
83 if (m_hasClipboardApi && hasPermissionsApi)
84 initClipboardPermissions();
85
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);
91 }
92}
93
97
98QMimeData *QWasmClipboard::mimeData(QClipboard::Mode mode)
99{
100 if (mode != QClipboard::Clipboard)
101 return nullptr;
102
103 return QPlatformClipboard::mimeData(mode);
104}
105
106void QWasmClipboard::setMimeData(QMimeData *mimeData, QClipboard::Mode mode)
107{
108 // handle setText/ setData programmatically
109 QPlatformClipboard::setMimeData(mimeData, mode);
110 if (m_hasClipboardApi)
111 writeToClipboardApi();
112 else
113 writeToClipboard();
114}
115
117{
118 if (event.type != EventType::KeyDown || !event.modifiers.testFlag(Qt::ControlModifier))
120
121 if (event.key != Qt::Key_C && event.key != Qt::Key_V && event.key != Qt::Key_X)
123
124 const bool isPaste = event.key == Qt::Key_V;
125
126 return m_hasClipboardApi && !isPaste
129}
130
131bool QWasmClipboard::supportsMode(QClipboard::Mode mode) const
132{
133 return mode == QClipboard::Clipboard;
134}
135
136bool QWasmClipboard::ownsMode(QClipboard::Mode mode) const
137{
138 Q_UNUSED(mode);
139 return false;
140}
141
142void QWasmClipboard::initClipboardPermissions()
143{
144 val permissions = val::global("navigator")["permissions"];
145
146 qstdweb::Promise::make(permissions, "query", { .catchFunc = [](emscripten::val) {} }, ([]() {
147 val readPermissionsMap = val::object();
148 readPermissionsMap.set("name", val("clipboard-read"));
149 return readPermissionsMap;
150 })());
151 qstdweb::Promise::make(permissions, "query", { .catchFunc = [](emscripten::val) {} }, ([]() {
152 val readPermissionsMap = val::object();
153 readPermissionsMap.set("name", val("clipboard-write"));
154 return readPermissionsMap;
155 })());
156}
157
159{
160 return m_hasClipboardApi;
161}
162
164{
165 // Chrome uses global handlers
166 return val::global("window")["chrome"].isUndefined();
167}
168
169void QWasmClipboard::writeToClipboardApi()
170{
171 Q_ASSERT(m_hasClipboardApi);
172
173 QMimeData *mimeData = this->mimeData(QClipboard::Clipboard);
174 if (!mimeData)
175 return;
176
177 // Support for plain text, html and images (png) are standardized,
178 // copy those to the clipboard data object.
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"))) {
188 // Serialize the Qt image data to browser supported png
189 QImage img = qvariant_cast<QImage>(mimeData->imageData());
190 QByteArray ba;
191 QBuffer buffer(&ba);
192 buffer.open(QIODevice::WriteOnly);
193 img.save(&buffer, "PNG");
194
195 qstdweb::Blob blob = qstdweb::Blob::fromArrayBuffer(qstdweb::Uint8Array::copyFrom(ba).buffer());
196 clipboardData.set(std::string("image/png"), blob.val());
197 }
198 }
199
200 // Return if there is no data (creating an empty ClipboardItem is an error)
201 if (val::global("Object").call<val>("keys", clipboardData)["length"].as<int>() == 0)
202 return;
203
204 // Write a single clipboard item containing the data formats to the clipboard
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",
211 {
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>());
216 }
217 },
218 clipboardItemArray);
219}
220
221void QWasmClipboard::writeToClipboard()
222{
223 // this works for firefox, chrome by generating
224 // copy event, but not safari
225 // execCommand has been deemed deprecated in the docs, but browsers do not seem
226 // interested in removing it. There is no replacement, so we use it here.
227 val document = val::global("document");
228 document.call<val>("execCommand", val("copy"));
229}
230
231void QWasmClipboard::sendClipboardData(emscripten::val event)
232{
233 dom::DataTransfer *transfer = new dom::DataTransfer(event["clipboardData"]);
234 const auto mimeCallback = std::function([transfer](QMimeData *data) {
235
236 // Persist clipboard data so that the app can read it when handling the CTRL+V
237 QWasmIntegration::get()->clipboard()->QPlatformClipboard::setMimeData(data, QClipboard::Clipboard);
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");
242 delete transfer;
243 });
244
245 transfer->toMimeDataWithFile(mimeCallback);
246}
247QT_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