8#include <QtGui/private/qinternalmimedata_p.h>
9#include <QtCore/qbytearraymatcher.h>
10#include <QtCore/qmap.h>
11#include <QtCore/qurl.h>
12#include <QtCore/qdir.h>
13#include <QtCore/qdebug.h>
14#include <QtCore/qbuffer.h>
15#include <QtGui/qimagereader.h>
16#include <QtGui/qimagewriter.h>
23using namespace Qt::StringLiterals;
26
27
28
29#define BMP_LCS_sRGB 0x73524742
30#define BMP_LCS_GM_IMAGES 0x00000004L
71 QByteArray msg = func;
72 msg +=
": Unable to convert DIB image. The image converter plugin for '";
74 msg +=
"' is not available. Available formats: ";
75 const auto &formats = QImageReader::supportedImageFormats();
76 for (
const QByteArray &af : formats) {
83static inline bool readDib(QBuffer &buffer, QImage &img)
86 if (!reader.canRead()) {
87 qWarning(
"%s", msgConversionError(
__FUNCTION__, dibFormatC).constData());
94#if QT_CONFIG(imageformat_bmp)
95static QByteArray writeDib(
const QImage &img)
99 buffer.open(QIODevice::ReadWrite);
100 QImageWriter writer(&buffer, dibFormatC);
101 if (!writer.canWrite()) {
102 qWarning(
"%s", msgConversionError(
__FUNCTION__, dibFormatC).constData());
105 if (!writer.write(img))
110static bool qt_write_dibv5(QDataStream &s, QImage image)
112 QIODevice* d = s.device();
113 if (!d->isWritable())
117 qsizetype bpl_bmp = qsizetype(image.width()) * 4;
118 qsizetype size = bpl_bmp * image.height();
119 if (qsizetype(DWORD(size)) != size)
122 BMP_BITMAPV5HEADER bi;
123 ZeroMemory(&bi,
sizeof(bi));
124 bi.bV5Size =
sizeof(BMP_BITMAPV5HEADER);
125 bi.bV5Width = image.width();
126 bi.bV5Height = image.height();
129 bi.bV5Compression = BI_BITFIELDS;
130 bi.bV5SizeImage = DWORD(bpl_bmp * image.height());
131 bi.bV5XPelsPerMeter = 0;
132 bi.bV5YPelsPerMeter = 0;
134 bi.bV5ClrImportant = 0;
135 bi.bV5BlueMask = 0x000000ff;
136 bi.bV5GreenMask = 0x0000ff00;
137 bi.bV5RedMask = 0x00ff0000;
138 bi.bV5AlphaMask = 0xff000000;
139 bi.bV5CSType = BMP_LCS_sRGB;
140 bi.bV5Intent = BMP_LCS_GM_IMAGES;
142 d->write(
reinterpret_cast<
const char*>(&bi), bi.bV5Size);
143 if (s.status() != QDataStream::Ok)
146 d->write(
reinterpret_cast<
const char *>(&bi.bV5RedMask),
sizeof(bi.bV5RedMask));
147 if (s.status() != QDataStream::Ok)
150 d->write(
reinterpret_cast<
const char *>(&bi.bV5GreenMask),
sizeof(bi.bV5GreenMask));
151 if (s.status() != QDataStream::Ok)
154 d->write(
reinterpret_cast<
const char *>(&bi.bV5BlueMask),
sizeof(bi.bV5BlueMask));
155 if (s.status() != QDataStream::Ok)
158 if (image.format() != QImage::Format_ARGB32)
159 image = std::move(image).convertToFormat(QImage::Format_ARGB32);
161 auto *buf =
new uchar[bpl_bmp];
163 memset(buf, 0, size_t(bpl_bmp));
164 for (
int y=image.height()-1; y>=0; y--) {
166 const QRgb *p =
reinterpret_cast<
const QRgb *>(image.constScanLine(y));
167 const QRgb *end = p + image.width();
170 int alpha = qAlpha(*p);
172 *b++ = uchar(qBlue(*p));
173 *b++ = uchar(qGreen(*p));
174 *b++ = uchar(qRed(*p));
184 d->write(
reinterpret_cast<
const char *>(buf), bpl_bmp);
185 if (s.status() != QDataStream::Ok) {
197static int getCf(
const FORMATETC &formatetc)
199 return formatetc.cfFormat;
205 formatetc.cfFormat = CLIPFORMAT(cf);
206 formatetc.dwAspect = DVASPECT_CONTENT;
207 formatetc.lindex = -1;
208 formatetc.ptd =
nullptr;
209 formatetc.tymed = TYMED_HGLOBAL;
215 HGLOBAL hData = GlobalAlloc(0, SIZE_T(data.size()));
219 void *out = GlobalLock(hData);
220 memcpy(out, data.data(), size_t(data.size()));
229 if (!(tymed & TYMED_HGLOBAL) && (tymed & TYMED_ISTREAM)) {
230 IStream *stream =
nullptr;
231 if (FAILED(::CreateStreamOnHGlobal(hData, TRUE, &stream))) {
235 pmedium->tymed = TYMED_ISTREAM;
236 pmedium->pstm = stream;
237 pmedium->pUnkForRelease =
nullptr;
241 pmedium->tymed = TYMED_HGLOBAL;
242 pmedium->hGlobal = hData;
243 pmedium->pUnkForRelease =
nullptr;
250 FORMATETC formatetc = setCf(cf);
251 formatetc.lindex = lindex;
253 if (pDataObj->GetData(&formatetc, &s) == S_OK) {
257 if (s.hGlobal && GlobalSize(s.hGlobal) > 0) {
258 const void *val = GlobalLock(s.hGlobal);
259 data = QByteArray::fromRawData(
reinterpret_cast<
const char *>(val),
int(GlobalSize(s.hGlobal)));
261 GlobalUnlock(s.hGlobal);
263 ReleaseStgMedium(&s);
265 if (data.isEmpty()) {
267 formatetc.tymed = TYMED_ISTREAM;
268 if (pDataObj->GetData(&formatetc, &s) == S_OK) {
270 ULONG actualRead = 0;
271 LARGE_INTEGER pos = {{0, 0}};
273 HRESULT hr = s.pstm->Seek(pos, STREAM_SEEK_SET,
nullptr);
274 while(SUCCEEDED(hr)){
275 hr = s.pstm->Read(szBuffer,
sizeof(szBuffer), &actualRead);
276 if (SUCCEEDED(hr) && actualRead > 0) {
277 data += QByteArray::fromRawData(szBuffer,
int(actualRead));
279 if (actualRead !=
sizeof(szBuffer))
283 ReleaseStgMedium(&s);
291 FORMATETC formatetc = setCf(cf);
292 if (pDataObj->QueryGetData(&formatetc) != S_OK){
293 formatetc.tymed = TYMED_ISTREAM;
294 return pDataObj->QueryGetData(&formatetc) == S_OK;
299#ifndef QT_NO_DEBUG_STREAM
300QDebug operator<<(QDebug d,
const FORMATETC &tc)
302 QDebugStateSaver saver(d);
304 d <<
"FORMATETC(cfFormat=" << tc.cfFormat <<
' ';
305 switch (tc.cfFormat) {
325 d <<
"CF_UNICODETEXT";
328 d <<
"CF_ENHMETAFILE";
331 d << QWindowsMimeRegistry::clipboardFormatName(tc.cfFormat);
334 d <<
", dwAspect=" << tc.dwAspect <<
", lindex=" << tc.lindex
335 <<
", tymed=" << tc.tymed <<
", ptd=" << tc.ptd <<
')';
339QDebug operator<<(QDebug d, IDataObject *dataObj)
341 QDebugStateSaver saver(d);
346 IEnumFORMATETC *enumFormatEtc;
347 if (SUCCEEDED(dataObj->EnumFormatEtc(DATADIR_GET, &enumFormatEtc)) && enumFormatEtc) {
348 FORMATETC formatEtc[1];
350 if (SUCCEEDED(enumFormatEtc->Reset())) {
351 while (SUCCEEDED(enumFormatEtc->Next(1, formatEtc, &fetched)) && fetched)
352 d << formatEtc[0] <<
',';
353 enumFormatEtc->Release();
371 bool convertFromMime(
const FORMATETC &formatetc,
const QMimeData *mimeData, STGMEDIUM *pmedium)
const override;
377 int cf = getCf(formatetc);
378 return (cf == CF_UNICODETEXT || (cf == CF_TEXT && GetACP() != CP_UTF8)) && mimeData->hasText();
382
383
384
385
388 if (canConvertFromMime(formatetc, mimeData)) {
390 int cf = getCf(formatetc);
392 data = mimeData->text().toLocal8Bit();
394 int maxsize=data.size()+data.size()/40+3;
395 QByteArray r(maxsize,
'\0');
397 const char* d = data.data();
398 const int s = data.size();
401 for (
int i=0; i<s; i++) {
413 if (j+3 >= maxsize) {
414 maxsize += maxsize/4;
420 return setData(r, pmedium);
422 if (cf == CF_UNICODETEXT) {
423 QString str = mimeData->text();
424 const QChar *u = str.unicode();
426 const int s = str.length();
427 int maxsize = s + s/40 + 3;
431 for (
int i=0; i < s; ++i) {
435 if (*u == u'\n' && !cr)
440 if (ri+3 >= maxsize) {
441 maxsize += maxsize/4;
447 const int byteLength = res.length() *
int(
sizeof(ushort));
448 QByteArray r(byteLength + 2,
'\0');
449 memcpy(r.data(), res.unicode(), size_t(byteLength));
452 return setData(r, pmedium);
460 return mimeType.startsWith(u"text/plain")
461 && (canGetData(CF_UNICODETEXT, pDataObj)
462 || canGetData(CF_TEXT, pDataObj));
467 int cf = getCf(formatetc);
468 if (cf == CF_UNICODETEXT || cf == CF_TEXT)
469 return u"text/plain"_s;
476 QList<FORMATETC> formatics;
477 if (mimeType.startsWith(u"text/plain") && mimeData->hasText()) {
478 formatics += setCf(CF_UNICODETEXT);
479 if (GetACP() != CP_UTF8)
480 formatics += setCf(CF_TEXT);
489 if (canConvertToMime(mime, pDataObj)) {
491 QByteArray data = getData(CF_UNICODETEXT, pDataObj);
492 if (!data.isEmpty()) {
493 str = QString::fromWCharArray(
reinterpret_cast<
const wchar_t *>(data.constData()));
494 str.replace(
"\r\n"_L1,
"\n"_L1);
496 data = getData(CF_TEXT, pDataObj);
497 if (!data.isEmpty()) {
498 const char* d = data.data();
499 const unsigned s =
unsigned(qstrlen(d));
500 QByteArray r(data.size()+1,
'\0');
503 for (
unsigned i = 0; i < s; ++i) {
509 str = QString::fromLocal8Bit(r);
512 if (preferredType.id() == QMetaType::QString)
515 ret =
std::move(str).toUtf8();
517 qCDebug(lcQpaMime) <<
__FUNCTION__ << ret;
529 bool convertFromMime(
const FORMATETC &formatetc,
const QMimeData *mimeData, STGMEDIUM *pmedium)
const override;
538 CF_INETURL_W = registerMimeType(u"UniformResourceLocatorW"_s);
539 CF_INETURL = registerMimeType(u"UniformResourceLocator"_s);
544 if (mimeData->hasUrls() && getCf(formatetc) == CF_HDROP) {
545 const auto urls = mimeData->urls();
546 for (
const QUrl &url : urls) {
547 if (url.isLocalFile())
551 return (getCf(formatetc) == CF_INETURL_W || getCf(formatetc) == CF_INETURL) && mimeData->hasUrls();
556 if (canConvertFromMime(formatetc, mimeData)) {
557 if (getCf(formatetc) == CF_HDROP) {
558 const auto &urls = mimeData->urls();
559 QStringList fileNames;
560 size_t size =
sizeof(DROPFILES) + 2;
561 for (
const QUrl &url : urls) {
562 const QString fn = QDir::toNativeSeparators(url.toLocalFile());
564 size +=
sizeof(ushort) * size_t(fn.length() + 1);
565 fileNames.append(fn);
569 QByteArray result(
int(size),
'\0');
570 auto* d =
reinterpret_cast<DROPFILES *>(result.data());
571 d->pFiles =
sizeof(DROPFILES);
572 GetCursorPos(&d->pt);
574 char *files = (
reinterpret_cast<
char*>(d)) + d->pFiles;
577 auto *f =
reinterpret_cast<
wchar_t *>(files);
578 for (
int i=0; i<fileNames.size(); i++) {
579 const auto l = size_t(fileNames.at(i).length());
580 memcpy(f, fileNames.at(i).data(), l *
sizeof(ushort));
586 return setData(result, pmedium);
588 if (getCf(formatetc) == CF_INETURL_W) {
589 const auto urls = mimeData->urls();
591 if (!urls.isEmpty()) {
592 const QString url = urls.at(0).toString();
593 result = QByteArray(
reinterpret_cast<
const char *>(url.data()),
594 url.length() *
int(
sizeof(ushort)));
598 return setData(result, pmedium);
600 if (getCf(formatetc) == CF_INETURL) {
601 const auto urls = mimeData->urls();
604 result = urls.at(0).toString().toLocal8Bit();
605 return setData(result, pmedium);
614 return mimeType == u"text/uri-list"
615 && (canGetData(CF_HDROP, pDataObj) || canGetData(CF_INETURL_W, pDataObj) || canGetData(CF_INETURL, pDataObj));
621 if (getCf(formatetc) == CF_HDROP || getCf(formatetc) == CF_INETURL_W || getCf(formatetc) == CF_INETURL)
622 format = u"text/uri-list"_s;
628 QList<FORMATETC> formatics;
629 if (mimeType == u"text/uri-list") {
630 if (canConvertFromMime(setCf(CF_HDROP), mimeData))
631 formatics += setCf(CF_HDROP);
632 if (canConvertFromMime(setCf(CF_INETURL_W), mimeData))
633 formatics += setCf(CF_INETURL_W);
634 if (canConvertFromMime(setCf(CF_INETURL), mimeData))
635 formatics += setCf(CF_INETURL);
642 if (mimeType == u"text/uri-list") {
643 if (canGetData(CF_HDROP, pDataObj)) {
644 QList<QVariant> urls;
646 QByteArray data = getData(CF_HDROP, pDataObj);
650 const auto *hdrop =
reinterpret_cast<
const DROPFILES *>(data.constData());
652 const auto *filesw =
reinterpret_cast<
const wchar_t *>(data.constData() + hdrop->pFiles);
655 QString fileurl = QString::fromWCharArray(filesw + i);
656 urls += QUrl::fromLocalFile(fileurl);
657 i += fileurl.length()+1;
660 const char* files =
reinterpret_cast<
const char *>(data.constData() + hdrop->pFiles);
663 urls += QUrl::fromLocalFile(QString::fromLocal8Bit(files+i));
664 i +=
int(strlen(files+i))+1;
668 if (preferredType.id() == QMetaType::QUrl && urls.size() == 1)
672 }
else if (canGetData(CF_INETURL_W, pDataObj)) {
673 QByteArray data = getData(CF_INETURL_W, pDataObj);
676 return QUrl(QString::fromWCharArray(
reinterpret_cast<
const wchar_t *>(data.constData())));
677 }
else if (canGetData(CF_INETURL, pDataObj)) {
678 QByteArray data = getData(CF_INETURL, pDataObj);
681 return QUrl(QString::fromLocal8Bit(data.constData()));
694 bool convertFromMime(
const FORMATETC &formatetc,
const QMimeData *mimeData, STGMEDIUM * pmedium)
const override;
708 CF_HTML = registerMimeType(u"HTML Format"_s);
713 QList<FORMATETC> formatetcs;
714 if (mimeType == u"text/html" && (!mimeData->html().isEmpty()))
715 formatetcs += setCf(CF_HTML);
721 if (getCf(formatetc) == CF_HTML)
722 return u"text/html"_s;
728 return mimeType == u"text/html" && canGetData(CF_HTML, pDataObj);
734 return getCf(formatetc) == CF_HTML && (!mimeData->html().isEmpty());
738
739
740
741
742
743
744
745
746
747
748
751 Q_UNUSED(preferredType);
753 if (canConvertToMime(mime, pDataObj)) {
754 QByteArray html = getData(CF_HTML, pDataObj);
755 static constexpr auto startMatcher = qMakeStaticByteArrayMatcher(
"StartHTML:");
756 static constexpr auto endMatcher = qMakeStaticByteArrayMatcher(
"EndHTML:");
757 qCDebug(lcQpaMime) <<
__FUNCTION__ <<
"raw:" << html;
758 int start = startMatcher.indexIn(html);
759 int end = endMatcher.indexIn(html);
762 int startOffset = start + 10;
764 while (html.at(i) !=
'\r' && html.at(i) !=
'\n')
766 QByteArray bytecount = html.mid(startOffset, i - startOffset);
767 start = bytecount.toInt();
771 int endOffset = end + 8;
773 while (html.at(i) !=
'\r' && html.at(i) !=
'\n')
775 QByteArray bytecount = html.mid(endOffset , i - endOffset);
776 end = bytecount.toInt();
779 if (end > start && start > 0) {
780 html = html.mid(start, end - start);
781 html.replace(
'\r',
"");
782 result = QString::fromUtf8(html);
790 if (canConvertFromMime(formatetc, mimeData)) {
791 QByteArray data = mimeData->html().toUtf8();
794 "StartHTML:0000000107\r\n"
795 "EndHTML:0000000000\r\n"
796 "StartFragment:0000000000\r\n"
797 "EndFragment:0000000000\r\n\r\n";
799 static constexpr auto startFragmentMatcher = qMakeStaticByteArrayMatcher(
"<!--StartFragment-->");
800 static constexpr auto endFragmentMatcher = qMakeStaticByteArrayMatcher(
"<!--EndFragment-->");
802 if (startFragmentMatcher.indexIn(data) == -1)
803 result +=
"<!--StartFragment-->";
805 if (endFragmentMatcher.indexIn(data) == -1)
806 result +=
"<!--EndFragment-->";
809 QByteArray pos = QByteArray::number(result.size());
810 memcpy(
reinterpret_cast<
char *>(result.data() + 53 - pos.length()), pos.constData(), size_t(pos.length()));
813 pos = QByteArray::number(startFragmentMatcher.indexIn(result) + 20);
814 memcpy(
reinterpret_cast<
char *>(result.data() + 79 - pos.length()), pos.constData(), size_t(pos.length()));
815 pos = QByteArray::number(endFragmentMatcher.indexIn(result));
816 memcpy(
reinterpret_cast<
char *>(result.data() + 103 - pos.length()), pos.constData(), size_t(pos.length()));
818 return setData(result, pmedium);
824#if QT_CONFIG(imageformat_bmp)
825class QWindowsMimeImage :
public QWindowsMimeConverter
830 bool canConvertFromMime(
const FORMATETC &formatetc,
const QMimeData *mimeData)
const override;
831 bool convertFromMime(
const FORMATETC &formatetc,
const QMimeData *mimeData, STGMEDIUM * pmedium)
const override;
832 QList<FORMATETC> formatsForMime(
const QString &mimeType,
const QMimeData *mimeData)
const override;
835 bool canConvertToMime(
const QString &mimeType, IDataObject *pDataObj)
const override;
836 QVariant convertToMime(
const QString &mime, IDataObject *pDataObj, QMetaType preferredType)
const override;
837 QString mimeForFormat(
const FORMATETC &formatetc)
const override;
839 bool hasOriginalDIBV5(IDataObject *pDataObj)
const;
843QWindowsMimeImage::QWindowsMimeImage()
845 CF_PNG = RegisterClipboardFormat(L"PNG");
848QList<FORMATETC> QWindowsMimeImage::formatsForMime(
const QString &mimeType,
const QMimeData *mimeData)
const
850 QList<FORMATETC> formatetcs;
851 if (mimeData->hasImage() && mimeType == u"application/x-qt-image") {
852 auto image = qvariant_cast<QImage>(mimeData->imageData());
853 if (!image.isNull() && image.hasAlphaChannel()) {
857 formatetcs += setCf(CF_PNG);
858 formatetcs += setCf(CF_DIBV5);
860 formatetcs += setCf(CF_DIB);
862 if (!formatetcs.isEmpty())
863 qCDebug(lcQpaMime) <<
__FUNCTION__ << mimeType << formatetcs;
867QString QWindowsMimeImage::mimeForFormat(
const FORMATETC &formatetc)
const
869 int cf = getCf(formatetc);
870 if (cf == CF_DIB || cf == CF_DIBV5 || cf ==
int(CF_PNG))
871 return u"application/x-qt-image"_s;
875bool QWindowsMimeImage::canConvertToMime(
const QString &mimeType, IDataObject *pDataObj)
const
877 return mimeType == u"application/x-qt-image"
878 && (canGetData(CF_DIB, pDataObj) || canGetData(CF_PNG, pDataObj));
881bool QWindowsMimeImage::canConvertFromMime(
const FORMATETC &formatetc,
const QMimeData *mimeData)
const
883 int cf = getCf(formatetc);
884 if (!mimeData->hasImage())
886 const QImage image = qvariant_cast<QImage>(mimeData->imageData());
891 return cf == CF_DIBV5 || cf == CF_DIB
892 || (cf ==
int(CF_PNG) && image.hasAlphaChannel());
895bool QWindowsMimeImage::convertFromMime(
const FORMATETC &formatetc,
const QMimeData *mimeData, STGMEDIUM * pmedium)
const
897 int cf = getCf(formatetc);
898 if ((cf == CF_DIB || cf == CF_DIBV5 || cf ==
int(CF_PNG)) && mimeData->hasImage()) {
899 auto img = qvariant_cast<QImage>(mimeData->imageData());
904 if (img.format() > QImage::Format_ARGB32)
905 img = std::move(img).convertToFormat(QImage::Format_RGB32);
906 const QByteArray ba = writeDib(img);
908 return setData(ba, pmedium);
909 }
else if (cf ==
int(CF_PNG)) {
911 const bool written = buffer.open(QIODevice::WriteOnly) && img.save(&buffer,
"PNG");
914 return setData(ba, pmedium, formatetc.tymed);
916 QDataStream s(&ba, QIODevice::WriteOnly);
917 s.setByteOrder(QDataStream::LittleEndian);
918 if (qt_write_dibv5(s, std::move(img)))
919 return setData(ba, pmedium);
925bool QWindowsMimeImage::hasOriginalDIBV5(IDataObject *pDataObj)
const
927 bool isSynthesized =
true;
928 IEnumFORMATETC *pEnum =
nullptr;
929 HRESULT res = pDataObj->EnumFormatEtc(1, &pEnum);
930 if (res == S_OK && pEnum) {
932 while ((res = pEnum->Next(1, &fc,
nullptr)) == S_OK) {
934 CoTaskMemFree(fc.ptd);
935 if (fc.cfFormat == CF_DIB)
937 if (fc.cfFormat == CF_DIBV5) {
938 isSynthesized =
false;
944 return !isSynthesized;
947QVariant QWindowsMimeImage::convertToMime(
const QString &mimeType, IDataObject *pDataObj, QMetaType preferredType)
const
949 Q_UNUSED(preferredType);
951 if (mimeType != u"application/x-qt-image")
955 const bool canGetDibV5 = canGetData(CF_DIBV5, pDataObj);
956 const bool hasOrigDibV5 = canGetDibV5 ? hasOriginalDIBV5(pDataObj) :
false;
957 qCDebug(lcQpaMime) <<
"canGetDibV5:" << canGetDibV5 <<
"hasOrigDibV5:" << hasOrigDibV5;
959 if (!hasOrigDibV5 && canGetData(CF_PNG, pDataObj)) {
960 qCDebug(lcQpaMime) <<
"Decoding PNG";
962 QByteArray data = getData(CF_PNG, pDataObj);
963 if (img.loadFromData(data,
"PNG")) {
968 if (canGetDibV5 || canGetData(CF_DIB, pDataObj)) {
969 qCDebug(lcQpaMime) <<
"Decoding DIB";
971 QByteArray data = getData(canGetDibV5 ? CF_DIBV5 : CF_DIB, pDataObj);
972 QBuffer buffer(&data);
973 if (readDib(buffer, img))
988 bool convertFromMime(
const FORMATETC &formatetc,
const QMimeData *mimeData, STGMEDIUM * pmedium)
const override;
997 QMap<
int, QString> outFormats;
998 QMap<
int, QString> inFormats;
1004 outFormats.insert(registerMimeType(u"application/x-color"_s), u"application/x-color"_s);
1005 inFormats.insert(registerMimeType(u"application/x-color"_s), u"application/x-color"_s);
1011 return formatetc.tymed & TYMED_HGLOBAL
1012 && outFormats.contains(formatetc.cfFormat)
1013 && mimeData->formats().contains(outFormats.value(formatetc.cfFormat));
1018 if (canConvertFromMime(formatetc, mimeData)) {
1020 if (outFormats.value(getCf(formatetc)) == u"text/html") {
1022 QString html = mimeData->html();
1024 const QChar *u = html.unicode();
1026 const int s = html.length();
1027 int maxsize = s + s/40 + 3;
1028 res.resize(maxsize);
1031 for (
int i=0; i < s; ++i) {
1035 if (*u == u'\n' && !cr)
1040 if (ri+3 >= maxsize) {
1041 maxsize += maxsize/4;
1042 res.resize(maxsize);
1047 const int byteLength = res.length() *
int(
sizeof(ushort));
1048 QByteArray r(byteLength + 2,
'\0');
1049 memcpy(r.data(), res.unicode(), size_t(byteLength));
1051 r[byteLength+1] = 0;
1054#if QT_CONFIG(draganddrop)
1055 data = QInternalMimeData::renderDataHelper(outFormats.value(getCf(formatetc)), mimeData);
1058 return setData(data, pmedium);
1065 QList<FORMATETC> formatetcs;
1066 const auto mit =
std::find(outFormats.cbegin(), outFormats.cend(), mimeType);
1067 if (mit != outFormats.cend() && mimeData->formats().contains(mimeType))
1068 formatetcs += setCf(mit.key());
1074 const auto mit =
std::find(inFormats.cbegin(), inFormats.cend(), mimeType);
1075 return mit != inFormats.cend() && canGetData(mit.key(), pDataObj);
1081 if (canConvertToMime(mimeType, pDataObj)) {
1082 QByteArray data = getData(inFormats.key(mimeType), pDataObj);
1083 if (!data.isEmpty()) {
1084 qCDebug(lcQpaMime) <<
__FUNCTION__;
1085 if (mimeType == u"text/html" && preferredType == QMetaType(QMetaType::QString)) {
1087 val = QString::fromWCharArray(
reinterpret_cast<
const wchar_t *>(data.constData()));
1098 return inFormats.value(getCf(formatetc));
1109 bool convertFromMime(
const FORMATETC &formatetc,
const QMimeData *mimeData, STGMEDIUM * pmedium)
const override;
1118 mutable QMap<
int, QString> formats;
1119 static QStringList ianaTypes;
1120 static QStringList excludeList;
1129 if (ianaTypes.isEmpty()) {
1130 ianaTypes.append(u"application/"_s);
1131 ianaTypes.append(u"audio/"_s);
1132 ianaTypes.append(u"example/"_s);
1133 ianaTypes.append(u"image/"_s);
1134 ianaTypes.append(u"message/"_s);
1135 ianaTypes.append(u"model/"_s);
1136 ianaTypes.append(u"multipart/"_s);
1137 ianaTypes.append(u"text/"_s);
1138 ianaTypes.append(u"video/"_s);
1141 if (excludeList.isEmpty()) {
1142 excludeList.append(u"HTML Format"_s);
1143 excludeList.append(u"UniformResourceLocator"_s);
1144 excludeList.append(u"text/html"_s);
1145 excludeList.append(u"text/plain"_s);
1146 excludeList.append(u"text/uri-list"_s);
1147 excludeList.append(u"application/x-qt-image"_s);
1148 excludeList.append(u"application/x-color"_s);
1155#if QT_CONFIG(draganddrop)
1156 return formatetc.tymed & TYMED_HGLOBAL
1157 && (formats.contains(formatetc.cfFormat)
1158 && QInternalMimeData::hasFormatHelper(formats.value(formatetc.cfFormat), mimeData));
1161 Q_UNUSED(formatetc);
1162 return formatetc.tymed & TYMED_HGLOBAL
1163 && formats.contains(formatetc.cfFormat);
1169#if QT_CONFIG(draganddrop)
1170 return canConvertFromMime(formatetc, mimeData)
1171 && setData(QInternalMimeData::renderDataHelper(formats.value(getCf(formatetc)), mimeData), pmedium);
1174 Q_UNUSED(formatetc);
1182 QList<FORMATETC> formatetcs;
1183 auto mit =
std::find(formats.begin(), formats.end(), mimeType);
1185 if (mit == formats.end() && !excludeList.contains(mimeType, Qt::CaseInsensitive))
1186 mit = formats.insert(registerMimeType(mimeType), mimeType);
1187 if (mit != formats.end())
1188 formatetcs += setCf(mit.key());
1190 if (!formatetcs.isEmpty())
1191 qCDebug(lcQpaMime) <<
__FUNCTION__ << mimeType << formatetcs;
1198 return mimeType.startsWith(QLatin1StringView(
x_qt_windows_mime), Qt::CaseInsensitive);
1204 int n = mimeType.lastIndexOf(u'\"') - len;
1205 QString ret = mimeType.mid(len, n);
1207 const int beginPos = mimeType.indexOf(u";index=");
1208 if (beginPos > -1) {
1209 const int endPos = mimeType.indexOf(u';', beginPos + 1);
1210 const int indexStartPos = beginPos + 7;
1212 *lindex = QStringView{mimeType}.mid(indexStartPos, endPos == -1 ? endPos : endPos - indexStartPos).toInt();
1222 if (isCustomMimeType(mimeType)) {
1224 QString clipFormat = customMimeType(mimeType);
1225 const UINT cf = RegisterClipboardFormat(
reinterpret_cast<
const wchar_t *> (clipFormat.utf16()));
1226 return canGetData(
int(cf), pDataObj);
1229 const auto mit =
std::find(formats.cbegin(), formats.cend(), mimeType);
1230 const int cf = mit != formats.cend() ? mit.key() : registerMimeType(mimeType);
1231 return canGetData(cf, pDataObj);
1236 Q_UNUSED(preferredType);
1238 if (canConvertToMime(mimeType, pDataObj)) {
1240 if (isCustomMimeType(mimeType)) {
1242 QString clipFormat = customMimeType(mimeType, &lindex);
1243 const UINT cf = RegisterClipboardFormat(
reinterpret_cast<
const wchar_t *> (clipFormat.utf16()));
1244 data = getData(
int(cf), pDataObj, lindex);
1246 const auto mit =
std::find(formats.cbegin(), formats.cend(), mimeType);
1247 const int cf = mit != formats.cend() ? mit.key() : registerMimeType(mimeType);
1248 data = getData(cf, pDataObj);
1250 if (!data.isEmpty())
1258 QString format = formats.value(getCf(formatetc));
1259 if (!format.isEmpty())
1262 const QString clipFormat = QWindowsMimeRegistry::clipboardFormatName(getCf(formatetc));
1263 if (!clipFormat.isEmpty()) {
1264#if QT_CONFIG(draganddrop)
1265 if (QInternalMimeData::canReadData(clipFormat))
1266 format = clipFormat;
1267 else if ((formatetc.cfFormat >= 0xC000)){
1269 if (!excludeList.contains(clipFormat, Qt::CaseInsensitive)) {
1271 bool ianaType =
false;
1272 int sz = ianaTypes.size();
1273 for (
int i = 0; i < sz; i++) {
1274 if (clipFormat.startsWith(ianaTypes[i], Qt::CaseInsensitive)) {
1280 format = QLatin1StringView(x_qt_windows_mime) + clipFormat + u'"';
1282 format = clipFormat;
1292
1293
1294
1295
1296
1302 qDeleteAll(m_mimes.begin(), m_mimes.begin() + m_internalMimeCount);
1307 ensureInitialized();
1308 for (
int i = m_mimes.size()-1; i >= 0; --i) {
1309 if (m_mimes.at(i)->canConvertToMime(mimeType, pDataObj))
1310 return m_mimes.at(i);
1317 qCDebug(lcQpaMime) <<
"QWindowsMimeConverter::allMimesForFormats()";
1318 ensureInitialized();
1319 QStringList formats;
1320 LPENUMFORMATETC FAR fmtenum;
1321 HRESULT hr = pDataObj->EnumFormatEtc(DATADIR_GET, &fmtenum);
1323 if (hr == NOERROR) {
1325 while (S_OK == fmtenum->Next(1, &fmtetc,
nullptr)) {
1326 for (
int i= m_mimes.size() - 1; i >= 0; --i) {
1327 QString format = m_mimes.at(i)->mimeForFormat(fmtetc);
1328 if (!format.isEmpty() && !formats.contains(format)) {
1331 qCDebug(lcQpaMime) <<
__FUNCTION__ << fmtetc << format;
1336 CoTaskMemFree(fmtetc.ptd);
1340 qCDebug(lcQpaMime) << pDataObj << formats;
1346 ensureInitialized();
1347 qCDebug(lcQpaMime) <<
__FUNCTION__ << formatetc;
1348 for (
int i = m_mimes.size()-1; i >= 0; --i) {
1349 if (m_mimes.at(i)->canConvertFromMime(formatetc, mimeData))
1350 return m_mimes.at(i);
1357 ensureInitialized();
1358 QList<FORMATETC> formatics;
1359#if !QT_CONFIG(draganddrop)
1362 formatics.reserve(20);
1363 const QStringList formats = QInternalMimeData::formatsHelper(mimeData);
1364 for (
int f = 0; f < formats.size(); ++f) {
1365 for (
int i = m_mimes.size() - 1; i >= 0; --i)
1366 formatics += m_mimes.at(i)->formatsForMime(formats.at(f), mimeData);
1374 if (m_internalMimeCount == 0) {
1375 m_internalMimeCount = -1;
1376#if QT_CONFIG(imageformat_bmp)
1377 (
void)
new QWindowsMimeImage;
1384 m_internalMimeCount = m_mimes.size();
1385 Q_ASSERT(m_internalMimeCount > 0);
1391 wchar_t buf[256] = {0};
1392 return GetClipboardFormatName(UINT(cf), buf, 255)
1393 ? QString::fromWCharArray(buf) : QString();
1397 IDataObject *pDataObj,
1398 QMetaType preferredType,
1399 QString *formatIn )
const
1401 for (
const QString &format : mimeTypes) {
1402 if (
const QWindowsMimeConverter *converter = converterToMime(format, pDataObj)) {
1403 if (converter->canConvertToMime(format, pDataObj)) {
1404 const QVariant dataV = converter->convertToMime(format, pDataObj, preferredType);
1405 if (dataV.isValid()) {
1406 qCDebug(lcQpaMime) <<
__FUNCTION__ << mimeTypes <<
"\nFormat: "
1407 << format << pDataObj <<
" returns " << dataV;
1415 qCDebug(lcQpaMime) <<
__FUNCTION__ <<
"fails" << mimeTypes << pDataObj << preferredType.id();
1421 ensureInitialized();
1422 m_mimes.append(mime);
1426
1427
1428
1429
1430
1431
1434 const QString mimeType = isCustomMimeType(mime) ? customMimeType(mime) : mime;
1435 const UINT f = RegisterClipboardFormat(
reinterpret_cast<
const wchar_t *> (mimeType.utf16()));
1437 qErrnoWarning(
"QWindowsMimeRegistry::registerMimeType: Failed to register clipboard format "
1438 "for %s", qPrintable(mime));
QList< FORMATETC > formatsForMime(const QString &mimeType, const QMimeData *mimeData) const override
Returns a QList of FORMATETC structures representing the different windows clipboard formats that can...
bool canConvertToMime(const QString &mimeType, IDataObject *pDataObj) const override
Returns true if the converter can convert to the mimeType from the available formats in pDataObj.
QString mimeForFormat(const FORMATETC &formatetc) const override
Returns the mime type that will be created form the format specified in formatetc,...
bool convertFromMime(const FORMATETC &formatetc, const QMimeData *mimeData, STGMEDIUM *pmedium) const override
Convert the mimeData to the format specified in formatetc.
QVariant convertToMime(const QString &mime, IDataObject *pDataObj, QMetaType preferredType) const override
Returns a QVariant containing the converted data for mimeType from pDataObj.
bool canConvertFromMime(const FORMATETC &formatetc, const QMimeData *mimeData) const override
Returns true if the converter can convert from the mimeData to the format specified in formatetc.
QString mimeForFormat(const FORMATETC &formatetc) const override
Returns the mime type that will be created form the format specified in formatetc,...
QList< FORMATETC > formatsForMime(const QString &mimeType, const QMimeData *mimeData) const override
Returns a QList of FORMATETC structures representing the different windows clipboard formats that can...
bool canConvertToMime(const QString &mimeType, IDataObject *pDataObj) const override
Returns true if the converter can convert to the mimeType from the available formats in pDataObj.
QVariant convertToMime(const QString &mime, IDataObject *pDataObj, QMetaType preferredType) const override
Returns a QVariant containing the converted data for mimeType from pDataObj.
bool convertFromMime(const FORMATETC &formatetc, const QMimeData *mimeData, STGMEDIUM *pmedium) const override
Convert the mimeData to the format specified in formatetc.
bool canConvertFromMime(const FORMATETC &formatetc, const QMimeData *mimeData) const override
Returns true if the converter can convert from the mimeData to the format specified in formatetc.
Singleton container for all relevant information.
bool canConvertFromMime(const FORMATETC &formatetc, const QMimeData *mimeData) const override
Returns true if the converter can convert from the mimeData to the format specified in formatetc.
bool convertFromMime(const FORMATETC &formatetc, const QMimeData *mimeData, STGMEDIUM *pmedium) const override
Convert the mimeData to the format specified in formatetc.
QString mimeForFormat(const FORMATETC &formatetc) const override
Returns the mime type that will be created form the format specified in formatetc,...
bool canConvertToMime(const QString &mimeType, IDataObject *pDataObj) const override
Returns true if the converter can convert to the mimeType from the available formats in pDataObj.
QVariant convertToMime(const QString &mime, IDataObject *pDataObj, QMetaType preferredType) const override
Returns a QVariant containing the converted data for mimeType from pDataObj.
QList< FORMATETC > formatsForMime(const QString &mimeType, const QMimeData *mimeData) const override
Returns a QList of FORMATETC structures representing the different windows clipboard formats that can...
Manages the list of QWindowsMimeConverter instances.
QVariant convertToMime(const QStringList &mimeTypes, IDataObject *pDataObj, QMetaType preferredType, QString *format=nullptr) const
QWindowsMimeConverter * converterFromMime(const FORMATETC &formatetc, const QMimeData *mimeData) const
void registerMime(QWindowsMimeConverter *mime)
QStringList allMimesForFormats(IDataObject *pDataObj) const
QList< FORMATETC > allFormatsForMime(const QMimeData *mimeData) const
QWindowsMimeConverter * converterToMime(const QString &mimeType, IDataObject *pDataObj) const
bool canConvertToMime(const QString &mimeType, IDataObject *pDataObj) const override
Returns true if the converter can convert to the mimeType from the available formats in pDataObj.
bool canConvertFromMime(const FORMATETC &formatetc, const QMimeData *mimeData) const override
Returns true if the converter can convert from the mimeData to the format specified in formatetc.
QVariant convertToMime(const QString &mime, LPDATAOBJECT pDataObj, QMetaType preferredType) const override
bool convertFromMime(const FORMATETC &formatetc, const QMimeData *mimeData, STGMEDIUM *pmedium) const override
Convert the mimeData to the format specified in formatetc.
QList< FORMATETC > formatsForMime(const QString &mimeType, const QMimeData *mimeData) const override
Returns a QList of FORMATETC structures representing the different windows clipboard formats that can...
QString mimeForFormat(const FORMATETC &formatetc) const override
Returns the mime type that will be created form the format specified in formatetc,...
QList< FORMATETC > formatsForMime(const QString &mimeType, const QMimeData *mimeData) const override
Returns a QList of FORMATETC structures representing the different windows clipboard formats that can...
QString mimeForFormat(const FORMATETC &formatetc) const override
Returns the mime type that will be created form the format specified in formatetc,...
bool canConvertToMime(const QString &mimeType, IDataObject *pDataObj) const override
Returns true if the converter can convert to the mimeType from the available formats in pDataObj.
QVariant convertToMime(const QString &mime, LPDATAOBJECT pDataObj, QMetaType preferredType) const override
bool canConvertFromMime(const FORMATETC &formatetc, const QMimeData *mimeData) const override
Returns true if the converter can convert from the mimeData to the format specified in formatetc.
bool convertFromMime(const FORMATETC &formatetc, const QMimeData *mimeData, STGMEDIUM *pmedium) const override
Convert the mimeData to the format specified in formatetc.
static QByteArray msgConversionError(const char *func, const char *format)
static const char x_qt_windows_mime[]
static bool canGetData(int cf, IDataObject *pDataObj)
static bool setData(const QByteArray &data, STGMEDIUM *pmedium, DWORD tymed=TYMED_HGLOBAL)
static FORMATETC setCf(int cf)
static bool isCustomMimeType(const QString &mimeType)
static QByteArray getData(int cf, IDataObject *pDataObj, int lindex=-1)
static QString customMimeType(const QString &mimeType, int *lindex=nullptr)
static bool readDib(QBuffer &buffer, QImage &img)
static int getCf(const FORMATETC &formatetc)
static const char dibFormatC[]