Merge branch 'bmp-dataobj-fix'
Fix crash when using wxDataObjectComposite and other clipboard-related cleanup. See #23659.
This commit is contained in:
commit
de0595108a
6 changed files with 72 additions and 526 deletions
|
|
@ -323,7 +323,6 @@ library:
|
||||||
manifest from wxWidgets RC file. See also wxUSE_RC_MANIFEST.}
|
manifest from wxWidgets RC file. See also wxUSE_RC_MANIFEST.}
|
||||||
@itemdef{wxUSE_OLE, Enables OLE helper routines.}
|
@itemdef{wxUSE_OLE, Enables OLE helper routines.}
|
||||||
@itemdef{wxUSE_OLE_AUTOMATION, Enable OLE automation utilities.}
|
@itemdef{wxUSE_OLE_AUTOMATION, Enable OLE automation utilities.}
|
||||||
@itemdef{wxUSE_OLE_CLIPBOARD, Use OLE clipboard.}
|
|
||||||
@itemdef{wxUSE_POSTSCRIPT_ARCHITECTURE_IN_MSW, Use PS printing in wxMSW.}
|
@itemdef{wxUSE_POSTSCRIPT_ARCHITECTURE_IN_MSW, Use PS printing in wxMSW.}
|
||||||
@itemdef{wxUSE_PS_PRINTING, See src/msw/dcprint.cpp file.}
|
@itemdef{wxUSE_PS_PRINTING, See src/msw/dcprint.cpp file.}
|
||||||
@itemdef{wxUSE_RC_MANIFEST, Include manifest for common controls library v6
|
@itemdef{wxUSE_RC_MANIFEST, Include manifest for common controls library v6
|
||||||
|
|
|
||||||
|
|
@ -302,6 +302,15 @@
|
||||||
# endif
|
# endif
|
||||||
# endif
|
# endif
|
||||||
|
|
||||||
|
# if wxUSE_CLIPBOARD
|
||||||
|
# ifdef wxABORT_ON_CONFIG_ERROR
|
||||||
|
# error "wxUSE_CLIPBOARD requires wxUSE_OLE"
|
||||||
|
# else
|
||||||
|
# undef wxUSE_CLIPBOARD
|
||||||
|
# define wxUSE_CLIPBOARD 0
|
||||||
|
# endif
|
||||||
|
# endif
|
||||||
|
|
||||||
# if wxUSE_DRAG_AND_DROP
|
# if wxUSE_DRAG_AND_DROP
|
||||||
# ifdef wxABORT_ON_CONFIG_ERROR
|
# ifdef wxABORT_ON_CONFIG_ERROR
|
||||||
# error "wxUSE_DRAG_AND_DROP requires wxUSE_OLE"
|
# error "wxUSE_DRAG_AND_DROP requires wxUSE_OLE"
|
||||||
|
|
|
||||||
|
|
@ -13,24 +13,20 @@
|
||||||
|
|
||||||
#if wxUSE_CLIPBOARD
|
#if wxUSE_CLIPBOARD
|
||||||
|
|
||||||
// These functions superseded by wxClipboard, but retained in order to
|
// Deprecated wxMSW-only function superseded by wxClipboard, don't use them and
|
||||||
// implement wxClipboard, and for compatibility.
|
// use that class instead.
|
||||||
|
#ifdef WXWIN_COMPATIBILITY_3_2
|
||||||
|
|
||||||
// open/close the clipboard
|
wxDEPRECATED_MSG("Use wxClipboard") WXDLLIMPEXP_CORE bool wxOpenClipboard();
|
||||||
WXDLLIMPEXP_CORE bool wxOpenClipboard();
|
wxDEPRECATED_MSG("Use wxClipboard") WXDLLIMPEXP_CORE bool wxIsClipboardOpened();
|
||||||
WXDLLIMPEXP_CORE bool wxIsClipboardOpened();
|
wxDEPRECATED_MSG("Use wxClipboard") WXDLLIMPEXP_CORE bool wxClipboardOpen();
|
||||||
#define wxClipboardOpen wxIsClipboardOpened
|
wxDEPRECATED_MSG("Use wxClipboard") WXDLLIMPEXP_CORE bool wxCloseClipboard();
|
||||||
WXDLLIMPEXP_CORE bool wxCloseClipboard();
|
wxDEPRECATED_MSG("Use wxClipboard") WXDLLIMPEXP_CORE bool wxEmptyClipboard();
|
||||||
|
|
||||||
// get/set data
|
#endif // WXWIN_COMPATIBILITY_3_2
|
||||||
WXDLLIMPEXP_CORE bool wxEmptyClipboard();
|
|
||||||
#if !wxUSE_OLE
|
|
||||||
WXDLLIMPEXP_CORE bool wxSetClipboardData(wxDataFormat dataFormat,
|
|
||||||
const void *data,
|
|
||||||
int width = 0, int height = 0);
|
|
||||||
#endif // !wxUSE_OLE
|
|
||||||
|
|
||||||
// clipboard formats
|
// Non-deprecated but still wxMSW-specific functions working with clipboard
|
||||||
|
// formats -- don't use them neither.
|
||||||
WXDLLIMPEXP_CORE bool wxIsClipboardFormatAvailable(wxDataFormat dataFormat);
|
WXDLLIMPEXP_CORE bool wxIsClipboardFormatAvailable(wxDataFormat dataFormat);
|
||||||
WXDLLIMPEXP_CORE wxDataFormat wxEnumClipboardFormats(wxDataFormat dataFormat);
|
WXDLLIMPEXP_CORE wxDataFormat wxEnumClipboardFormats(wxDataFormat dataFormat);
|
||||||
WXDLLIMPEXP_CORE int wxRegisterClipboardFormat(wxChar *formatName);
|
WXDLLIMPEXP_CORE int wxRegisterClipboardFormat(wxChar *formatName);
|
||||||
|
|
|
||||||
|
|
@ -34,7 +34,6 @@
|
||||||
#include "wx/intl.h"
|
#include "wx/intl.h"
|
||||||
#include "wx/log.h"
|
#include "wx/log.h"
|
||||||
#include "wx/dataobj.h"
|
#include "wx/dataobj.h"
|
||||||
#include "wx/dcmemory.h"
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#if wxUSE_METAFILE
|
#if wxUSE_METAFILE
|
||||||
|
|
@ -51,17 +50,7 @@
|
||||||
#include "wx/msw/dib.h"
|
#include "wx/msw/dib.h"
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#if wxUSE_OLE
|
#include <ole2.h>
|
||||||
// use OLE clipboard
|
|
||||||
#define wxUSE_OLE_CLIPBOARD 1
|
|
||||||
#else // !wxUSE_DATAOBJ
|
|
||||||
// use Win clipboard API
|
|
||||||
#define wxUSE_OLE_CLIPBOARD 0
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#if wxUSE_OLE_CLIPBOARD
|
|
||||||
#include <ole2.h>
|
|
||||||
#endif // wxUSE_OLE_CLIPBOARD
|
|
||||||
|
|
||||||
// ===========================================================================
|
// ===========================================================================
|
||||||
// implementation
|
// implementation
|
||||||
|
|
@ -71,10 +60,13 @@
|
||||||
// old-style clipboard functions using Windows API
|
// old-style clipboard functions using Windows API
|
||||||
// ---------------------------------------------------------------------------
|
// ---------------------------------------------------------------------------
|
||||||
|
|
||||||
static bool gs_wxClipboardIsOpen = false;
|
|
||||||
static int gs_htmlcfid = 0;
|
static int gs_htmlcfid = 0;
|
||||||
static int gs_pngcfid = 0;
|
static int gs_pngcfid = 0;
|
||||||
|
|
||||||
|
#ifdef WXWIN_COMPATIBILITY_3_2
|
||||||
|
|
||||||
|
static bool gs_wxClipboardIsOpen = false;
|
||||||
|
|
||||||
bool wxOpenClipboard()
|
bool wxOpenClipboard()
|
||||||
{
|
{
|
||||||
wxCHECK_MSG( !gs_wxClipboardIsOpen, true, wxT("clipboard already opened.") );
|
wxCHECK_MSG( !gs_wxClipboardIsOpen, true, wxT("clipboard already opened.") );
|
||||||
|
|
@ -132,6 +124,13 @@ bool wxIsClipboardOpened()
|
||||||
return gs_wxClipboardIsOpen;
|
return gs_wxClipboardIsOpen;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool wxClipboardOpen()
|
||||||
|
{
|
||||||
|
return gs_wxClipboardIsOpen;
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif // WXWIN_COMPATIBILITY_3_2
|
||||||
|
|
||||||
bool wxIsClipboardFormatAvailable(wxDataFormat dataFormat)
|
bool wxIsClipboardFormatAvailable(wxDataFormat dataFormat)
|
||||||
{
|
{
|
||||||
wxDataFormat::NativeFormat cf = dataFormat.GetFormatId();
|
wxDataFormat::NativeFormat cf = dataFormat.GetFormatId();
|
||||||
|
|
@ -163,294 +162,6 @@ bool wxIsClipboardFormatAvailable(wxDataFormat dataFormat)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
#if !wxUSE_OLE_CLIPBOARD
|
|
||||||
namespace
|
|
||||||
{
|
|
||||||
struct wxRawImageData
|
|
||||||
{
|
|
||||||
size_t m_size;
|
|
||||||
void* m_data;
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
bool wxSetClipboardData(wxDataFormat dataFormat,
|
|
||||||
const void *data,
|
|
||||||
int width, int height)
|
|
||||||
{
|
|
||||||
HANDLE handle = 0; // return value of SetClipboardData
|
|
||||||
|
|
||||||
switch (dataFormat)
|
|
||||||
{
|
|
||||||
case wxDF_BITMAP:
|
|
||||||
{
|
|
||||||
wxBitmap *bitmap = (wxBitmap *)data;
|
|
||||||
|
|
||||||
HDC hdcMem = CreateCompatibleDC(nullptr);
|
|
||||||
HDC hdcSrc = CreateCompatibleDC(nullptr);
|
|
||||||
HBITMAP old = (HBITMAP)
|
|
||||||
::SelectObject(hdcSrc, (HBITMAP)bitmap->GetHBITMAP());
|
|
||||||
HBITMAP hBitmap = CreateCompatibleBitmap(hdcSrc,
|
|
||||||
bitmap->GetWidth(),
|
|
||||||
bitmap->GetHeight());
|
|
||||||
if (!hBitmap)
|
|
||||||
{
|
|
||||||
SelectObject(hdcSrc, old);
|
|
||||||
DeleteDC(hdcMem);
|
|
||||||
DeleteDC(hdcSrc);
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
HBITMAP old1 = (HBITMAP) SelectObject(hdcMem, hBitmap);
|
|
||||||
BitBlt(hdcMem, 0, 0, bitmap->GetWidth(), bitmap->GetHeight(),
|
|
||||||
hdcSrc, 0, 0, SRCCOPY);
|
|
||||||
|
|
||||||
// Select new bitmap out of memory DC
|
|
||||||
SelectObject(hdcMem, old1);
|
|
||||||
|
|
||||||
// Set the data
|
|
||||||
handle = ::SetClipboardData(CF_BITMAP, hBitmap);
|
|
||||||
|
|
||||||
// Clean up
|
|
||||||
SelectObject(hdcSrc, old);
|
|
||||||
DeleteDC(hdcSrc);
|
|
||||||
DeleteDC(hdcMem);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
#if wxUSE_WXDIB
|
|
||||||
case wxDF_DIB:
|
|
||||||
{
|
|
||||||
wxBitmap *bitmap = (wxBitmap *)data;
|
|
||||||
|
|
||||||
if ( bitmap && bitmap->IsOk() )
|
|
||||||
{
|
|
||||||
wxDIB dib(*bitmap);
|
|
||||||
|
|
||||||
if ( dib.IsOk() )
|
|
||||||
{
|
|
||||||
DIBSECTION ds;
|
|
||||||
int n = ::GetObject(dib.GetHandle(), sizeof(DIBSECTION), &ds);
|
|
||||||
wxASSERT( n == sizeof(DIBSECTION) && ds.dsBm.bmBits );
|
|
||||||
// Number of colours in the palette.
|
|
||||||
int numColors;
|
|
||||||
switch ( ds.dsBmih.biCompression )
|
|
||||||
{
|
|
||||||
case BI_BITFIELDS:
|
|
||||||
numColors = 3;
|
|
||||||
break;
|
|
||||||
case BI_RGB:
|
|
||||||
case BI_RLE8:
|
|
||||||
case BI_RLE4:
|
|
||||||
numColors = ds.dsBmih.biClrUsed;
|
|
||||||
if ( !numColors )
|
|
||||||
{
|
|
||||||
numColors = ds.dsBmih.biBitCount <= 8 ? 1 << ds.dsBmih.biBitCount : 0;
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
numColors = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
unsigned long bmpSize = wxDIB::GetLineSize(ds.dsBmih.biWidth, ds.dsBmih.biBitCount) *
|
|
||||||
abs(ds.dsBmih.biHeight);
|
|
||||||
HANDLE hMem;
|
|
||||||
hMem = ::GlobalAlloc(GHND, ds.dsBmih.biSize + numColors*sizeof(RGBQUAD) + bmpSize);
|
|
||||||
if ( !hMem )
|
|
||||||
break;
|
|
||||||
|
|
||||||
{
|
|
||||||
GlobalPtrLock ptr(hMem);
|
|
||||||
char* pDst = (char*)ptr.Get();
|
|
||||||
memcpy(pDst, &ds.dsBmih, ds.dsBmih.biSize);
|
|
||||||
pDst += ds.dsBmih.biSize;
|
|
||||||
if ( numColors > 0 )
|
|
||||||
{
|
|
||||||
// Get colour table.
|
|
||||||
MemoryHDC hDC;
|
|
||||||
SelectInHDC sDC(hDC, dib.GetHandle());
|
|
||||||
::GetDIBColorTable(hDC, 0, numColors, (RGBQUAD*)pDst);
|
|
||||||
pDst += numColors*sizeof(RGBQUAD);
|
|
||||||
}
|
|
||||||
memcpy(pDst, dib.GetData(), bmpSize);
|
|
||||||
} // unlock hMem
|
|
||||||
|
|
||||||
handle = ::SetClipboardData(CF_DIB, hMem);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
|
|
||||||
// VZ: I'm told that this code works, but it doesn't seem to work for me
|
|
||||||
// and, anyhow, I'd be highly surprised if it did. So I leave it here
|
|
||||||
// but IMNSHO it is completely broken.
|
|
||||||
#if wxUSE_METAFILE && !defined(wxMETAFILE_IS_ENH)
|
|
||||||
case wxDF_METAFILE:
|
|
||||||
{
|
|
||||||
wxMetafile *wxMF = (wxMetafile *)data;
|
|
||||||
HANDLE data = GlobalAlloc(GHND, sizeof(METAFILEPICT) + 1);
|
|
||||||
{
|
|
||||||
GlobalPtrLock ptr(data);
|
|
||||||
METAFILEPICT *mf = (METAFILEPICT *)data.Get();
|
|
||||||
|
|
||||||
mf->mm = wxMF->GetWindowsMappingMode();
|
|
||||||
mf->xExt = width;
|
|
||||||
mf->yExt = height;
|
|
||||||
mf->hMF = (HMETAFILE) wxMF->GetHMETAFILE();
|
|
||||||
wxMF->SetHMETAFILE(nullptr);
|
|
||||||
} // unlock data
|
|
||||||
|
|
||||||
handle = SetClipboardData(CF_METAFILEPICT, data);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
#endif // wxUSE_METAFILE
|
|
||||||
|
|
||||||
#if wxUSE_ENH_METAFILE
|
|
||||||
case wxDF_ENHMETAFILE:
|
|
||||||
{
|
|
||||||
wxEnhMetaFile *emf = (wxEnhMetaFile *)data;
|
|
||||||
wxEnhMetaFile emfCopy = *emf;
|
|
||||||
|
|
||||||
handle = SetClipboardData(CF_ENHMETAFILE,
|
|
||||||
(void *)emfCopy.GetHENHMETAFILE());
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
#endif // wxUSE_ENH_METAFILE
|
|
||||||
|
|
||||||
case CF_SYLK:
|
|
||||||
case CF_DIF:
|
|
||||||
case CF_TIFF:
|
|
||||||
case CF_PALETTE:
|
|
||||||
default:
|
|
||||||
{
|
|
||||||
wxLogError(_("Unsupported clipboard format."));
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
case wxDF_OEMTEXT:
|
|
||||||
dataFormat = wxDF_TEXT;
|
|
||||||
// fall through
|
|
||||||
|
|
||||||
case wxDF_TEXT:
|
|
||||||
{
|
|
||||||
char *s = (char *)data;
|
|
||||||
|
|
||||||
width = strlen(s) + 1;
|
|
||||||
height = 1;
|
|
||||||
DWORD l = (width * height);
|
|
||||||
HANDLE hGlobalMemory = GlobalAlloc(GHND, l);
|
|
||||||
if ( hGlobalMemory )
|
|
||||||
{
|
|
||||||
memcpy(GlobalPtrLock(hGlobalMemory), s, l);
|
|
||||||
}
|
|
||||||
|
|
||||||
handle = SetClipboardData(dataFormat, hGlobalMemory);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
case wxDF_UNICODETEXT:
|
|
||||||
{
|
|
||||||
LPWSTR s = (LPWSTR)data;
|
|
||||||
DWORD size = sizeof(WCHAR) * (lstrlenW(s) + 1);
|
|
||||||
HANDLE hGlobalMemory = ::GlobalAlloc(GHND, size);
|
|
||||||
if ( hGlobalMemory )
|
|
||||||
{
|
|
||||||
memcpy(GlobalPtrLock(hGlobalMemory), s, size);
|
|
||||||
}
|
|
||||||
|
|
||||||
handle = ::SetClipboardData(CF_UNICODETEXT, hGlobalMemory);
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
|
|
||||||
case wxDF_HTML:
|
|
||||||
{
|
|
||||||
char* html = (char *)data;
|
|
||||||
|
|
||||||
// Create temporary buffer for HTML header...
|
|
||||||
char *buf = new char [400 + strlen(html)];
|
|
||||||
if(!buf) return false;
|
|
||||||
|
|
||||||
// Create a template string for the HTML header...
|
|
||||||
strcpy(buf,
|
|
||||||
"Version:0.9\r\n"
|
|
||||||
"StartHTML:00000000\r\n"
|
|
||||||
"EndHTML:00000000\r\n"
|
|
||||||
"StartFragment:00000000\r\n"
|
|
||||||
"EndFragment:00000000\r\n"
|
|
||||||
"<html><body>\r\n"
|
|
||||||
"<!--StartFragment -->\r\n");
|
|
||||||
|
|
||||||
// Append the HTML...
|
|
||||||
strcat(buf, html);
|
|
||||||
strcat(buf, "\r\n");
|
|
||||||
// Finish up the HTML format...
|
|
||||||
strcat(buf,
|
|
||||||
"<!--EndFragment-->\r\n"
|
|
||||||
"</body>\r\n"
|
|
||||||
"</html>");
|
|
||||||
|
|
||||||
// Now go back, calculate all the lengths, and write out the
|
|
||||||
// necessary header information. Note, wsprintf() truncates the
|
|
||||||
// string when you overwrite it so you follow up with code to replace
|
|
||||||
// the 0 appended at the end with a '\r'...
|
|
||||||
char *ptr = strstr(buf, "StartHTML");
|
|
||||||
sprintf(ptr+10, "%08u", (unsigned)(strstr(buf, "<html>") - buf));
|
|
||||||
*(ptr+10+8) = '\r';
|
|
||||||
|
|
||||||
ptr = strstr(buf, "EndHTML");
|
|
||||||
sprintf(ptr+8, "%08u", (unsigned)strlen(buf));
|
|
||||||
*(ptr+8+8) = '\r';
|
|
||||||
|
|
||||||
ptr = strstr(buf, "StartFragment");
|
|
||||||
sprintf(ptr+14, "%08u", (unsigned)(strstr(buf, "<!--StartFrag") - buf));
|
|
||||||
*(ptr+14+8) = '\r';
|
|
||||||
|
|
||||||
ptr = strstr(buf, "EndFragment");
|
|
||||||
sprintf(ptr+12, "%08u", (unsigned)(strstr(buf, "<!--EndFrag") - buf));
|
|
||||||
*(ptr+12+8) = '\r';
|
|
||||||
|
|
||||||
// Now you have everything in place ready to put on the
|
|
||||||
// clipboard.
|
|
||||||
|
|
||||||
// Allocate global memory for transfer...
|
|
||||||
HGLOBAL hText = GlobalAlloc(GMEM_MOVEABLE |GMEM_DDESHARE, strlen(buf)+4);
|
|
||||||
|
|
||||||
// Put your string in the global memory...
|
|
||||||
strcpy((char*)GlobalPtrLock(hText).Get(), buf);
|
|
||||||
|
|
||||||
handle = ::SetClipboardData(gs_htmlcfid, hText);
|
|
||||||
|
|
||||||
// Free memory...
|
|
||||||
GlobalFree(hText);
|
|
||||||
|
|
||||||
// Clean up...
|
|
||||||
delete [] buf;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
case wxDF_PNG:
|
|
||||||
{
|
|
||||||
const wxRawImageData* imgData = reinterpret_cast<const wxRawImageData*>(data);
|
|
||||||
|
|
||||||
GlobalPtr hImage(imgData->m_size, GMEM_MOVEABLE | GMEM_DDESHARE);
|
|
||||||
memcpy(GlobalPtrLock(hImage).Get(), imgData->m_data, imgData->m_size);
|
|
||||||
handle = ::SetClipboardData(gs_pngcfid, hImage);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if ( handle == 0 )
|
|
||||||
{
|
|
||||||
wxLogSysError(_("Failed to set clipboard data."));
|
|
||||||
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
#endif // !wxUSE_OLE_CLIPBOARD
|
|
||||||
|
|
||||||
wxDataFormat wxEnumClipboardFormats(wxDataFormat dataFormat)
|
wxDataFormat wxEnumClipboardFormats(wxDataFormat dataFormat)
|
||||||
{
|
{
|
||||||
return (wxDataFormat::NativeFormat)::EnumClipboardFormats(dataFormat);
|
return (wxDataFormat::NativeFormat)::EnumClipboardFormats(dataFormat);
|
||||||
|
|
@ -493,7 +204,6 @@ void wxClipboard::Clear()
|
||||||
if ( IsUsingPrimarySelection() )
|
if ( IsUsingPrimarySelection() )
|
||||||
return;
|
return;
|
||||||
|
|
||||||
#if wxUSE_OLE_CLIPBOARD
|
|
||||||
if (m_lastDataObject)
|
if (m_lastDataObject)
|
||||||
{
|
{
|
||||||
// don't touch data set by other applications
|
// don't touch data set by other applications
|
||||||
|
|
@ -508,12 +218,10 @@ void wxClipboard::Clear()
|
||||||
}
|
}
|
||||||
m_lastDataObject = nullptr;
|
m_lastDataObject = nullptr;
|
||||||
}
|
}
|
||||||
#endif // wxUSE_OLE_CLIPBOARD
|
|
||||||
}
|
}
|
||||||
|
|
||||||
bool wxClipboard::Flush()
|
bool wxClipboard::Flush()
|
||||||
{
|
{
|
||||||
#if wxUSE_OLE_CLIPBOARD
|
|
||||||
if (m_lastDataObject)
|
if (m_lastDataObject)
|
||||||
{
|
{
|
||||||
// don't touch data set by other applications
|
// don't touch data set by other applications
|
||||||
|
|
@ -532,9 +240,6 @@ bool wxClipboard::Flush()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return false;
|
return false;
|
||||||
#else // !wxUSE_OLE_CLIPBOARD
|
|
||||||
return false;
|
|
||||||
#endif // wxUSE_OLE_CLIPBOARD/!wxUSE_OLE_CLIPBOARD
|
|
||||||
}
|
}
|
||||||
|
|
||||||
bool wxClipboard::Open()
|
bool wxClipboard::Open()
|
||||||
|
|
@ -547,20 +252,13 @@ bool wxClipboard::Open()
|
||||||
|
|
||||||
// OLE opens clipboard for us
|
// OLE opens clipboard for us
|
||||||
m_isOpened = true;
|
m_isOpened = true;
|
||||||
#if wxUSE_OLE_CLIPBOARD
|
|
||||||
return true;
|
return true;
|
||||||
#else
|
|
||||||
return wxOpenClipboard();
|
|
||||||
#endif
|
|
||||||
}
|
}
|
||||||
|
|
||||||
bool wxClipboard::IsOpened() const
|
bool wxClipboard::IsOpened() const
|
||||||
{
|
{
|
||||||
#if wxUSE_OLE_CLIPBOARD
|
|
||||||
return m_isOpened;
|
return m_isOpened;
|
||||||
#else
|
|
||||||
return wxIsClipboardOpened();
|
|
||||||
#endif
|
|
||||||
}
|
}
|
||||||
|
|
||||||
bool wxClipboard::SetData( wxDataObject *data )
|
bool wxClipboard::SetData( wxDataObject *data )
|
||||||
|
|
@ -568,10 +266,6 @@ bool wxClipboard::SetData( wxDataObject *data )
|
||||||
if ( IsUsingPrimarySelection() )
|
if ( IsUsingPrimarySelection() )
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
#if !wxUSE_OLE_CLIPBOARD
|
|
||||||
(void)wxEmptyClipboard();
|
|
||||||
#endif // wxUSE_OLE_CLIPBOARD
|
|
||||||
|
|
||||||
if ( data )
|
if ( data )
|
||||||
return AddData(data);
|
return AddData(data);
|
||||||
else
|
else
|
||||||
|
|
@ -585,27 +279,6 @@ bool wxClipboard::AddData( wxDataObject *data )
|
||||||
|
|
||||||
wxCHECK_MSG( data, false, wxT("data is invalid") );
|
wxCHECK_MSG( data, false, wxT("data is invalid") );
|
||||||
|
|
||||||
const wxDataFormat format = data->GetPreferredFormat();
|
|
||||||
if ( format == wxDF_BITMAP || format == wxDF_DIB )
|
|
||||||
{
|
|
||||||
wxBitmapDataObject* bmpData = (wxBitmapDataObject*)data;
|
|
||||||
wxBitmap bmp = bmpData->GetBitmap();
|
|
||||||
wxASSERT_MSG( bmp.IsOk(), wxS("Invalid bitmap") );
|
|
||||||
// Replace 0RGB bitmap with its RGB copy
|
|
||||||
// to ensure compatibility with applications
|
|
||||||
// not recognizing bitmaps in 0RGB format.
|
|
||||||
if ( bmp.GetDepth() == 32 && !bmp.HasAlpha() )
|
|
||||||
{
|
|
||||||
wxBitmap bmpRGB(bmp.GetSize(), 24);
|
|
||||||
wxMemoryDC dc(bmpRGB);
|
|
||||||
dc.DrawBitmap(bmp, 0, 0);
|
|
||||||
dc.SelectObject(wxNullBitmap);
|
|
||||||
|
|
||||||
bmpData->SetBitmap(bmpRGB);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#if wxUSE_OLE_CLIPBOARD
|
|
||||||
HRESULT hr = OleSetClipboard(data->GetInterface());
|
HRESULT hr = OleSetClipboard(data->GetInterface());
|
||||||
if ( FAILED(hr) )
|
if ( FAILED(hr) )
|
||||||
{
|
{
|
||||||
|
|
@ -630,89 +303,11 @@ bool wxClipboard::AddData( wxDataObject *data )
|
||||||
data->SetAutoDelete();
|
data->SetAutoDelete();
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
#elif wxUSE_DATAOBJ
|
|
||||||
wxCHECK_MSG( wxIsClipboardOpened(), false, wxT("clipboard not open") );
|
|
||||||
|
|
||||||
bool bRet = false;
|
|
||||||
switch ( format )
|
|
||||||
{
|
|
||||||
case wxDF_TEXT:
|
|
||||||
case wxDF_OEMTEXT:
|
|
||||||
{
|
|
||||||
wxTextDataObject* textDataObject = (wxTextDataObject*) data;
|
|
||||||
wxString str(textDataObject->GetText());
|
|
||||||
bRet = wxSetClipboardData(format, str.c_str());
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
|
|
||||||
case wxDF_UNICODETEXT:
|
|
||||||
{
|
|
||||||
wxTextDataObject* textDataObject = (wxTextDataObject*)data;
|
|
||||||
wxString str(textDataObject->GetText());
|
|
||||||
bRet = wxSetClipboardData(format, str.wc_str());
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
|
|
||||||
case wxDF_BITMAP:
|
|
||||||
case wxDF_DIB:
|
|
||||||
{
|
|
||||||
wxBitmapDataObject* bitmapDataObject = (wxBitmapDataObject*) data;
|
|
||||||
wxBitmap bitmap(bitmapDataObject->GetBitmap());
|
|
||||||
bRet = wxSetClipboardData(format, &bitmap);
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
|
|
||||||
case wxDF_PNG:
|
|
||||||
{
|
|
||||||
wxCustomDataObject* imageDataObject = reinterpret_cast<wxCustomDataObject*>(data);
|
|
||||||
wxRawImageData imgData;
|
|
||||||
imgData.m_size = imageDataObject->GetDataSize();
|
|
||||||
imgData.m_data = imageDataObject->GetData();
|
|
||||||
bRet = wxSetClipboardData(format, &imgData);
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
|
|
||||||
#if wxUSE_METAFILE
|
|
||||||
case wxDF_METAFILE:
|
|
||||||
{
|
|
||||||
#if 1
|
|
||||||
// TODO
|
|
||||||
wxLogError(wxT("Not implemented because wxMetafileDataObject does not contain width and height values."));
|
|
||||||
#else
|
|
||||||
wxMetafileDataObject* metaFileDataObject =
|
|
||||||
(wxMetafileDataObject*) data;
|
|
||||||
wxMetafile metaFile = metaFileDataObject->GetMetafile();
|
|
||||||
bRet = wxSetClipboardData(wxDF_METAFILE, &metaFile,
|
|
||||||
metaFileDataObject->GetWidth(),
|
|
||||||
metaFileDataObject->GetHeight());
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
#endif // wxUSE_METAFILE
|
|
||||||
|
|
||||||
default:
|
|
||||||
{
|
|
||||||
// This didn't compile, of course
|
|
||||||
// return wxSetClipboardData(data);
|
|
||||||
// TODO
|
|
||||||
wxLogError(wxT("Not implemented."));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
// Delete owned, no longer necessary data.
|
|
||||||
delete data;
|
|
||||||
return bRet;
|
|
||||||
#else // !wxUSE_DATAOBJ
|
|
||||||
return false;
|
|
||||||
#endif // wxUSE_DATAOBJ/!wxUSE_DATAOBJ
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void wxClipboard::Close()
|
void wxClipboard::Close()
|
||||||
{
|
{
|
||||||
m_isOpened = false;
|
m_isOpened = false;
|
||||||
// OLE closes clipboard for us
|
|
||||||
#if !wxUSE_OLE_CLIPBOARD
|
|
||||||
wxCloseClipboard();
|
|
||||||
#endif
|
|
||||||
}
|
}
|
||||||
|
|
||||||
bool wxClipboard::IsSupported( const wxDataFormat& format )
|
bool wxClipboard::IsSupported( const wxDataFormat& format )
|
||||||
|
|
@ -725,7 +320,6 @@ bool wxClipboard::GetData( wxDataObject& data )
|
||||||
if ( IsUsingPrimarySelection() )
|
if ( IsUsingPrimarySelection() )
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
#if wxUSE_OLE_CLIPBOARD
|
|
||||||
IDataObject *pDataObject = nullptr;
|
IDataObject *pDataObject = nullptr;
|
||||||
HRESULT hr = OleGetClipboard(&pDataObject);
|
HRESULT hr = OleGetClipboard(&pDataObject);
|
||||||
if ( FAILED(hr) || !pDataObject )
|
if ( FAILED(hr) || !pDataObject )
|
||||||
|
|
@ -877,84 +471,6 @@ bool wxClipboard::GetData( wxDataObject& data )
|
||||||
pDataObject->Release();
|
pDataObject->Release();
|
||||||
|
|
||||||
return result;
|
return result;
|
||||||
#elif wxUSE_DATAOBJ
|
|
||||||
wxCHECK_MSG( wxIsClipboardOpened(), false, wxT("clipboard not open") );
|
|
||||||
|
|
||||||
wxDataFormat format = data.GetPreferredFormat();
|
|
||||||
switch ( format )
|
|
||||||
{
|
|
||||||
case wxDF_TEXT:
|
|
||||||
case wxDF_OEMTEXT:
|
|
||||||
case wxDF_UNICODETEXT:
|
|
||||||
{
|
|
||||||
// System provides an automatic type conversion
|
|
||||||
// from CF_TEXT and CF_OEMTEXT to CF_UNICODETEXT
|
|
||||||
HANDLE hMem = ::GetClipboardData(CF_UNICODETEXT);
|
|
||||||
if ( hMem )
|
|
||||||
{
|
|
||||||
wxTextDataObject& textDataObject = (wxTextDataObject &)data;
|
|
||||||
|
|
||||||
GlobalPtrLock ptr(hMem);
|
|
||||||
return textDataObject.SetData(ptr.GetSize(), ptr);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
case wxDF_BITMAP:
|
|
||||||
{
|
|
||||||
HANDLE hBmp = ::GetClipboardData(CF_BITMAP);
|
|
||||||
if ( hBmp )
|
|
||||||
{
|
|
||||||
wxBitmapDataObject2& bitmapDataObject = (wxBitmapDataObject2 &)data;
|
|
||||||
return bitmapDataObject.SetData(0, &hBmp);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
case wxDF_DIB:
|
|
||||||
{
|
|
||||||
HANDLE hMem = ::GetClipboardData(CF_DIB);
|
|
||||||
if ( hMem )
|
|
||||||
{
|
|
||||||
wxBitmapDataObject& bitmapDataObject = (wxBitmapDataObject &)data;
|
|
||||||
|
|
||||||
GlobalPtrLock ptr(hMem);
|
|
||||||
return bitmapDataObject.SetData(ptr.GetSize(), ptr);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
|
|
||||||
#if wxUSE_METAFILE && !defined(wxMETAFILE_IS_ENH)
|
|
||||||
case wxDF_METAFILE:
|
|
||||||
{
|
|
||||||
HANDLE hMem = ::GetClipboardData(CF_METAFILEPICT);
|
|
||||||
if ( hMem )
|
|
||||||
{
|
|
||||||
wxMetafileDataObject& metaFileDataObject = (wxMetafileDataObject &)data;
|
|
||||||
|
|
||||||
GlobalPtrLock ptr(hMem);
|
|
||||||
return metaFileDataObject.SetData(wxDF_METAFILE, ptr.GetSize(), ptr);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
#endif // wxUSE_METAFILE && !defined(wxMETAFILE_IS_ENH)
|
|
||||||
|
|
||||||
#if wxUSE_ENH_METAFILE
|
|
||||||
case wxDF_ENHMETAFILE:
|
|
||||||
{
|
|
||||||
HANDLE hFile = ::GetClipboardData(CF_ENHMETAFILE);
|
|
||||||
if ( hFile )
|
|
||||||
{
|
|
||||||
wxMetafileDataObject& metaFileDataObject = (wxMetafileDataObject &)data;
|
|
||||||
return metaFileDataObject.SetData(wxDF_ENHMETAFILE, 0, &hFile);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
#endif // wxUSE_ENH_METAFILE
|
|
||||||
}
|
|
||||||
return false;
|
|
||||||
#else // !wxUSE_DATAOBJ
|
|
||||||
wxFAIL_MSG( wxT("no clipboard implementation") );
|
|
||||||
return false;
|
|
||||||
#endif // wxUSE_OLE_CLIPBOARD/wxUSE_DATAOBJ
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#endif // wxUSE_CLIPBOARD
|
#endif // wxUSE_CLIPBOARD
|
||||||
|
|
|
||||||
|
|
@ -109,18 +109,7 @@ bool wxMetafile::SetClipboard(int width, int height)
|
||||||
if (!m_refData)
|
if (!m_refData)
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
bool alreadyOpen = wxClipboardOpen();
|
return wxTheClipboard->SetData(new wxMetafileDataObject(*this));
|
||||||
if (!alreadyOpen)
|
|
||||||
{
|
|
||||||
wxOpenClipboard();
|
|
||||||
if (!wxEmptyClipboard())
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
bool success = wxSetClipboardData(wxDF_METAFILE, this, width,height);
|
|
||||||
if (!alreadyOpen)
|
|
||||||
wxCloseClipboard();
|
|
||||||
|
|
||||||
return success;
|
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -23,6 +23,7 @@
|
||||||
#if wxUSE_DATAOBJ
|
#if wxUSE_DATAOBJ
|
||||||
|
|
||||||
#ifndef WX_PRECOMP
|
#ifndef WX_PRECOMP
|
||||||
|
#include "wx/dcmemory.h"
|
||||||
#include "wx/intl.h"
|
#include "wx/intl.h"
|
||||||
#include "wx/log.h"
|
#include "wx/log.h"
|
||||||
#include "wx/utils.h"
|
#include "wx/utils.h"
|
||||||
|
|
@ -1042,9 +1043,41 @@ const wxChar *wxDataObject::GetFormatName(wxDataFormat format)
|
||||||
// wxBitmapDataObject supports CF_DIB format
|
// wxBitmapDataObject supports CF_DIB format
|
||||||
// ----------------------------------------------------------------------------
|
// ----------------------------------------------------------------------------
|
||||||
|
|
||||||
|
namespace
|
||||||
|
{
|
||||||
|
|
||||||
|
// Modify bitmap if necessary, i.e. if it uses 0RGB format in which alpha
|
||||||
|
// channel is present but is entirely 0, to make it just plain RGB, i.e.
|
||||||
|
// without alpha channel at all, to ensure compatibility with the applications
|
||||||
|
// not recognizing the special case of 0RGB and handling such bitmaps as
|
||||||
|
// completely transparent, see #17640.
|
||||||
|
void RemoveAlphaIfNecessary(wxBitmap& bmp)
|
||||||
|
{
|
||||||
|
// Replace 0RGB bitmap with its RGB copy to ensure compatibility with
|
||||||
|
// applications not recognizing bitmaps in 0RGB format, see #17640.
|
||||||
|
if ( bmp.GetDepth() == 32 && !bmp.HasAlpha() )
|
||||||
|
{
|
||||||
|
wxBitmap bmpRGB(bmp.GetSize(), 24);
|
||||||
|
{
|
||||||
|
wxMemoryDC dc(bmpRGB);
|
||||||
|
dc.DrawBitmap(bmp, 0, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
bmp = bmpRGB;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
} // anonymous namespace
|
||||||
|
|
||||||
size_t wxBitmapDataObject::GetDataSize() const
|
size_t wxBitmapDataObject::GetDataSize() const
|
||||||
{
|
{
|
||||||
#if wxUSE_WXDIB
|
#if wxUSE_WXDIB
|
||||||
|
wxBitmap& bmp = const_cast<wxBitmapDataObject*>(this)->m_bitmap;
|
||||||
|
|
||||||
|
// Note that we need to do this here too and not just in GetDataHere()
|
||||||
|
// because the size of the bitmap without the alpha channel is different.
|
||||||
|
RemoveAlphaIfNecessary(bmp);
|
||||||
|
|
||||||
return wxDIB::ConvertFromBitmap(nullptr, GetHbitmapOf(GetBitmap()));
|
return wxDIB::ConvertFromBitmap(nullptr, GetHbitmapOf(GetBitmap()));
|
||||||
#else
|
#else
|
||||||
return 0;
|
return 0;
|
||||||
|
|
@ -1054,9 +1087,13 @@ size_t wxBitmapDataObject::GetDataSize() const
|
||||||
bool wxBitmapDataObject::GetDataHere(void *buf) const
|
bool wxBitmapDataObject::GetDataHere(void *buf) const
|
||||||
{
|
{
|
||||||
#if wxUSE_WXDIB
|
#if wxUSE_WXDIB
|
||||||
|
wxBitmap& bmp = const_cast<wxBitmapDataObject*>(this)->m_bitmap;
|
||||||
|
|
||||||
|
RemoveAlphaIfNecessary(bmp);
|
||||||
|
|
||||||
BITMAPINFO * const pbi = (BITMAPINFO *)buf;
|
BITMAPINFO * const pbi = (BITMAPINFO *)buf;
|
||||||
|
|
||||||
return wxDIB::ConvertFromBitmap(pbi, GetHbitmapOf(GetBitmap())) != 0;
|
return wxDIB::ConvertFromBitmap(pbi, GetHbitmapOf(bmp)) != 0;
|
||||||
#else
|
#else
|
||||||
wxUnusedVar(buf);
|
wxUnusedVar(buf);
|
||||||
return false;
|
return false;
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue