76 for (
const auto &format : mimeData.formats()) {
77 if (format.startsWith(
"text/")) {
78 auto data = mimeData.data(format);
79 setData(format.toStdString(), QString::fromLocal8Bit(data).toStdString());
80 }
else if (format ==
"application/x-qt-image") {
81 auto image = qvariant_cast<QImage>(mimeData.imageData());
83 QBuffer buffer(&byteArray);
84 buffer.open(QIODevice::WriteOnly);
85 image.save(&buffer,
"PNG");
88 setData(format.toStdString(),
"QB64" + QString::fromLocal8Bit(byteArray.toBase64()).toStdString());
90 auto data = mimeData.data(format);
91 auto encoded =
"QB64" + QString::fromLocal8Bit(data.toBase64()).toStdString();
92 setData(format.toStdString(), std::move(encoded));
102 enum class ItemKind {
110 MimeContext(
int itemCount, std::function<
void(QMimeData *)> callback)
111 :m_remainingItemCount(itemCount), m_callback(callback)
117 if (--m_remainingItemCount > 0)
121 allUrls.append(mimeData->urls());
122 allUrls.append(fileUrls);
123 mimeData->setUrls(allUrls);
125 m_callback(mimeData);
131 for (QUrl fileUrl: fileUrls) {
132 if (!QWasmFileEngineHandler::isWasmFileName(fileUrl.toString()))
133 QFile(fileUrl.toLocalFile()).remove();
139 QMimeData *mimeData =
new QMimeData();
140 QList<QUrl> fileUrls;
143 int m_remainingItemCount;
144 std::function<
void(QMimeData *)> m_callback;
147 const auto items = webDataTransfer[
"items"];
148 const int itemCount = items[
"length"].as<
int>();
149 const int fileCount = webDataTransfer[
"files"][
"length"].as<
int>();
150 MimeContext *mimeContext =
new MimeContext(itemCount, callback);
152 for (
int i = 0; i < itemCount; ++i) {
153 const auto item = items[i];
154 const auto itemKind =
155 item[
"kind"].as<
std::string>() ==
"string" ? ItemKind::String : ItemKind::File;
156 const auto itemMimeType = QString::fromStdString(item[
"type"].as<std::string>());
159 case ItemKind::File: {
160 qstdweb::File webfile(item.call<emscripten::val>(
"getAsFile"));
165 if (qstdweb::haveAsyncify()) {
166 QUrl fileUrl(QWasmFileEngineHandler::addFile(webfile));
167 mimeContext->fileUrls.append(fileUrl);
168 mimeContext->deref();
171 if (webfile.size() > 1e+9) {
172 qWarning() <<
"File is too large (> 1GB) and will be skipped. File size is" << webfile.size();
173 mimeContext->deref();
178 QByteArray fileContent(webfile.size(), Qt::Uninitialized);
179 webfile.stream(fileContent.data(), [=]() {
180 QDir qtTmpDir(
"/qt/tmp/");
181 qtTmpDir.mkpath(qtTmpDir.path());
183 QUrl fileUrl = QUrl::fromLocalFile(qtTmpDir.filePath(QString::fromStdString(webfile.name())));
184 mimeContext->fileUrls.append(fileUrl);
186 QFile file(fileUrl.toLocalFile());
187 if (!file.open(QFile::WriteOnly)) {
188 qWarning() <<
"File was not opened";
189 mimeContext->deref();
192 if (file.write(fileContent) < 0)
193 qWarning() <<
"Write failed";
195 mimeContext->deref();
205 QString mimeFormat = QString::fromStdString(webfile.type());
206 if (fileCount == 1 && mimeFormat.contains(
"image/")) {
208 if (image.loadFromData(fileContent))
209 mimeContext->mimeData->setImageData(image);
214 case ItemKind::String:
215 if (itemMimeType.contains(
"STRING", Qt::CaseSensitive)
216 || itemMimeType.contains(
"TEXT", Qt::CaseSensitive)) {
217 mimeContext->deref();
221 QString data = QString::fromEcmaString(webDataTransfer.call<emscripten::val>(
222 "getData", emscripten::val(itemMimeType.toStdString())));
224 if (!data.isEmpty()) {
225 if (itemMimeType ==
"text/html")
226 mimeContext->mimeData->setHtml(data);
227 else if (itemMimeType.isEmpty() || itemMimeType ==
"text/plain")
228 mimeContext->mimeData->setText(data);
229 else if (itemMimeType.isEmpty() || itemMimeType ==
"text/uri-list") {
232 mimeContext->mimeData->setUrls(urls);
233 }
else if (itemMimeType ==
"application/x-qt-image") {
234 if (data.startsWith(
"QB64")) {
236 auto ba = QByteArray::fromBase64(QByteArray::fromStdString(data.toStdString()));
238 image.loadFromData(ba);
239 mimeContext->mimeData->setImageData(image);
243 if (data.startsWith(
"QB64")) {
245 mimeContext->mimeData->setData(itemMimeType,
246 QByteArray::fromBase64(QByteArray::fromStdString(
247 data.toStdString())));
249 mimeContext->mimeData->setData(itemMimeType, data.toLocal8Bit());
253 mimeContext->deref();
304void drawImageToWebImageDataArray(
const QImage &sourceImage, emscripten::val destinationImageData,
305 const QRect &sourceRect)
307 Q_ASSERT_X(destinationImageData[
"constructor"][
"name"].as<
std::string>() ==
"ImageData",
308 Q_FUNC_INFO,
"The destination should be an ImageData instance");
310 constexpr int BytesPerColor = 4;
311 if (sourceRect.width() == sourceImage.width()) {
318 auto imageMemory = emscripten::typed_memory_view(sourceRect.width() * sourceRect.height()
320 sourceImage.constScanLine(sourceRect.y()));
321 destinationImageData[
"data"].call<
void>(
322 "set", imageMemory, sourceRect.y() * sourceImage.width() * BytesPerColor);
331 for (
int row = 0; row < sourceRect.height(); ++row) {
332 auto scanlineMemory =
333 emscripten::typed_memory_view(sourceRect.width() * BytesPerColor,
334 sourceImage.constScanLine(row + sourceRect.y())
335 + BytesPerColor * sourceRect.x());
336 destinationImageData[
"data"].call<
void>(
"set", scanlineMemory,
337 (sourceRect.y() + row) * sourceImage.width()
339 + sourceRect.x() * BytesPerColor);