X-Git-Url: https://git.saurik.com/wxWidgets.git/blobdiff_plain/26f86486b088495eeff542ec7d43caf78601fac1..47f12f58dbd5e62c5d1c635142a26c3e2208d012:/src/msw/clipbrd.cpp?ds=sidebyside diff --git a/src/msw/clipbrd.cpp b/src/msw/clipbrd.cpp index f15f9d8fc8..6b93ebf988 100644 --- a/src/msw/clipbrd.cpp +++ b/src/msw/clipbrd.cpp @@ -41,26 +41,43 @@ #include "wx/frame.h" #include "wx/bitmap.h" #include "wx/utils.h" + #include "wx/intl.h" #endif #if wxUSE_METAFILE #include "wx/metafile.h" #endif +#include "wx/log.h" #include "wx/clipbrd.h" +#include #include #include "wx/msw/private.h" #include "wx/msw/dib.h" -// wxDataObject is tied to OLE/drag and drop implementation, -// therefore so is wxClipboard :-( -#if wxUSE_DRAG_AND_DROP +// wxDataObject is tied to OLE/drag and drop implementation, therefore so are +// the functions using wxDataObject in wxClipboard +#define wxUSE_DATAOBJ wxUSE_DRAG_AND_DROP + +#if wxUSE_DATAOBJ #include "wx/dataobj.h" + + // use OLE clipboard + #define wxUSE_OLE_CLIPBOARD 1 +#else // !wxUSE_DATAOBJ + // use Win clipboard API + #define wxUSE_OLE_CLIPBOARD 0 #endif -#include +#if wxUSE_OLE_CLIPBOARD + #include +#endif // wxUSE_OLE_CLIPBOARD + +#ifdef __WIN16__ + #define memcpy hmemcpy +#endif // =========================================================================== // implementation @@ -74,7 +91,7 @@ static bool gs_wxClipboardIsOpen = FALSE; bool wxOpenClipboard() { - wxCHECK_MSG( !gs_wxClipboardIsOpen, TRUE, "clipboard already opened." ); + wxCHECK_MSG( !gs_wxClipboardIsOpen, TRUE, wxT("clipboard already opened.") ); wxWindow *win = wxTheApp->GetTopWindow(); if ( win ) @@ -88,7 +105,7 @@ bool wxOpenClipboard() } else { - wxLogDebug("Can not open clipboard without a main window,"); + wxLogDebug(wxT("Can not open clipboard without a main window.")); return FALSE; } @@ -96,7 +113,7 @@ bool wxOpenClipboard() bool wxCloseClipboard() { - wxCHECK_MSG( gs_wxClipboardIsOpen, FALSE, "clipboard is not opened" ); + wxCHECK_MSG( gs_wxClipboardIsOpen, FALSE, wxT("clipboard is not opened") ); gs_wxClipboardIsOpen = FALSE; @@ -129,7 +146,10 @@ bool wxIsClipboardOpened() bool wxIsClipboardFormatAvailable(wxDataFormat dataFormat) { - return ::IsClipboardFormatAvailable(dataFormat) != 0; + // for bitmaps, DIBs will also do + return (::IsClipboardFormatAvailable(dataFormat) != 0) || + (dataFormat.GetFormatId() == CF_BITMAP && + ::IsClipboardFormatAvailable(CF_DIB)); } bool wxSetClipboardData(wxDataFormat dataFormat, @@ -182,9 +202,9 @@ bool wxSetClipboardData(wxDataFormat dataFormat, wxBitmap *bitmap = (wxBitmap *)data; HBITMAP hBitmap = (HBITMAP)bitmap->GetHBITMAP(); // NULL palette means to use the system one - HANDLE hDIB = BitmapToDIB(hBitmap, (HPALETTE)NULL); + HANDLE hDIB = wxBitmapToDIB(hBitmap, (HPALETTE)NULL); handle = SetClipboardData(CF_DIB, hDIB); -#endif +#endif // wxUSE_IMAGE_LOADING_IN_MSW break; } @@ -193,11 +213,7 @@ bool wxSetClipboardData(wxDataFormat dataFormat, { wxMetafile *wxMF = (wxMetafile *)data; HANDLE data = GlobalAlloc(GHND, sizeof(METAFILEPICT) + 1); -#ifdef __WINDOWS_386__ - METAFILEPICT *mf = (METAFILEPICT *)MK_FP32(GlobalLock(data)); -#else METAFILEPICT *mf = (METAFILEPICT *)GlobalLock(data); -#endif mf->mm = wxMF->GetWindowsMappingMode(); mf->xExt = width; @@ -234,19 +250,9 @@ bool wxSetClipboardData(wxDataFormat dataFormat, HANDLE hGlobalMemory = GlobalAlloc(GHND, l); if ( hGlobalMemory ) { -#ifdef __WINDOWS_386__ - LPSTR lpGlobalMemory = (LPSTR)MK_FP32(GlobalLock(hGlobalMemory)); -#else LPSTR lpGlobalMemory = (LPSTR)GlobalLock(hGlobalMemory); -#endif -#ifdef __WIN32__ memcpy(lpGlobalMemory, s, l); -#elif defined(__WATCOMC__) && defined(__WINDOWS_386__) - memcpy(lpGlobalMemory, s, l); -#else - hmemcpy(lpGlobalMemory, s, l); -#endif GlobalUnlock(hGlobalMemory); } @@ -324,7 +330,6 @@ void *wxGetClipboardData(wxDataFormat dataFormat, long *len) case CF_TIFF: case CF_PALETTE: case wxDF_DIB: - default: { wxLogError(_("Unsupported clipboard format.")); return FALSE; @@ -348,25 +353,39 @@ void *wxGetClipboardData(wxDataFormat dataFormat, long *len) if (!s) break; -#ifdef __WINDOWS_386__ - LPSTR lpGlobalMemory = (LPSTR)MK_FP32(GlobalLock(hGlobalMemory)); -#else LPSTR lpGlobalMemory = (LPSTR)::GlobalLock(hGlobalMemory); -#endif -#ifdef __WIN32__ - memcpy(s, lpGlobalMemory, hsize); -#elif __WATCOMC__ && defined(__WINDOWS_386__) memcpy(s, lpGlobalMemory, hsize); -#else - hmemcpy(s, lpGlobalMemory, hsize); -#endif ::GlobalUnlock(hGlobalMemory); retval = s; break; } + + default: + { + HANDLE hGlobalMemory = ::GetClipboardData(dataFormat); + if ( !hGlobalMemory ) + break; + + DWORD size = ::GlobalSize(hGlobalMemory); + if ( len ) + *len = size; + + void *buf = malloc(size); + if ( !buf ) + break; + + LPSTR lpGlobalMemory = (LPSTR)::GlobalLock(hGlobalMemory); + + memcpy(buf, lpGlobalMemory, size); + + ::GlobalUnlock(hGlobalMemory); + + retval = buf; + break; + } } if ( !retval ) @@ -377,18 +396,18 @@ void *wxGetClipboardData(wxDataFormat dataFormat, long *len) return retval; } -wxDataFormat wxEnumClipboardFormats(wxDataFormat dataFormat) +wxDataFormat wxEnumClipboardFormats(wxDataFormat dataFormat) { - return (wxDataFormat)::EnumClipboardFormats(dataFormat); + return ::EnumClipboardFormats(dataFormat); } -int wxRegisterClipboardFormat(char *formatName) +int wxRegisterClipboardFormat(wxChar *formatName) { return ::RegisterClipboardFormat(formatName); } bool wxGetClipboardFormatName(wxDataFormat dataFormat, - char *formatName, + wxChar *formatName, int maxCount) { return ::GetClipboardFormatName((int)dataFormat, formatName, maxCount) > 0; @@ -400,29 +419,73 @@ bool wxGetClipboardFormatName(wxDataFormat dataFormat, IMPLEMENT_DYNAMIC_CLASS(wxClipboard, wxObject) -wxClipboard* wxTheClipboard = (wxClipboard *)NULL; - wxClipboard::wxClipboard() { + m_clearOnExit = FALSE; } wxClipboard::~wxClipboard() { - Clear(); + if ( m_clearOnExit ) + { + Clear(); + } } void wxClipboard::Clear() { +#if wxUSE_OLE_CLIPBOARD + if ( FAILED(OleSetClipboard(NULL)) ) + { + wxLogLastError("OleSetClipboard(NULL)"); + } +#endif +} + +bool wxClipboard::Flush() +{ +#if wxUSE_OLE_CLIPBOARD + if ( FAILED(OleFlushClipboard()) ) + { + wxLogLastError("OleFlushClipboard"); + + return FALSE; + } + else + { + m_clearOnExit = FALSE; + + return TRUE; + } +#else // !wxUSE_OLE_CLIPBOARD + return FALSE; +#endif // wxUSE_OLE_CLIPBOARD/!wxUSE_OLE_CLIPBOARD } bool wxClipboard::Open() { + // OLE opens clipboard for us +#if wxUSE_OLE_CLIPBOARD + return TRUE; +#else return wxOpenClipboard(); +#endif +} + +bool wxClipboard::IsOpened() const +{ +#if wxUSE_OLE_CLIPBOARD + return TRUE; +#else + return wxIsClipboardOpened(); +#endif } bool wxClipboard::SetData( wxDataObject *data ) { +#if !wxUSE_OLE_CLIPBOARD (void)wxEmptyClipboard(); +#endif // wxUSE_OLE_CLIPBOARD if ( data ) return AddData(data); @@ -432,10 +495,35 @@ bool wxClipboard::SetData( wxDataObject *data ) bool wxClipboard::AddData( wxDataObject *data ) { - wxCHECK_MSG( data, FALSE, "data is invalid" ); + wxCHECK_MSG( data, FALSE, wxT("data is invalid") ); + +#if wxUSE_OLE_CLIPBOARD + HRESULT hr = OleSetClipboard(data->GetInterface()); + if ( FAILED(hr) ) + { + wxLogSysError(hr, _("Failed to put data on the clipboard")); + + // don't free anything in this case + + return FALSE; + } + + // we have a problem here because we should delete wxDataObject, but we + // can't do it because IDataObject which we just gave to the clipboard + // would try to use it when it will need the data. IDataObject is ref + // counted and so doesn't suffer from such problem, so we release it now + // and tell it to delete wxDataObject when it is deleted itself. + data->SetAutoDelete(); + + // we have to call either OleSetClipboard(NULL) or OleFlushClipboard() when + // using OLE clipboard when the app terminates - by default, we call + // OleSetClipboard(NULL) which won't waste RAM, but the app can call + // wxClipboard::Flush() to chaneg this + m_clearOnExit = TRUE; -#if wxUSE_DRAG_AND_DROP - wxCHECK_MSG( wxIsClipboardOpened(), FALSE, "clipboard not open" ); + return TRUE; +#elif wxUSE_DATAOBJ + wxCHECK_MSG( wxIsClipboardOpened(), FALSE, wxT("clipboard not open") ); wxDataFormat format = data->GetFormat(); @@ -470,21 +558,19 @@ bool wxClipboard::AddData( wxDataObject *data ) #endif // wxUSE_METAFILE default: - wxLogError(_("Can not put data in format '%s' on clipboard."), - wxDataObject::GetFormatName(format)); - - return FALSE; + return wxSetClipboardData(data); } - - return FALSE; -#else +#else // !wxUSE_DATAOBJ return FALSE; -#endif +#endif // wxUSE_DATAOBJ/!wxUSE_DATAOBJ } void wxClipboard::Close() { + // OLE closes clipboard for us +#if !wxUSE_OLE_CLIPBOARD wxCloseClipboard(); +#endif } bool wxClipboard::IsSupported( wxDataFormat format ) @@ -492,86 +578,195 @@ bool wxClipboard::IsSupported( wxDataFormat format ) return wxIsClipboardFormatAvailable(format); } -bool wxClipboard::GetData( wxDataObject *data ) +bool wxClipboard::GetData( wxDataObject& data ) { - wxCHECK_MSG( wxIsClipboardOpened(), FALSE, "clipboard not open" ); +#if wxUSE_OLE_CLIPBOARD + IDataObject *pDataObject = NULL; + HRESULT hr = OleGetClipboard(&pDataObject); + if ( FAILED(hr) || !pDataObject ) + { + wxLogSysError(hr, _("Failed to get data from the clipboard")); -#if wxUSE_DRAG_AND_DROP - wxDataFormat format = data->GetFormat(); + return FALSE; + } + + // build the list of supported formats + size_t nFormats = data.GetFormatCount(wxDataObject::Set); + wxDataFormat format, *formats; + if ( nFormats == 1 ) + { + // the most common case + formats = &format; + } + else + { + // bad luck, need to alloc mem + formats = new wxDataFormat[nFormats]; + } + + data.GetAllFormats(formats, wxDataObject::Set); + + // get the format enumerator + bool result = FALSE; + wxArrayInt supportedFormats; + IEnumFORMATETC *pEnumFormatEtc = NULL; + hr = pDataObject->EnumFormatEtc(DATADIR_GET, &pEnumFormatEtc); + if ( FAILED(hr) || !pEnumFormatEtc ) + { + wxLogSysError(hr, + _("Failed to retrieve the supported clipboard formats")); + } + else + { + // ask for the supported formats and see if there are any we support + FORMATETC formatEtc; + for ( ;; ) + { + ULONG nCount; + hr = pEnumFormatEtc->Next(1, &formatEtc, &nCount); + + // don't use FAILED() because S_FALSE would pass it + if ( hr != S_OK ) + { + // no more formats + break; + } + + CLIPFORMAT cf = formatEtc.cfFormat; + +#ifdef __WXDEBUG__ + wxLogTrace(wxTRACE_OleCalls, + wxT("Object on the clipboard supports format %s."), + wxDataObject::GetFormatName(cf)); +#endif // Debug + + // is supported? + for ( size_t n = 0; n < nFormats; n++ ) + { + if ( formats[n].GetFormatId() == cf ) + { + if ( supportedFormats.Index(cf) == wxNOT_FOUND ) + { + supportedFormats.Add(cf); + } + } + } + } + + pEnumFormatEtc->Release(); + } + + if ( formats != &format ) + { + delete [] formats; + } + //else: we didn't allocate any memory + + if ( !supportedFormats.IsEmpty() ) + { + FORMATETC formatEtc; + formatEtc.ptd = NULL; + formatEtc.dwAspect = DVASPECT_CONTENT; + formatEtc.lindex = -1; + formatEtc.tymed = TYMED_HGLOBAL; + + size_t nSupportedFormats = supportedFormats.GetCount(); + for ( size_t n = 0; !result && (n < nSupportedFormats); n++ ) + { + STGMEDIUM medium; + formatEtc.cfFormat = supportedFormats[n]; + + // try to get data + hr = pDataObject->GetData(&formatEtc, &medium); + if ( FAILED(hr) ) + { + // try other tymed for GDI objects + if ( formatEtc.cfFormat == CF_BITMAP ) + { + formatEtc.tymed = TYMED_HGLOBAL; + hr = pDataObject->GetData(&formatEtc, &medium); + } + } + + if ( SUCCEEDED(hr) ) + { + // pass the data to the data object + hr = data.GetInterface()->SetData(&formatEtc, &medium, TRUE); + if ( FAILED(hr) ) + { + wxLogDebug(wxT("Failed to set data in wxIDataObject")); + + // IDataObject only takes the ownership of data if it + // successfully got it - which is not the case here + ReleaseStgMedium(&medium); + } + else + { + result = TRUE; + } + } + //else: unsupported tymed? + } + } + //else: unsupported format + + // clean up and return + pDataObject->Release(); + + return result; +#elif wxUSE_DATAOBJ + wxCHECK_MSG( wxIsClipboardOpened(), FALSE, wxT("clipboard not open") ); + + wxDataFormat format = data.GetFormat(); switch ( format ) { case wxDF_TEXT: case wxDF_OEMTEXT: { - wxTextDataObject* textDataObject = (wxTextDataObject*) data; - char* s = (char*) wxGetClipboardData(format); - if ( s ) - { - textDataObject->SetText(s); - delete[] s; - return TRUE; - } - else + wxTextDataObject& textDataObject = (wxTextDataObject &)data; + char* s = (char*)wxGetClipboardData(format); + if ( !s ) return FALSE; + + textDataObject.SetText(s); + delete [] s; + + return TRUE; } case wxDF_BITMAP: case wxDF_DIB: { - wxBitmapDataObject* bitmapDataObject = (wxBitmapDataObject *)data; + wxBitmapDataObject& bitmapDataObject = (wxBitmapDataObject &)data; wxBitmap* bitmap = (wxBitmap *)wxGetClipboardData(data->GetFormat()); - if (bitmap) - { - bitmapDataObject->SetBitmap(* bitmap); - delete bitmap; - return TRUE; - } - else + if ( !bitmap ) return FALSE; + + bitmapDataObject.SetBitmap(*bitmap); + delete bitmap; + + return TRUE; } #if wxUSE_METAFILE case wxDF_METAFILE: { - wxMetafileDataObject* metaFileDataObject = (wxMetafileDataObject *)data; + wxMetafileDataObject& metaFileDataObject = (wxMetafileDataObject &)data; wxMetafile* metaFile = (wxMetafile *)wxGetClipboardData(wxDF_METAFILE); - if (metaFile) - { - metaFileDataObject->SetMetafile(*metaFile); - delete metaFile; - return TRUE; - } - else + if ( !metaFile ) return FALSE; - } -#endif - default: - wxLogError(_("Can not get data in format '%s' from clipboard."), - wxDataObject::GetFormatName(format)); - - return FALSE; - } -#else - return FALSE; -#endif -} -//----------------------------------------------------------------------------- -// wxClipboardModule -//----------------------------------------------------------------------------- + metaFileDataObject.SetMetafile(*metaFile); + delete metaFile; -IMPLEMENT_DYNAMIC_CLASS(wxClipboardModule,wxModule) - -bool wxClipboardModule::OnInit() -{ - wxTheClipboard = new wxClipboard(); - - return TRUE; -} + return TRUE; + } +#endif // wxUSE_METAFILE + } +#else // !wxUSE_DATAOBJ + wxFAIL_MSG( wxT("no clipboard implementation") ); +#endif // wxUSE_OLE_CLIPBOARD/wxUSE_DATAOBJ -void wxClipboardModule::OnExit() -{ - if (wxTheClipboard) delete wxTheClipboard; - wxTheClipboard = (wxClipboard*) NULL; + return FALSE; } #else