// -*- C++ -*- /** * \file qt4/GuiClipboard.cpp * This file is part of LyX, the document processor. * Licence details can be found in the file COPYING. * * \author John Levon * \author Abdelrazak Younes * * Full author contact details are available in file CREDITS. */ #include #include "FileDialog.h" #include "GuiClipboard.h" #include "qt_helpers.h" #include "Buffer.h" #include "BufferView.h" #include "Cursor.h" #include "support/assert.h" #include "support/convert.h" #include "support/debug.h" #include "support/filetools.h" #include "support/FileFilterList.h" #include "support/gettext.h" #include "support/lstrings.h" #ifdef Q_WS_MACX #include "support/linkback/LinkBackProxy.h" #endif // Q_WS_MACX #include "frontends/alert.h" #include #include #include #include #include #include #include #include #include #include #ifdef Q_WS_WIN #include #ifdef Q_CYGWIN_WIN #include #endif #include #endif // Q_WS_WIN #include using namespace std; using namespace lyx::support; static char const * const lyx_mime_type = "application/x-lyx"; static char const * const pdf_mime_type = "application/pdf"; static char const * const emf_mime_type = "image/x-emf"; static char const * const wmf_mime_type = "image/x-wmf"; namespace lyx { namespace frontend { #ifdef Q_WS_WIN static FORMATETC cfFromMime(QString const & mimetype) { FORMATETC formatetc; if (mimetype == emf_mime_type) { formatetc.cfFormat = CF_ENHMETAFILE; formatetc.tymed = TYMED_ENHMF; } else if (mimetype == wmf_mime_type) { formatetc.cfFormat = CF_METAFILEPICT; formatetc.tymed = TYMED_MFPICT; } formatetc.ptd = 0; formatetc.dwAspect = DVASPECT_CONTENT; formatetc.lindex = -1; return formatetc; } class QWindowsMimeMetafile : public QWindowsMime { public: bool canConvertFromMime(FORMATETC const & formatetc, QMimeData const * mimedata) const; bool canConvertToMime(QString const & mimetype, IDataObject * pDataObj) const; bool convertFromMime(FORMATETC const & formatetc, const QMimeData * mimedata, STGMEDIUM * pmedium) const; QVariant convertToMime(QString const & mimetype, IDataObject * pDataObj, QVariant::Type preferredType) const; QVector formatsForMime(QString const & mimeType, QMimeData const * mimeData) const; QString mimeForFormat(FORMATETC const &) const; }; QString QWindowsMimeMetafile::mimeForFormat(FORMATETC const & formatetc) const { QString f; if (formatetc.cfFormat == CF_ENHMETAFILE) f = emf_mime_type; else if (formatetc.cfFormat == CF_METAFILEPICT) f = wmf_mime_type; return f; } bool QWindowsMimeMetafile::canConvertFromMime(FORMATETC const & formatetc, QMimeData const * mimedata) const { return false; } bool QWindowsMimeMetafile::canConvertToMime(QString const & mimetype, IDataObject * pDataObj) const { if (mimetype != emf_mime_type && mimetype != wmf_mime_type) return false; FORMATETC formatetc = cfFromMime(mimetype); return pDataObj->QueryGetData(&formatetc) == S_OK; } bool QWindowsMimeMetafile::convertFromMime(FORMATETC const & formatetc, QMimeData const * mimedata, STGMEDIUM * pmedium) const { return false; } QVariant QWindowsMimeMetafile::convertToMime(QString const & mimetype, IDataObject * pDataObj, QVariant::Type preferredType) const { QByteArray data; if (!canConvertToMime(mimetype, pDataObj)) return data; FORMATETC formatetc = cfFromMime(mimetype); STGMEDIUM s; if (pDataObj->GetData(&formatetc, &s) != S_OK) return data; int dataSize; if (s.tymed == TYMED_ENHMF) { dataSize = GetEnhMetaFileBits(s.hEnhMetaFile, 0, 0); data.resize(dataSize); dataSize = GetEnhMetaFileBits(s.hEnhMetaFile, dataSize, (LPBYTE)data.data()); } else if (s.tymed == TYMED_MFPICT) { dataSize = GetMetaFileBitsEx((HMETAFILE)s.hMetaFilePict, 0, 0); data.resize(dataSize); dataSize = GetMetaFileBitsEx((HMETAFILE)s.hMetaFilePict, dataSize, (LPBYTE)data.data()); } data.detach(); ReleaseStgMedium(&s); return data; } QVector QWindowsMimeMetafile::formatsForMime( QString const & mimetype, QMimeData const * mimedata) const { QVector formats; formats += cfFromMime(mimetype); return formats; } static QWindowsMimeMetafile * metafileWindowsMime; #endif // Q_WS_WIN #ifdef Q_WS_MACX class QMacPasteboardMimeGraphics : public QMacPasteboardMime { public: QMacPasteboardMimeGraphics() : QMacPasteboardMime(MIME_QT_CONVERTOR|MIME_ALL) {} QString convertorName(); QString flavorFor(QString const & mime); QString mimeFor(QString flav); bool canConvert(QString const & mime, QString flav); QVariant convertToMime(QString const & mime, QList data, QString flav); QList convertFromMime(QString const & mime, QVariant data, QString flav); }; QString QMacPasteboardMimeGraphics::convertorName() { return "Graphics"; } QString QMacPasteboardMimeGraphics::flavorFor(QString const & mime) { LYXERR(Debug::ACTION, "flavorFor " << fromqstr(mime)); if (mime == QLatin1String(pdf_mime_type)) return QLatin1String("com.adobe.pdf"); return QString(); } QString QMacPasteboardMimeGraphics::mimeFor(QString flav) { LYXERR(Debug::ACTION, "mimeFor " << fromqstr(flav)); if (flav == QLatin1String("com.adobe.pdf")) return QLatin1String(pdf_mime_type); return QString(); } bool QMacPasteboardMimeGraphics::canConvert(QString const & mime, QString flav) { return mimeFor(flav) == mime; } QVariant QMacPasteboardMimeGraphics::convertToMime(QString const & mime, QList data, QString) { if(data.count() > 1) qWarning("QMacPasteboardMimeGraphics: Cannot handle multiple member data"); return data.first(); } QList QMacPasteboardMimeGraphics::convertFromMime(QString const & mime, QVariant data, QString) { QList ret; ret.append(data.toByteArray()); return ret; } static QMacPasteboardMimeGraphics * graphicsPasteboardMime; #endif // Q_WS_MACX GuiClipboard::GuiClipboard() { connect(qApp->clipboard(), SIGNAL(dataChanged()), this, SLOT(on_dataChanged())); // initialize clipboard status. on_dataChanged(); #ifdef Q_WS_MACX if (!graphicsPasteboardMime) graphicsPasteboardMime = new QMacPasteboardMimeGraphics(); #endif // Q_WS_MACX #ifdef Q_WS_WIN if (!metafileWindowsMime) metafileWindowsMime = new QWindowsMimeMetafile(); #endif // Q_WS_WIN } GuiClipboard::~GuiClipboard() { #ifdef Q_WS_MACX closeAllLinkBackLinks(); #endif // Q_WS_MACX } string const GuiClipboard::getAsLyX() const { LYXERR(Debug::ACTION, "GuiClipboard::getAsLyX(): `"); // We don't convert encodings here since the encoding of the // clipboard contents is specified in the data itself QMimeData const * source = qApp->clipboard()->mimeData(QClipboard::Clipboard); if (!source) { LYXERR(Debug::ACTION, "' (no QMimeData)"); return string(); } if (source->hasFormat(lyx_mime_type)) { // data from ourself or some other LyX instance QByteArray const ar = source->data(lyx_mime_type); string const s(ar.data(), ar.count()); LYXERR(Debug::ACTION, s << "'"); return s; } LYXERR(Debug::ACTION, "'"); return string(); } FileName GuiClipboard::getPastedGraphicsFileName(Cursor const & cur, Clipboard::GraphicsType & type) const { // create file dialog filter according to the existing types in the clipboard vector types; if (hasGraphicsContents(Clipboard::EmfGraphicsType)) types.push_back(Clipboard::EmfGraphicsType); if (hasGraphicsContents(Clipboard::WmfGraphicsType)) types.push_back(Clipboard::WmfGraphicsType); if (hasGraphicsContents(Clipboard::LinkBackGraphicsType)) types.push_back(Clipboard::LinkBackGraphicsType); if (hasGraphicsContents(Clipboard::PdfGraphicsType)) types.push_back(Clipboard::PdfGraphicsType); if (hasGraphicsContents(Clipboard::PngGraphicsType)) types.push_back(Clipboard::PngGraphicsType); if (hasGraphicsContents(Clipboard::JpegGraphicsType)) types.push_back(Clipboard::JpegGraphicsType); LASSERT(!types.empty(), /**/); // select prefered type if AnyGraphicsType was passed if (type == Clipboard::AnyGraphicsType) type = types.front(); // which extension? map extensions; map typeNames; extensions[Clipboard::EmfGraphicsType] = "emf"; extensions[Clipboard::WmfGraphicsType] = "wmf"; extensions[Clipboard::LinkBackGraphicsType] = "linkback"; extensions[Clipboard::PdfGraphicsType] = "pdf"; extensions[Clipboard::PngGraphicsType] = "png"; extensions[Clipboard::JpegGraphicsType] = "jpeg"; typeNames[Clipboard::EmfGraphicsType] = _("Enhanced Metafile"); typeNames[Clipboard::WmfGraphicsType] = _("Windows Metafile"); typeNames[Clipboard::LinkBackGraphicsType] = _("LinkBack PDF"); typeNames[Clipboard::PdfGraphicsType] = _("PDF"); typeNames[Clipboard::PngGraphicsType] = _("PNG"); typeNames[Clipboard::JpegGraphicsType] = _("JPEG"); // find unused filename with primary extension string document_path = cur.buffer().fileName().onlyPath().absFilename(); unsigned newfile_number = 0; FileName filename; do { ++newfile_number; filename = FileName(addName(document_path, to_utf8(_("pasted")) + convert(newfile_number) + "." + extensions[type])); } while (filename.isReadableFile()); while (true) { // create file type filter, putting the prefered on to the front docstring filterSpec; for (size_t i = 0; i != types.size(); ++i) { docstring s = bformat(_("%1$s Files"), typeNames[types[i]]) + " (*." + from_ascii(extensions[types[i]]) + ")"; if (types[i] == type) filterSpec = s + filterSpec; else filterSpec += ";;" + s; } FileFilterList const filter(filterSpec); // show save dialog for the graphic FileDialog dlg(qt_("Choose a filename to save the pasted graphic as")); FileDialog::Result result = dlg.save(toqstr(filename.onlyPath().absFilename()), filter, toqstr(filename.onlyFileName())); if (result.first == FileDialog::Later) return FileName(); string newFilename = fromqstr(result.second); if (newFilename.empty()) { cur.bv().message(_("Canceled.")); return FileName(); } filename.set(newFilename); // check the extension (the user could have changed it) if (!suffixIs(ascii_lowercase(filename.absFilename()), "." + extensions[type])) { // the user changed the extension. Check if the type is available size_t i; for (i = 1; i != types.size(); ++i) { if (suffixIs(ascii_lowercase(filename.absFilename()), "." + extensions[types[i]])) { type = types[i]; break; } } // invalid extension found, or none at all. In the latter // case set the default extensions. if (i == types.size() && filename.onlyFileName().find('.') == string::npos) { filename.changeExtension("." + extensions[type]); } } // check whether the file exists and warn the user if (!filename.exists()) break; int ret = frontend::Alert::prompt( _("Overwrite external file?"), bformat(_("File %1$s already exists, do you want to overwrite it?"), from_utf8(filename.absFilename())), 1, 1, _("&Overwrite"), _("&Cancel")); if (ret == 0) // overwrite, hence break the dialog loop break; // not overwrite, hence show the dialog again (i.e. loop) } return filename; } FileName GuiClipboard::getAsGraphics(Cursor const & cur, GraphicsType type) const { // get the filename from the user FileName filename = getPastedGraphicsFileName(cur, type); if (filename.empty()) return FileName(); // handle image cases first if (type == PngGraphicsType || type == JpegGraphicsType) { // get image from QImage from clipboard QImage image = qApp->clipboard()->image(); if (image.isNull()) { LYXERR(Debug::ACTION, "No image in clipboard"); return FileName(); } // convert into graphics format QByteArray ar; QBuffer buffer(&ar); buffer.open(QIODevice::WriteOnly); if (type == PngGraphicsType) image.save(toqstr(filename.absFilename()), "PNG"); else if (type == JpegGraphicsType) image.save(toqstr(filename.absFilename()), "JPEG"); else LASSERT(false, /**/); return filename; } // get mime data QMimeData const * source = qApp->clipboard()->mimeData(QClipboard::Clipboard); if (!source) { LYXERR(Debug::ACTION, "0 bytes (no QMimeData)"); return FileName(); } // get mime for type QString mime; switch (type) { case PdfGraphicsType: mime = pdf_mime_type; break; case LinkBackGraphicsType: mime = pdf_mime_type; break; case EmfGraphicsType: mime = emf_mime_type; break; case WmfGraphicsType: mime = wmf_mime_type; break; default: LASSERT(false, /**/); } // get data if (!source->hasFormat(mime)) return FileName(); // data from ourself or some other LyX instance QByteArray const ar = source->data(mime); LYXERR(Debug::ACTION, "Getting from clipboard: mime = " << mime.data() << "length = " << ar.count()); QFile f(toqstr(filename.absFilename())); if (!f.open(QIODevice::WriteOnly | QIODevice::Truncate)) { LYXERR(Debug::ACTION, "Error opening file " << filename.absFilename() << " for writing"); return FileName(); } // write the (LinkBack) PDF data f.write(ar); if (type == LinkBackGraphicsType) { #ifdef Q_WS_MACX void const * linkBackData; unsigned linkBackLen; getLinkBackData(&linkBackData, &linkBackLen); f.write((char *)linkBackData, linkBackLen); quint32 pdfLen = ar.size(); QDataStream ds(&f); ds << pdfLen; // big endian by default #else // only non-Mac this should never happen LASSERT(false, /**/); #endif // Q_WS_MACX } f.close(); return filename; } docstring const GuiClipboard::getAsText() const { // text data from other applications QString const str = qApp->clipboard()->text(QClipboard::Clipboard) .normalized(QString::NormalizationForm_C); LYXERR(Debug::ACTION, "GuiClipboard::getAsText(): `" << fromqstr(str) << "'"); if (str.isNull()) return docstring(); return internalLineEnding(qstring_to_ucs4(str)); } void GuiClipboard::put(string const & lyx, docstring const & text) { LYXERR(Debug::ACTION, "GuiClipboard::put(`" << lyx << "' `" << to_utf8(text) << "')"); // We don't convert the encoding of lyx since the encoding of the // clipboard contents is specified in the data itself QMimeData * data = new QMimeData; if (!lyx.empty()) { QByteArray const qlyx(lyx.c_str(), lyx.size()); data->setData(lyx_mime_type, qlyx); } // Don't test for text.empty() since we want to be able to clear the // clipboard. QString const qtext = toqstr(text); data->setText(qtext); qApp->clipboard()->setMimeData(data, QClipboard::Clipboard); } bool GuiClipboard::hasLyXContents() const { QMimeData const * const source = qApp->clipboard()->mimeData(QClipboard::Clipboard); return source && source->hasFormat(lyx_mime_type); } bool GuiClipboard::hasGraphicsContents(Clipboard::GraphicsType type) const { if (type == AnyGraphicsType) { return hasGraphicsContents(PdfGraphicsType) || hasGraphicsContents(PngGraphicsType) || hasGraphicsContents(JpegGraphicsType) || hasGraphicsContents(EmfGraphicsType) || hasGraphicsContents(WmfGraphicsType) || hasGraphicsContents(LinkBackGraphicsType); } QMimeData const * const source = qApp->clipboard()->mimeData(QClipboard::Clipboard); // handle image cases first if (type == PngGraphicsType || type == JpegGraphicsType) return source->hasImage(); // handle LinkBack for Mac #ifdef Q_WS_MACX if (type == LinkBackGraphicsType) return isLinkBackDataInPasteboard(); #else if (type == LinkBackGraphicsType) return false; #endif // Q_WS_MACX // get mime data QStringList const & formats = source->formats(); LYXERR(Debug::ACTION, "We found " << formats.size() << " formats"); for (int i = 0; i < formats.size(); ++i) { LYXERR(Debug::ACTION, "Found format " << fromqstr(formats[i])); } // compute mime for type QString mime; switch (type) { case EmfGraphicsType: mime = emf_mime_type; break; case WmfGraphicsType: mime = wmf_mime_type; break; case PdfGraphicsType: mime = pdf_mime_type; break; default: LASSERT(false, /**/); } return source && source->hasFormat(mime); } bool GuiClipboard::isInternal() const { // ownsClipboard() is also true for stuff coming from dialogs, e.g. // the preamble dialog // FIXME: This does only work on X11, since ownsClipboard() is // hardwired to return false on Windows and OS X. return qApp->clipboard()->ownsClipboard() && hasLyXContents(); } bool GuiClipboard::hasInternal() const { // Windows and Mac OS X does not have the concept of ownership; // the clipboard is a fully global resource so all applications // are notified of changes. #if (defined(Q_WS_X11)) return true; #else return false; #endif } void GuiClipboard::on_dataChanged() { QMimeData const * const source = qApp->clipboard()->mimeData(QClipboard::Clipboard); QStringList l = source->formats(); LYXERR(Debug::ACTION, "Qt Clipboard changed. We found the following mime types:"); for (int i = 0; i < l.count(); i++) { LYXERR(Debug::ACTION, fromqstr(l.value(i))); } text_clipboard_empty_ = qApp->clipboard()-> text(QClipboard::Clipboard).isEmpty(); has_lyx_contents_ = hasLyXContents(); has_graphics_contents_ = hasGraphicsContents(); } bool GuiClipboard::empty() const { // We need to check both the plaintext and the LyX version of the // clipboard. The plaintext version is empty if the LyX version // contains only one inset, and the LyX version is empty if the // clipboard does not come from LyX. if (!text_clipboard_empty_) return false; return !has_lyx_contents_ && !has_graphics_contents_; } } // namespace frontend } // namespace lyx #include "GuiClipboard_moc.cpp"