/src/gui/kernel/qmime_mac.cpp

https://bitbucket.org/ultra_iter/qt-vtl · C++ · 1308 lines · 965 code · 152 blank · 191 comment · 204 complexity · 35d769a4f9cc4d03934717db35cf3cd0 MD5 · raw file

  1. /****************************************************************************
  2. **
  3. ** Copyright (C) 2012 Nokia Corporation and/or its subsidiary(-ies).
  4. ** All rights reserved.
  5. ** Contact: Nokia Corporation (qt-info@nokia.com)
  6. **
  7. ** This file is part of the QtGui module of the Qt Toolkit.
  8. **
  9. ** $QT_BEGIN_LICENSE:LGPL$
  10. ** GNU Lesser General Public License Usage
  11. ** This file may be used under the terms of the GNU Lesser General Public
  12. ** License version 2.1 as published by the Free Software Foundation and
  13. ** appearing in the file LICENSE.LGPL included in the packaging of this
  14. ** file. Please review the following information to ensure the GNU Lesser
  15. ** General Public License version 2.1 requirements will be met:
  16. ** http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.
  17. **
  18. ** In addition, as a special exception, Nokia gives you certain additional
  19. ** rights. These rights are described in the Nokia Qt LGPL Exception
  20. ** version 1.1, included in the file LGPL_EXCEPTION.txt in this package.
  21. **
  22. ** GNU General Public License Usage
  23. ** Alternatively, this file may be used under the terms of the GNU General
  24. ** Public License version 3.0 as published by the Free Software Foundation
  25. ** and appearing in the file LICENSE.GPL included in the packaging of this
  26. ** file. Please review the following information to ensure the GNU General
  27. ** Public License version 3.0 requirements will be met:
  28. ** http://www.gnu.org/copyleft/gpl.html.
  29. **
  30. ** Other Usage
  31. ** Alternatively, this file may be used in accordance with the terms and
  32. ** conditions contained in a signed written agreement between you and Nokia.
  33. **
  34. **
  35. **
  36. **
  37. **
  38. ** $QT_END_LICENSE$
  39. **
  40. ****************************************************************************/
  41. #include "qmime.h"
  42. //#define USE_INTERNET_CONFIG
  43. #ifndef USE_INTERNET_CONFIG
  44. # include "qfile.h"
  45. # include "qfileinfo.h"
  46. # include "qtextstream.h"
  47. # include "qdir.h"
  48. # include <unistd.h>
  49. # include <sys/types.h>
  50. # include <sys/stat.h>
  51. # include <sys/fcntl.h>
  52. #endif
  53. #include "qdebug.h"
  54. #include "qpixmap.h"
  55. #include "qimagewriter.h"
  56. #include "qimagereader.h"
  57. #include "qdatastream.h"
  58. #include "qbuffer.h"
  59. #include "qdatetime.h"
  60. #include "qapplication_p.h"
  61. #include "qtextcodec.h"
  62. #include "qregexp.h"
  63. #include "qurl.h"
  64. #include "qmap.h"
  65. #include <private/qt_mac_p.h>
  66. #ifdef Q_WS_MAC32
  67. #include <QuickTime/QuickTime.h>
  68. #include <qlibrary.h>
  69. #endif
  70. QT_BEGIN_NAMESPACE
  71. extern CGImageRef qt_mac_createCGImageFromQImage(const QImage &img, const QImage **imagePtr = 0); // qpaintengine_mac.cpp
  72. typedef QList<QMacPasteboardMime*> MimeList;
  73. Q_GLOBAL_STATIC(MimeList, globalMimeList)
  74. static void cleanup_mimes()
  75. {
  76. MimeList *mimes = globalMimeList();
  77. while (!mimes->isEmpty())
  78. delete mimes->takeFirst();
  79. }
  80. Q_GLOBAL_STATIC(QStringList, globalDraggedTypesList)
  81. /*!
  82. \fn void qRegisterDraggedTypes(const QStringList &types)
  83. \relates QMacPasteboardMime
  84. Registers the given \a types as custom pasteboard types.
  85. This function should be called to enable the Drag and Drop events
  86. for custom pasteboard types on Cocoa implementations. This is required
  87. in addition to a QMacPasteboardMime subclass implementation. By default
  88. drag and drop is enabled for all standard pasteboard types.
  89. \sa QMacPasteboardMime
  90. */
  91. Q_GUI_EXPORT void qRegisterDraggedTypes(const QStringList &types)
  92. {
  93. (*globalDraggedTypesList()) += types;
  94. }
  95. const QStringList& qEnabledDraggedTypes()
  96. {
  97. return (*globalDraggedTypesList());
  98. }
  99. /*****************************************************************************
  100. QDnD debug facilities
  101. *****************************************************************************/
  102. //#define DEBUG_MIME_MAPS
  103. ScrapFlavorType qt_mac_mime_type = 'CUTE';
  104. CFStringRef qt_mac_mime_typeUTI = CFSTR("com.pasteboard.trolltech.marker");
  105. /*!
  106. \class QMacPasteboardMime
  107. \brief The QMacPasteboardMime class converts between a MIME type and a
  108. \l{http://developer.apple.com/macosx/uniformtypeidentifiers.html}{Uniform
  109. Type Identifier (UTI)} format.
  110. \since 4.2
  111. \ingroup draganddrop
  112. Qt's drag and drop and clipboard facilities use the MIME
  113. standard. On X11, this maps trivially to the Xdnd protocol. On
  114. Mac, although some applications use MIME to describe clipboard
  115. contents, it is more common to use Apple's UTI format.
  116. QMacPasteboardMime's role is to bridge the gap between MIME and UTI;
  117. By subclasses this class, one can extend Qt's drag and drop
  118. and clipboard handling to convert to and from unsupported, or proprietary, UTI formats.
  119. A subclass of QMacPasteboardMime will automatically be registered, and active, upon instantiation.
  120. Qt has predefined support for the following UTIs:
  121. \list
  122. \i public.utf8-plain-text - converts to "text/plain"
  123. \i public.utf16-plain-text - converts to "text/plain"
  124. \i public.html - converts to "text/html"
  125. \i public.url - converts to "text/uri-list"
  126. \i public.file-url - converts to "text/uri-list"
  127. \i public.tiff - converts to "application/x-qt-image"
  128. \i public.vcard - converts to "text/plain"
  129. \i com.apple.traditional-mac-plain-text - converts to "text/plain"
  130. \i com.apple.pict - converts to "application/x-qt-image"
  131. \endlist
  132. When working with MIME data, Qt will interate through all instances of QMacPasteboardMime to
  133. find an instance that can convert to, or from, a specific MIME type. It will do this by calling
  134. canConvert() on each instance, starting with (and choosing) the last created instance first.
  135. The actual conversions will be done by using convertToMime() and convertFromMime().
  136. \note The API uses the term "flavor" in some cases. This is for backwards
  137. compatibility reasons, and should now be understood as UTIs.
  138. */
  139. /*! \enum QMacPasteboardMime::QMacPasteboardMimeType
  140. \internal
  141. */
  142. /*!
  143. Constructs a new conversion object of type \a t, adding it to the
  144. globally accessed list of available convertors.
  145. */
  146. QMacPasteboardMime::QMacPasteboardMime(char t) : type(t)
  147. {
  148. globalMimeList()->append(this);
  149. }
  150. /*!
  151. Destroys a conversion object, removing it from the global
  152. list of available convertors.
  153. */
  154. QMacPasteboardMime::~QMacPasteboardMime()
  155. {
  156. if(!QApplication::closingDown())
  157. globalMimeList()->removeAll(this);
  158. }
  159. class QMacPasteboardMimeAny : public QMacPasteboardMime {
  160. private:
  161. public:
  162. QMacPasteboardMimeAny() : QMacPasteboardMime(MIME_QT_CONVERTOR|MIME_ALL) {
  163. }
  164. ~QMacPasteboardMimeAny() {
  165. }
  166. QString convertorName();
  167. QString flavorFor(const QString &mime);
  168. QString mimeFor(QString flav);
  169. bool canConvert(const QString &mime, QString flav);
  170. QVariant convertToMime(const QString &mime, QList<QByteArray> data, QString flav);
  171. QList<QByteArray> convertFromMime(const QString &mime, QVariant data, QString flav);
  172. };
  173. QString QMacPasteboardMimeAny::convertorName()
  174. {
  175. return QLatin1String("Any-Mime");
  176. }
  177. QString QMacPasteboardMimeAny::flavorFor(const QString &mime)
  178. {
  179. // do not handle the mime type name in the drag pasteboard
  180. if(mime == QLatin1String("application/x-qt-mime-type-name"))
  181. return QString();
  182. QString ret = QLatin1String("com.trolltech.anymime.") + mime;
  183. return ret.replace(QLatin1Char('/'), QLatin1String("--"));
  184. }
  185. QString QMacPasteboardMimeAny::mimeFor(QString flav)
  186. {
  187. const QString any_prefix = QLatin1String("com.trolltech.anymime.");
  188. if(flav.size() > any_prefix.length() && flav.startsWith(any_prefix))
  189. return flav.mid(any_prefix.length()).replace(QLatin1String("--"), QLatin1String("/"));
  190. return QString();
  191. }
  192. bool QMacPasteboardMimeAny::canConvert(const QString &mime, QString flav)
  193. {
  194. return mimeFor(flav) == mime;
  195. }
  196. QVariant QMacPasteboardMimeAny::convertToMime(const QString &mime, QList<QByteArray> data, QString)
  197. {
  198. if(data.count() > 1)
  199. qWarning("QMacPasteboardMimeAny: Cannot handle multiple member data");
  200. QVariant ret;
  201. if (mime == QLatin1String("text/plain"))
  202. ret = QString::fromUtf8(data.first());
  203. else
  204. ret = data.first();
  205. return ret;
  206. }
  207. QList<QByteArray> QMacPasteboardMimeAny::convertFromMime(const QString &mime, QVariant data, QString)
  208. {
  209. QList<QByteArray> ret;
  210. if (mime == QLatin1String("text/plain"))
  211. ret.append(data.toString().toUtf8());
  212. else
  213. ret.append(data.toByteArray());
  214. return ret;
  215. }
  216. class QMacPasteboardMimeTypeName : public QMacPasteboardMime {
  217. private:
  218. public:
  219. QMacPasteboardMimeTypeName() : QMacPasteboardMime(MIME_QT_CONVERTOR|MIME_ALL) {
  220. }
  221. ~QMacPasteboardMimeTypeName() {
  222. }
  223. QString convertorName();
  224. QString flavorFor(const QString &mime);
  225. QString mimeFor(QString flav);
  226. bool canConvert(const QString &mime, QString flav);
  227. QVariant convertToMime(const QString &mime, QList<QByteArray> data, QString flav);
  228. QList<QByteArray> convertFromMime(const QString &mime, QVariant data, QString flav);
  229. };
  230. QString QMacPasteboardMimeTypeName::convertorName()
  231. {
  232. return QLatin1String("Qt-Mime-Type");
  233. }
  234. QString QMacPasteboardMimeTypeName::flavorFor(const QString &mime)
  235. {
  236. if(mime == QLatin1String("application/x-qt-mime-type-name"))
  237. return QLatin1String("com.trolltech.qt.MimeTypeName");
  238. return QString();
  239. }
  240. QString QMacPasteboardMimeTypeName::mimeFor(QString)
  241. {
  242. return QString();
  243. }
  244. bool QMacPasteboardMimeTypeName::canConvert(const QString &, QString)
  245. {
  246. return false;
  247. }
  248. QVariant QMacPasteboardMimeTypeName::convertToMime(const QString &, QList<QByteArray>, QString)
  249. {
  250. QVariant ret;
  251. return ret;
  252. }
  253. QList<QByteArray> QMacPasteboardMimeTypeName::convertFromMime(const QString &, QVariant, QString)
  254. {
  255. QList<QByteArray> ret;
  256. ret.append(QString("x-qt-mime-type-name").toUtf8());
  257. return ret;
  258. }
  259. class QMacPasteboardMimePlainText : public QMacPasteboardMime {
  260. public:
  261. QMacPasteboardMimePlainText() : QMacPasteboardMime(MIME_ALL) { }
  262. QString convertorName();
  263. QString flavorFor(const QString &mime);
  264. QString mimeFor(QString flav);
  265. bool canConvert(const QString &mime, QString flav);
  266. QVariant convertToMime(const QString &mime, QList<QByteArray> data, QString flav);
  267. QList<QByteArray> convertFromMime(const QString &mime, QVariant data, QString flav);
  268. };
  269. QString QMacPasteboardMimePlainText::convertorName()
  270. {
  271. return QLatin1String("PlainText");
  272. }
  273. QString QMacPasteboardMimePlainText::flavorFor(const QString &mime)
  274. {
  275. if (mime == QLatin1String("text/plain"))
  276. return QLatin1String("com.apple.traditional-mac-plain-text");
  277. return QString();
  278. }
  279. QString QMacPasteboardMimePlainText::mimeFor(QString flav)
  280. {
  281. if (flav == QLatin1String("com.apple.traditional-mac-plain-text"))
  282. return QLatin1String("text/plain");
  283. return QString();
  284. }
  285. bool QMacPasteboardMimePlainText::canConvert(const QString &mime, QString flav)
  286. {
  287. return flavorFor(mime) == flav;
  288. }
  289. QVariant QMacPasteboardMimePlainText::convertToMime(const QString &mimetype, QList<QByteArray> data, QString flavor)
  290. {
  291. if(data.count() > 1)
  292. qWarning("QMacPasteboardMimePlainText: Cannot handle multiple member data");
  293. const QByteArray &firstData = data.first();
  294. QVariant ret;
  295. if(flavor == QCFString(QLatin1String("com.apple.traditional-mac-plain-text"))) {
  296. QCFString str(CFStringCreateWithBytes(kCFAllocatorDefault,
  297. reinterpret_cast<const UInt8 *>(firstData.constData()),
  298. firstData.size(), CFStringGetSystemEncoding(), false));
  299. ret = QString(str);
  300. } else {
  301. qWarning("QMime::convertToMime: unhandled mimetype: %s", qPrintable(mimetype));
  302. }
  303. return ret;
  304. }
  305. QList<QByteArray> QMacPasteboardMimePlainText::convertFromMime(const QString &, QVariant data, QString flavor)
  306. {
  307. QList<QByteArray> ret;
  308. QString string = data.toString();
  309. if(flavor == QCFString(QLatin1String("com.apple.traditional-mac-plain-text")))
  310. ret.append(string.toLatin1());
  311. return ret;
  312. }
  313. class QMacPasteboardMimeUnicodeText : public QMacPasteboardMime {
  314. public:
  315. QMacPasteboardMimeUnicodeText() : QMacPasteboardMime(MIME_ALL) { }
  316. QString convertorName();
  317. QString flavorFor(const QString &mime);
  318. QString mimeFor(QString flav);
  319. bool canConvert(const QString &mime, QString flav);
  320. QVariant convertToMime(const QString &mime, QList<QByteArray> data, QString flav);
  321. QList<QByteArray> convertFromMime(const QString &mime, QVariant data, QString flav);
  322. };
  323. QString QMacPasteboardMimeUnicodeText::convertorName()
  324. {
  325. return QLatin1String("UnicodeText");
  326. }
  327. QString QMacPasteboardMimeUnicodeText::flavorFor(const QString &mime)
  328. {
  329. if (mime == QLatin1String("text/plain"))
  330. return QLatin1String("public.utf16-plain-text");
  331. int i = mime.indexOf(QLatin1String("charset="));
  332. if (i >= 0) {
  333. QString cs(mime.mid(i+8).toLower());
  334. i = cs.indexOf(QLatin1Char(';'));
  335. if (i>=0)
  336. cs = cs.left(i);
  337. if (cs == QLatin1String("system"))
  338. return QLatin1String("public.utf8-plain-text");
  339. else if (cs == QLatin1String("iso-10646-ucs-2")
  340. || cs == QLatin1String("utf16"))
  341. return QLatin1String("public.utf16-plain-text");
  342. }
  343. return QString();
  344. }
  345. QString QMacPasteboardMimeUnicodeText::mimeFor(QString flav)
  346. {
  347. if (flav == QLatin1String("public.utf16-plain-text") || flav == QLatin1String("public.utf8-plain-text"))
  348. return QLatin1String("text/plain");
  349. return QString();
  350. }
  351. bool QMacPasteboardMimeUnicodeText::canConvert(const QString &mime, QString flav)
  352. {
  353. return flavorFor(mime) == flav;
  354. }
  355. QVariant QMacPasteboardMimeUnicodeText::convertToMime(const QString &mimetype, QList<QByteArray> data, QString flavor)
  356. {
  357. if(data.count() > 1)
  358. qWarning("QMacPasteboardMimeUnicodeText: Cannot handle multiple member data");
  359. const QByteArray &firstData = data.first();
  360. // I can only handle two types (system and unicode) so deal with them that way
  361. QVariant ret;
  362. if(flavor == QLatin1String("public.utf8-plain-text")) {
  363. QCFString str(CFStringCreateWithBytes(kCFAllocatorDefault,
  364. reinterpret_cast<const UInt8 *>(firstData.constData()),
  365. firstData.size(), CFStringGetSystemEncoding(), false));
  366. ret = QString(str);
  367. } else if (flavor == QLatin1String("public.utf16-plain-text")) {
  368. ret = QString(reinterpret_cast<const QChar *>(firstData.constData()),
  369. firstData.size() / sizeof(QChar));
  370. } else {
  371. qWarning("QMime::convertToMime: unhandled mimetype: %s", qPrintable(mimetype));
  372. }
  373. return ret;
  374. }
  375. QList<QByteArray> QMacPasteboardMimeUnicodeText::convertFromMime(const QString &, QVariant data, QString flavor)
  376. {
  377. QList<QByteArray> ret;
  378. QString string = data.toString();
  379. if(flavor == QLatin1String("public.utf8-plain-text"))
  380. ret.append(string.toUtf8());
  381. else if (flavor == QLatin1String("public.utf16-plain-text"))
  382. ret.append(QByteArray((char*)string.utf16(), string.length()*2));
  383. return ret;
  384. }
  385. class QMacPasteboardMimeHTMLText : public QMacPasteboardMime {
  386. public:
  387. QMacPasteboardMimeHTMLText() : QMacPasteboardMime(MIME_ALL) { }
  388. QString convertorName();
  389. QString flavorFor(const QString &mime);
  390. QString mimeFor(QString flav);
  391. bool canConvert(const QString &mime, QString flav);
  392. QVariant convertToMime(const QString &mime, QList<QByteArray> data, QString flav);
  393. QList<QByteArray> convertFromMime(const QString &mime, QVariant data, QString flav);
  394. };
  395. QString QMacPasteboardMimeHTMLText::convertorName()
  396. {
  397. return QLatin1String("HTML");
  398. }
  399. QString QMacPasteboardMimeHTMLText::flavorFor(const QString &mime)
  400. {
  401. if (mime == QLatin1String("text/html"))
  402. return QLatin1String("public.html");
  403. return QString();
  404. }
  405. QString QMacPasteboardMimeHTMLText::mimeFor(QString flav)
  406. {
  407. if (flav == QLatin1String("public.html"))
  408. return QLatin1String("text/html");
  409. return QString();
  410. }
  411. bool QMacPasteboardMimeHTMLText::canConvert(const QString &mime, QString flav)
  412. {
  413. return flavorFor(mime) == flav;
  414. }
  415. QVariant QMacPasteboardMimeHTMLText::convertToMime(const QString &mimeType, QList<QByteArray> data, QString flavor)
  416. {
  417. if (!canConvert(mimeType, flavor))
  418. return QVariant();
  419. if (data.count() > 1)
  420. qWarning("QMacPasteboardMimeHTMLText: Cannot handle multiple member data");
  421. return data.first();
  422. }
  423. QList<QByteArray> QMacPasteboardMimeHTMLText::convertFromMime(const QString &mime, QVariant data, QString flavor)
  424. {
  425. QList<QByteArray> ret;
  426. if (!canConvert(mime, flavor))
  427. return ret;
  428. ret.append(data.toByteArray());
  429. return ret;
  430. }
  431. #ifdef Q_WS_MAC32
  432. // This can be removed once 10.6 is the minimum (or we have to require 64-bit) whichever comes first.
  433. typedef ComponentResult (*PtrGraphicsImportSetDataHandle)(GraphicsImportComponent, Handle);
  434. typedef ComponentResult (*PtrGraphicsImportCreateCGImage)(GraphicsImportComponent, CGImageRef*, UInt32);
  435. typedef ComponentResult (*PtrGraphicsExportSetInputCGImage)(GraphicsExportComponent, CGImageRef);
  436. typedef ComponentResult (*PtrGraphicsExportSetOutputHandle)(GraphicsExportComponent, Handle);
  437. typedef ComponentResult (*PtrGraphicsExportDoExport)(GraphicsExportComponent, unsigned long *);
  438. static PtrGraphicsImportSetDataHandle ptrGraphicsImportSetDataHandle = 0;
  439. static PtrGraphicsImportCreateCGImage ptrGraphicsImportCreateCGImage = 0;
  440. static PtrGraphicsExportSetInputCGImage ptrGraphicsExportSetInputCGImage = 0;
  441. static PtrGraphicsExportSetOutputHandle ptrGraphicsExportSetOutputHandle = 0;
  442. static PtrGraphicsExportDoExport ptrGraphicsExportDoExport = 0;
  443. static bool resolveMimeQuickTimeSymbols()
  444. {
  445. static bool triedResolve = false;
  446. if (!triedResolve) {
  447. QLibrary library(QLatin1String("/System/Library/Frameworks/QuickTime.framework/QuickTime"));
  448. ptrGraphicsImportSetDataHandle = reinterpret_cast<PtrGraphicsImportSetDataHandle>(library.resolve("GraphicsImportSetDataHandle"));
  449. ptrGraphicsImportCreateCGImage = reinterpret_cast<PtrGraphicsImportCreateCGImage>(library.resolve("GraphicsImportCreateCGImage"));
  450. ptrGraphicsExportSetInputCGImage = reinterpret_cast<PtrGraphicsExportSetInputCGImage>(library.resolve("GraphicsExportSetInputCGImage"));
  451. ptrGraphicsExportSetOutputHandle = reinterpret_cast<PtrGraphicsExportSetOutputHandle>(library.resolve("GraphicsExportSetOutputHandle"));
  452. ptrGraphicsExportDoExport = reinterpret_cast<PtrGraphicsExportDoExport>(library.resolve("GraphicsExportDoExport"));
  453. triedResolve = true;
  454. }
  455. return ptrGraphicsImportSetDataHandle != 0
  456. && ptrGraphicsImportCreateCGImage != 0 && ptrGraphicsExportSetInputCGImage != 0
  457. && ptrGraphicsExportSetOutputHandle != 0 && ptrGraphicsExportDoExport != 0;
  458. }
  459. class QMacPasteboardMimePict : public QMacPasteboardMime {
  460. public:
  461. QMacPasteboardMimePict() : QMacPasteboardMime(MIME_ALL) { }
  462. QString convertorName();
  463. QString flavorFor(const QString &mime);
  464. QString mimeFor(QString flav);
  465. bool canConvert(const QString &mime, QString flav);
  466. QVariant convertToMime(const QString &mime, QList<QByteArray> data, QString flav);
  467. QList<QByteArray> convertFromMime(const QString &mime, QVariant data, QString flav);
  468. };
  469. QString QMacPasteboardMimePict::convertorName()
  470. {
  471. return QLatin1String("Pict");
  472. }
  473. QString QMacPasteboardMimePict::flavorFor(const QString &mime)
  474. {
  475. if(mime.startsWith(QLatin1String("application/x-qt-image")))
  476. return QLatin1String("com.apple.pict");
  477. return QString();
  478. }
  479. QString QMacPasteboardMimePict::mimeFor(QString flav)
  480. {
  481. if(flav == QLatin1String("com.apple.pict"))
  482. return QLatin1String("application/x-qt-image");
  483. return QString();
  484. }
  485. bool QMacPasteboardMimePict::canConvert(const QString &mime, QString flav)
  486. {
  487. return flav == QLatin1String("com.apple.pict")
  488. && mime == QLatin1String("application/x-qt-image");
  489. }
  490. QVariant QMacPasteboardMimePict::convertToMime(const QString &mime, QList<QByteArray> data, QString flav)
  491. {
  492. if(data.count() > 1)
  493. qWarning("QMacPasteboardMimePict: Cannot handle multiple member data");
  494. QVariant ret;
  495. if (!resolveMimeQuickTimeSymbols())
  496. return ret;
  497. if(!canConvert(mime, flav))
  498. return ret;
  499. const QByteArray &a = data.first();
  500. // This function expects the 512 header (just to skip it, so create the extra space for it).
  501. Handle pic = NewHandle(a.size() + 512);
  502. memcpy(*pic + 512, a.constData(), a.size());
  503. GraphicsImportComponent graphicsImporter;
  504. ComponentResult result = OpenADefaultComponent(GraphicsImporterComponentType,
  505. kQTFileTypePicture, &graphicsImporter);
  506. QCFType<CGImageRef> cgImage;
  507. if (!result)
  508. result = ptrGraphicsImportSetDataHandle(graphicsImporter, pic);
  509. if (!result)
  510. result = ptrGraphicsImportCreateCGImage(graphicsImporter, &cgImage,
  511. kGraphicsImportCreateCGImageUsingCurrentSettings);
  512. if (!result)
  513. ret = QVariant(QPixmap::fromMacCGImageRef(cgImage).toImage());
  514. CloseComponent(graphicsImporter);
  515. DisposeHandle(pic);
  516. return ret;
  517. }
  518. QList<QByteArray> QMacPasteboardMimePict::convertFromMime(const QString &mime, QVariant variant,
  519. QString flav)
  520. {
  521. QList<QByteArray> ret;
  522. if (!resolveMimeQuickTimeSymbols())
  523. return ret;
  524. if (!canConvert(mime, flav))
  525. return ret;
  526. QCFType<CGImageRef> cgimage = qt_mac_createCGImageFromQImage(qvariant_cast<QImage>(variant));
  527. Handle pic = NewHandle(0);
  528. GraphicsExportComponent graphicsExporter;
  529. ComponentResult result = OpenADefaultComponent(GraphicsExporterComponentType,
  530. kQTFileTypePicture, &graphicsExporter);
  531. if (!result) {
  532. unsigned long sizeWritten;
  533. result = ptrGraphicsExportSetInputCGImage(graphicsExporter, cgimage);
  534. if (!result)
  535. result = ptrGraphicsExportSetOutputHandle(graphicsExporter, pic);
  536. if (!result)
  537. result = ptrGraphicsExportDoExport(graphicsExporter, &sizeWritten);
  538. CloseComponent(graphicsExporter);
  539. }
  540. int size = GetHandleSize((Handle)pic);
  541. // Skip the Picture File header (512 bytes) and feed the raw data
  542. QByteArray ar(reinterpret_cast<char *>(*pic + 512), size - 512);
  543. ret.append(ar);
  544. DisposeHandle(pic);
  545. return ret;
  546. }
  547. #endif //Q_WS_MAC32
  548. class QMacPasteboardMimeTiff : public QMacPasteboardMime {
  549. public:
  550. QMacPasteboardMimeTiff() : QMacPasteboardMime(MIME_ALL) { }
  551. QString convertorName();
  552. QString flavorFor(const QString &mime);
  553. QString mimeFor(QString flav);
  554. bool canConvert(const QString &mime, QString flav);
  555. QVariant convertToMime(const QString &mime, QList<QByteArray> data, QString flav);
  556. QList<QByteArray> convertFromMime(const QString &mime, QVariant data, QString flav);
  557. };
  558. QString QMacPasteboardMimeTiff::convertorName()
  559. {
  560. return QLatin1String("Tiff");
  561. }
  562. QString QMacPasteboardMimeTiff::flavorFor(const QString &mime)
  563. {
  564. if(mime.startsWith(QLatin1String("application/x-qt-image")))
  565. return QLatin1String("public.tiff");
  566. return QString();
  567. }
  568. QString QMacPasteboardMimeTiff::mimeFor(QString flav)
  569. {
  570. if(flav == QLatin1String("public.tiff"))
  571. return QLatin1String("application/x-qt-image");
  572. return QString();
  573. }
  574. bool QMacPasteboardMimeTiff::canConvert(const QString &mime, QString flav)
  575. {
  576. return flav == QLatin1String("public.tiff") && mime == QLatin1String("application/x-qt-image");
  577. }
  578. QVariant QMacPasteboardMimeTiff::convertToMime(const QString &mime, QList<QByteArray> data, QString flav)
  579. {
  580. if(data.count() > 1)
  581. qWarning("QMacPasteboardMimeTiff: Cannot handle multiple member data");
  582. QVariant ret;
  583. if (!canConvert(mime, flav))
  584. return ret;
  585. const QByteArray &a = data.first();
  586. QCFType<CGImageRef> image;
  587. QCFType<CFDataRef> tiffData = CFDataCreateWithBytesNoCopy(0,
  588. reinterpret_cast<const UInt8 *>(a.constData()),
  589. a.size(), kCFAllocatorNull);
  590. QCFType<CGImageSourceRef> imageSource = CGImageSourceCreateWithData(tiffData, 0);
  591. image = CGImageSourceCreateImageAtIndex(imageSource, 0, 0);
  592. if (image != 0)
  593. ret = QVariant(QPixmap::fromMacCGImageRef(image).toImage());
  594. return ret;
  595. }
  596. QList<QByteArray> QMacPasteboardMimeTiff::convertFromMime(const QString &mime, QVariant variant, QString flav)
  597. {
  598. QList<QByteArray> ret;
  599. if (!canConvert(mime, flav))
  600. return ret;
  601. QImage img = qvariant_cast<QImage>(variant);
  602. QCFType<CGImageRef> cgimage = qt_mac_createCGImageFromQImage(img);
  603. #if (MAC_OS_X_VERSION_MAX_ALLOWED >= MAC_OS_X_VERSION_10_4)
  604. if (QSysInfo::MacintoshVersion >= QSysInfo::MV_10_4) {
  605. QCFType<CFMutableDataRef> data = CFDataCreateMutable(0, 0);
  606. QCFType<CGImageDestinationRef> imageDestination = CGImageDestinationCreateWithData(data, kUTTypeTIFF, 1, 0);
  607. if (imageDestination != 0) {
  608. CFTypeRef keys[2];
  609. QCFType<CFTypeRef> values[2];
  610. QCFType<CFDictionaryRef> options;
  611. keys[0] = kCGImagePropertyPixelWidth;
  612. keys[1] = kCGImagePropertyPixelHeight;
  613. int width = img.width();
  614. int height = img.height();
  615. values[0] = CFNumberCreate(0, kCFNumberIntType, &width);
  616. values[1] = CFNumberCreate(0, kCFNumberIntType, &height);
  617. options = CFDictionaryCreate(0, reinterpret_cast<const void **>(keys),
  618. reinterpret_cast<const void **>(values), 2,
  619. &kCFTypeDictionaryKeyCallBacks,
  620. &kCFTypeDictionaryValueCallBacks);
  621. CGImageDestinationAddImage(imageDestination, cgimage, options);
  622. CGImageDestinationFinalize(imageDestination);
  623. }
  624. QByteArray ar(CFDataGetLength(data), 0);
  625. CFDataGetBytes(data,
  626. CFRangeMake(0, ar.size()),
  627. reinterpret_cast<UInt8 *>(ar.data()));
  628. ret.append(ar);
  629. } else
  630. #endif
  631. {
  632. #ifdef Q_WS_MAC32
  633. Handle tiff = NewHandle(0);
  634. if (resolveMimeQuickTimeSymbols()) {
  635. GraphicsExportComponent graphicsExporter;
  636. ComponentResult result = OpenADefaultComponent(GraphicsExporterComponentType,
  637. kQTFileTypeTIFF, &graphicsExporter);
  638. if (!result) {
  639. unsigned long sizeWritten;
  640. result = ptrGraphicsExportSetInputCGImage(graphicsExporter, cgimage);
  641. if (!result)
  642. result = ptrGraphicsExportSetOutputHandle(graphicsExporter, tiff);
  643. if (!result)
  644. result = ptrGraphicsExportDoExport(graphicsExporter, &sizeWritten);
  645. CloseComponent(graphicsExporter);
  646. }
  647. }
  648. int size = GetHandleSize((Handle)tiff);
  649. QByteArray ar(reinterpret_cast<char *>(*tiff), size);
  650. ret.append(ar);
  651. DisposeHandle(tiff);
  652. #endif
  653. }
  654. return ret;
  655. }
  656. class QMacPasteboardMimeFileUri : public QMacPasteboardMime {
  657. public:
  658. QMacPasteboardMimeFileUri() : QMacPasteboardMime(MIME_ALL) { }
  659. QString convertorName();
  660. QString flavorFor(const QString &mime);
  661. QString mimeFor(QString flav);
  662. bool canConvert(const QString &mime, QString flav);
  663. QVariant convertToMime(const QString &mime, QList<QByteArray> data, QString flav);
  664. QList<QByteArray> convertFromMime(const QString &mime, QVariant data, QString flav);
  665. };
  666. QString QMacPasteboardMimeFileUri::convertorName()
  667. {
  668. return QLatin1String("FileURL");
  669. }
  670. QString QMacPasteboardMimeFileUri::flavorFor(const QString &mime)
  671. {
  672. if (mime == QLatin1String("text/uri-list"))
  673. return QCFString(UTTypeCreatePreferredIdentifierForTag(kUTTagClassOSType, CFSTR("furl"), 0));
  674. return QString();
  675. }
  676. QString QMacPasteboardMimeFileUri::mimeFor(QString flav)
  677. {
  678. if (flav == QCFString(UTTypeCreatePreferredIdentifierForTag(kUTTagClassOSType, CFSTR("furl"), 0)))
  679. return QLatin1String("text/uri-list");
  680. return QString();
  681. }
  682. bool QMacPasteboardMimeFileUri::canConvert(const QString &mime, QString flav)
  683. {
  684. return mime == QLatin1String("text/uri-list")
  685. && flav == QCFString(UTTypeCreatePreferredIdentifierForTag(kUTTagClassOSType, CFSTR("furl"), 0));
  686. }
  687. QVariant QMacPasteboardMimeFileUri::convertToMime(const QString &mime, QList<QByteArray> data, QString flav)
  688. {
  689. if(!canConvert(mime, flav))
  690. return QVariant();
  691. QList<QVariant> ret;
  692. for(int i = 0; i < data.size(); ++i) {
  693. QUrl url = QUrl::fromEncoded(data.at(i));
  694. if (url.host().toLower() == QLatin1String("localhost"))
  695. url.setHost(QString());
  696. url.setPath(url.path().normalized(QString::NormalizationForm_C));
  697. ret.append(url);
  698. }
  699. return QVariant(ret);
  700. }
  701. QList<QByteArray> QMacPasteboardMimeFileUri::convertFromMime(const QString &mime, QVariant data, QString flav)
  702. {
  703. QList<QByteArray> ret;
  704. if (!canConvert(mime, flav))
  705. return ret;
  706. QList<QVariant> urls = data.toList();
  707. for(int i = 0; i < urls.size(); ++i) {
  708. QUrl url = urls.at(i).toUrl();
  709. if (url.scheme().isEmpty())
  710. url.setScheme(QLatin1String("file"));
  711. if (url.scheme().toLower() == QLatin1String("file")) {
  712. if (url.host().isEmpty())
  713. url.setHost(QLatin1String("localhost"));
  714. url.setPath(url.path().normalized(QString::NormalizationForm_D));
  715. }
  716. ret.append(url.toEncoded());
  717. }
  718. return ret;
  719. }
  720. class QMacPasteboardMimeUrl : public QMacPasteboardMime {
  721. public:
  722. QMacPasteboardMimeUrl() : QMacPasteboardMime(MIME_ALL) { }
  723. QString convertorName();
  724. QString flavorFor(const QString &mime);
  725. QString mimeFor(QString flav);
  726. bool canConvert(const QString &mime, QString flav);
  727. QVariant convertToMime(const QString &mime, QList<QByteArray> data, QString flav);
  728. QList<QByteArray> convertFromMime(const QString &mime, QVariant data, QString flav);
  729. };
  730. QString QMacPasteboardMimeUrl::convertorName()
  731. {
  732. return QLatin1String("URL");
  733. }
  734. QString QMacPasteboardMimeUrl::flavorFor(const QString &mime)
  735. {
  736. if(mime.startsWith(QLatin1String("text/uri-list")))
  737. return QLatin1String("public.url");
  738. return QString();
  739. }
  740. QString QMacPasteboardMimeUrl::mimeFor(QString flav)
  741. {
  742. if(flav == QLatin1String("public.url"))
  743. return QLatin1String("text/uri-list");
  744. return QString();
  745. }
  746. bool QMacPasteboardMimeUrl::canConvert(const QString &mime, QString flav)
  747. {
  748. return flav == QLatin1String("public.url")
  749. && mime == QLatin1String("text/uri-list");
  750. }
  751. QVariant QMacPasteboardMimeUrl::convertToMime(const QString &mime, QList<QByteArray> data, QString flav)
  752. {
  753. if(!canConvert(mime, flav))
  754. return QVariant();
  755. QList<QVariant> ret;
  756. for (int i=0; i<data.size(); ++i) {
  757. QUrl url = QUrl::fromEncoded(data.at(i));
  758. if (url.host().toLower() == QLatin1String("localhost"))
  759. url.setHost(QString());
  760. url.setPath(url.path().normalized(QString::NormalizationForm_C));
  761. ret.append(url);
  762. }
  763. return QVariant(ret);
  764. }
  765. QList<QByteArray> QMacPasteboardMimeUrl::convertFromMime(const QString &mime, QVariant data, QString flav)
  766. {
  767. QList<QByteArray> ret;
  768. if (!canConvert(mime, flav))
  769. return ret;
  770. QList<QVariant> urls = data.toList();
  771. for(int i=0; i<urls.size(); ++i) {
  772. QUrl url = urls.at(i).toUrl();
  773. if (url.scheme().isEmpty())
  774. url.setScheme(QLatin1String("file"));
  775. if (url.scheme().toLower() == QLatin1String("file")) {
  776. if (url.host().isEmpty())
  777. url.setHost(QLatin1String("localhost"));
  778. url.setPath(url.path().normalized(QString::NormalizationForm_D));
  779. }
  780. ret.append(url.toEncoded());
  781. }
  782. return ret;
  783. }
  784. class QMacPasteboardMimeVCard : public QMacPasteboardMime
  785. {
  786. public:
  787. QMacPasteboardMimeVCard() : QMacPasteboardMime(MIME_ALL){ }
  788. QString convertorName();
  789. QString flavorFor(const QString &mime);
  790. QString mimeFor(QString flav);
  791. bool canConvert(const QString &mime, QString flav);
  792. QVariant convertToMime(const QString &mime, QList<QByteArray> data, QString flav);
  793. QList<QByteArray> convertFromMime(const QString &mime, QVariant data, QString flav);
  794. };
  795. QString QMacPasteboardMimeVCard::convertorName()
  796. {
  797. return QString("VCard");
  798. }
  799. bool QMacPasteboardMimeVCard::canConvert(const QString &mime, QString flav)
  800. {
  801. return mimeFor(flav) == mime;
  802. }
  803. QString QMacPasteboardMimeVCard::flavorFor(const QString &mime)
  804. {
  805. if(mime.startsWith(QLatin1String("text/plain")))
  806. return QLatin1String("public.vcard");
  807. return QString();
  808. }
  809. QString QMacPasteboardMimeVCard::mimeFor(QString flav)
  810. {
  811. if (flav == QLatin1String("public.vcard"))
  812. return QLatin1String("text/plain");
  813. return QString();
  814. }
  815. QVariant QMacPasteboardMimeVCard::convertToMime(const QString &mime, QList<QByteArray> data, QString)
  816. {
  817. QByteArray cards;
  818. if (mime == QLatin1String("text/plain")) {
  819. for (int i=0; i<data.size(); ++i)
  820. cards += data[i];
  821. }
  822. return QVariant(cards);
  823. }
  824. QList<QByteArray> QMacPasteboardMimeVCard::convertFromMime(const QString &mime, QVariant data, QString)
  825. {
  826. QList<QByteArray> ret;
  827. if (mime == QLatin1String("text/plain"))
  828. ret.append(data.toString().toUtf8());
  829. return ret;
  830. }
  831. #ifdef QT3_SUPPORT
  832. class QMacPasteboardMimeQt3Any : public QMacPasteboardMime {
  833. private:
  834. int current_max;
  835. QFile library_file;
  836. QDateTime mime_registry_loaded;
  837. QMap<QString, int> mime_registry;
  838. int registerMimeType(const QString &mime);
  839. bool loadMimeRegistry();
  840. public:
  841. QMacPasteboardMimeQt3Any() : QMacPasteboardMime(MIME_QT3_CONVERTOR) {
  842. current_max = 'QT00';
  843. }
  844. ~QMacPasteboardMimeQt3Any() {
  845. }
  846. QString convertorName();
  847. QString flavorFor(const QString &mime);
  848. QString mimeFor(QString flav);
  849. bool canConvert(const QString &mime, QString flav);
  850. QVariant convertToMime(const QString &mime, QList<QByteArray> data, QString flav);
  851. QList<QByteArray> convertFromMime(const QString &mime, QVariant data, QString flav);
  852. };
  853. static bool qt_mac_openMimeRegistry(bool global, QIODevice::OpenMode mode, QFile &file)
  854. {
  855. QString dir = QLatin1String("/Library/Qt");
  856. if(!global)
  857. dir.prepend(QDir::homePath());
  858. file.setFileName(dir + QLatin1String("/.mime_types"));
  859. if(mode != QIODevice::ReadOnly) {
  860. if(!QFile::exists(dir)) {
  861. // Do it with a system call as I don't see much worth in
  862. // doing it with QDir since we have to chmod anyway.
  863. bool success = ::mkdir(dir.toLocal8Bit().constData(), S_IRUSR | S_IWUSR | S_IXUSR) == 0;
  864. if (success)
  865. success = ::chmod(dir.toLocal8Bit().constData(), S_IRUSR | S_IWUSR | S_IXUSR
  866. | S_IRGRP | S_IWGRP | S_IXGRP | S_IROTH | S_IWOTH | S_IXOTH) == 0;
  867. if (!success)
  868. return false;
  869. }
  870. if (!file.exists()) {
  871. // Create the file and chmod it so that everyone can write to it.
  872. int fd = ::open(file.fileName().toLocal8Bit().constData(), O_WRONLY | O_CREAT, S_IRUSR | S_IWUSR);
  873. bool success = fd != -1;
  874. if (success)
  875. success = ::fchmod(fd, S_IRUSR | S_IWUSR | S_IRGRP | S_IWGRP | S_IROTH | S_IWOTH) == 0;
  876. if (fd != -1)
  877. ::close(fd);
  878. if(!success)
  879. return false;
  880. }
  881. }
  882. return file.open(mode);
  883. }
  884. static void qt_mac_loadMimeRegistry(QFile &file, QMap<QString, int> &registry, int &max)
  885. {
  886. file.reset();
  887. QTextStream stream(&file);
  888. while(!stream.atEnd()) {
  889. QString mime = stream.readLine();
  890. int mactype = stream.readLine().toInt();
  891. if(mactype > max)
  892. max = mactype;
  893. registry.insert(mime, mactype);
  894. }
  895. }
  896. bool QMacPasteboardMimeQt3Any::loadMimeRegistry()
  897. {
  898. if(!library_file.isOpen()) {
  899. if(!qt_mac_openMimeRegistry(true, QIODevice::ReadWrite, library_file)) {
  900. QFile global;
  901. if(qt_mac_openMimeRegistry(true, QIODevice::ReadOnly, global)) {
  902. qt_mac_loadMimeRegistry(global, mime_registry, current_max);
  903. global.close();
  904. }
  905. if(!qt_mac_openMimeRegistry(false, QIODevice::ReadWrite, library_file)) {
  906. qWarning("QMacPasteboardMimeAnyQt3Mime: Failure to open mime resources %s -- %s", library_file.fileName().toLatin1().constData(),
  907. library_file.errorString().toLatin1().constData());
  908. return false;
  909. }
  910. }
  911. }
  912. QFileInfo fi(library_file);
  913. if(!mime_registry_loaded.isNull() && mime_registry_loaded == fi.lastModified())
  914. return true;
  915. mime_registry_loaded = fi.lastModified();
  916. qt_mac_loadMimeRegistry(library_file, mime_registry, current_max);
  917. return true;
  918. }
  919. int QMacPasteboardMimeQt3Any::registerMimeType(const QString &mime)
  920. {
  921. if(!mime_registry.contains(mime)) {
  922. if(!loadMimeRegistry()) {
  923. qWarning("QMacPasteboardMimeAnyQt3Mime: Internal error");
  924. return 0;
  925. }
  926. if(!mime_registry.contains(mime)) {
  927. if(!library_file.isOpen()) {
  928. if(!library_file.open(QIODevice::WriteOnly)) {
  929. qWarning("QMacPasteboardMimeAnyQt3Mime: Failure to open %s -- %s", library_file.fileName().toLatin1().constData(),
  930. library_file.errorString().toLatin1().constData());
  931. return false;
  932. }
  933. }
  934. int ret = ++current_max;
  935. mime_registry_loaded = QFileInfo(library_file).lastModified();
  936. QTextStream stream(&library_file);
  937. stream << mime << endl;
  938. stream << ret << endl;
  939. mime_registry.insert(mime, ret);
  940. library_file.flush(); //flush and set mtime
  941. return ret;
  942. }
  943. }
  944. return mime_registry[mime];
  945. }
  946. QString QMacPasteboardMimeQt3Any::convertorName()
  947. {
  948. return QLatin1String("Qt3-Any-Mime");
  949. }
  950. QString QMacPasteboardMimeQt3Any::flavorFor(const QString &mime)
  951. {
  952. const int os_flav = registerMimeType(mime);
  953. QCFType<CFArrayRef> ids = UTTypeCreateAllIdentifiersForTag(0, kUTTagClassOSType,
  954. QCFString(UTCreateStringForOSType(os_flav)));
  955. if(ids) {
  956. const int type_count = CFArrayGetCount(ids);
  957. if(type_count) {
  958. if(type_count > 1)
  959. qDebug("Can't happen!");
  960. return QCFString::toQString((CFStringRef)CFArrayGetValueAtIndex(ids, 0));
  961. }
  962. }
  963. return QString();
  964. }
  965. QString QMacPasteboardMimeQt3Any::mimeFor(QString flav)
  966. {
  967. loadMimeRegistry();
  968. const int os_flav = UTGetOSTypeFromString(UTTypeCopyPreferredTagWithClass(QCFString(flav), kUTTagClassOSType));
  969. for(QMap<QString, int>::const_iterator it = mime_registry.constBegin();
  970. it != mime_registry.constEnd(); ++it) {
  971. if(it.value() == os_flav)
  972. return QString::fromLatin1(it.key().toLatin1());
  973. }
  974. return QString();
  975. }
  976. bool QMacPasteboardMimeQt3Any::canConvert(const QString &mime, QString flav)
  977. {
  978. loadMimeRegistry();
  979. const int os_flav = UTGetOSTypeFromString(UTTypeCopyPreferredTagWithClass(QCFString(flav), kUTTagClassOSType));
  980. if(mime_registry.contains(mime) && mime_registry[mime] == os_flav)
  981. return true;
  982. return false;
  983. }
  984. QVariant QMacPasteboardMimeQt3Any::convertToMime(const QString &, QList<QByteArray>, QString)
  985. {
  986. qWarning("QMacPasteboardMimeAnyQt3Mime: Cannot write anything!");
  987. return QVariant();
  988. }
  989. QList<QByteArray> QMacPasteboardMimeQt3Any::convertFromMime(const QString &mime, QVariant data, QString)
  990. {
  991. QList<QByteArray> ret;
  992. if (mime == QLatin1String("text/plain")) {
  993. ret.append(data.toString().toUtf8());
  994. } else {
  995. ret.append(data.toByteArray());
  996. }
  997. return ret;
  998. }
  999. #endif
  1000. /*!
  1001. \internal
  1002. This is an internal function.
  1003. */
  1004. void QMacPasteboardMime::initialize()
  1005. {
  1006. if(globalMimeList()->isEmpty()) {
  1007. qAddPostRoutine(cleanup_mimes);
  1008. //standard types that we wrap
  1009. new QMacPasteboardMimeTiff;
  1010. #ifdef Q_WS_MAC32
  1011. // 10.6 does automatic synthesis to and from PICT to standard image types (like TIFF),
  1012. // so don't bother doing it ourselves, especially since it's not available in 64-bit.
  1013. if (QSysInfo::MacintoshVersion < QSysInfo::MV_10_6)
  1014. new QMacPasteboardMimePict;
  1015. #endif
  1016. new QMacPasteboardMimeUnicodeText;
  1017. new QMacPasteboardMimePlainText;
  1018. new QMacPasteboardMimeHTMLText;
  1019. new QMacPasteboardMimeFileUri;
  1020. new QMacPasteboardMimeUrl;
  1021. new QMacPasteboardMimeTypeName;
  1022. new QMacPasteboardMimeVCard;
  1023. //make sure our "non-standard" types are always last! --Sam
  1024. new QMacPasteboardMimeAny;
  1025. #ifdef QT3_SUPPORT
  1026. new QMacPasteboardMimeQt3Any;
  1027. #endif
  1028. }
  1029. }
  1030. /*!
  1031. Returns the most-recently created QMacPasteboardMime of type \a t that can convert
  1032. between the \a mime and \a flav formats. Returns 0 if no such convertor
  1033. exists.
  1034. */
  1035. QMacPasteboardMime*
  1036. QMacPasteboardMime::convertor(uchar t, const QString &mime, QString flav)
  1037. {
  1038. MimeList *mimes = globalMimeList();
  1039. for(MimeList::const_iterator it = mimes->constBegin(); it != mimes->constEnd(); ++it) {
  1040. #ifdef DEBUG_MIME_MAPS
  1041. qDebug("QMacPasteboardMime::convertor: seeing if %s (%d) can convert %s to %d[%c%c%c%c] [%d]",
  1042. (*it)->convertorName().toLatin1().constData(),
  1043. (*it)->type & t, mime.toLatin1().constData(),
  1044. flav, (flav >> 24) & 0xFF, (flav >> 16) & 0xFF, (flav >> 8) & 0xFF, (flav) & 0xFF,
  1045. (*it)->canConvert(mime,flav));
  1046. for(int i = 0; i < (*it)->countFlavors(); ++i) {
  1047. int f = (*it)->flavor(i);
  1048. qDebug(" %d) %d[%c%c%c%c] [%s]", i, f,
  1049. (f >> 24) & 0xFF, (f >> 16) & 0xFF, (f >> 8) & 0xFF, (f) & 0xFF,
  1050. (*it)->convertorName().toLatin1().constData());
  1051. }
  1052. #endif
  1053. if(((*it)->type & t) && (*it)->canConvert(mime, flav))
  1054. return (*it);
  1055. }
  1056. return 0;
  1057. }
  1058. /*!
  1059. Returns a MIME type of type \a t for \a flav, or 0 if none exists.
  1060. */
  1061. QString QMacPasteboardMime::flavorToMime(uchar t, QString flav)
  1062. {
  1063. MimeList *mimes = globalMimeList();
  1064. for(MimeList::const_iterator it = mimes->constBegin(); it != mimes->constEnd(); ++it) {
  1065. #ifdef DEBUG_MIME_MAPS
  1066. qDebug("QMacMIme::flavorToMime: attempting %s (%d) for flavor %d[%c%c%c%c] [%s]",
  1067. (*it)->convertorName().toLatin1().constData(),
  1068. (*it)->type & t, flav, (flav >> 24) & 0xFF, (flav >> 16) & 0xFF, (flav >> 8) & 0xFF, (flav) & 0xFF,
  1069. (*it)->mimeFor(flav).toLatin1().constData());
  1070. #endif
  1071. if((*it)->type & t) {
  1072. QString mimeType = (*it)->mimeFor(flav);
  1073. if(!mimeType.isNull())
  1074. return mimeType;
  1075. }
  1076. }
  1077. return QString();
  1078. }
  1079. /*!
  1080. Returns a list of all currently defined QMacPasteboardMime objects of type \a t.
  1081. */
  1082. QList<QMacPasteboardMime*> QMacPasteboardMime::all(uchar t)
  1083. {
  1084. MimeList ret;
  1085. MimeList *mimes = globalMimeList();
  1086. for(MimeList::const_iterator it = mimes->constBegin(); it != mimes->constEnd(); ++it) {
  1087. if((*it)->type & t)
  1088. ret.append((*it));
  1089. }
  1090. return ret;
  1091. }
  1092. /*!
  1093. \fn QString QMacPasteboardMime::convertorName()
  1094. Returns a name for the convertor.
  1095. All subclasses must reimplement this pure virtual function.
  1096. */
  1097. /*!
  1098. \fn bool QMacPasteboardMime::canConvert(const QString &mime, QString flav)
  1099. Returns true if the convertor can convert (both ways) between
  1100. \a mime and \a flav; otherwise returns false.
  1101. All subclasses must reimplement this pure virtual function.
  1102. */
  1103. /*!
  1104. \fn QString QMacPasteboardMime::mimeFor(QString flav)
  1105. Returns the MIME UTI used for Mac flavor \a flav, or 0 if this
  1106. convertor does not support \a flav.
  1107. All subclasses must reimplement this pure virtual function.
  1108. */
  1109. /*!
  1110. \fn QString QMacPasteboardMime::flavorFor(const QString &mime)
  1111. Returns the Mac UTI used for MIME type \a mime, or 0 if this
  1112. convertor does not support \a mime.
  1113. All subclasses must reimplement this pure virtual function.
  1114. */
  1115. /*!
  1116. \fn QVariant QMacPasteboardMime::convertToMime(const QString &mime, QList<QByteArray> data, QString flav)
  1117. Returns \a data converted from Mac UTI \a flav to MIME type \a
  1118. mime.
  1119. Note that Mac flavors must all be self-terminating. The input \a
  1120. data may contain trailing data.
  1121. All subclasses must reimplement this pure virtual function.
  1122. */
  1123. /*!
  1124. \fn QList<QByteArray> QMacPasteboardMime::convertFromMime(const QString &mime, QVariant data, QString flav)
  1125. Returns \a data converted from MIME type \a mime
  1126. to Mac UTI \a flav.
  1127. Note that Mac flavors must all be self-terminating. The return
  1128. value may contain trailing data.
  1129. All subclasses must reimplement this pure virtual function.
  1130. */
  1131. QT_END_NAMESPACE