simplified the code by using new wxClipboardSync class abstracting wait for clipboard results and wxScopeGuard to ensure that the 'waiting' flag is reset in every case
git-svn-id: https://svn.wxwidgets.org/svn/wx/wxWidgets/trunk@45180 c3d73ce0-8a6f-49c7-b76d-6d57e0e08775
This commit is contained in:
@@ -49,7 +49,10 @@ public:
|
|||||||
virtual void UsePrimarySelection(bool primary = TRUE)
|
virtual void UsePrimarySelection(bool primary = TRUE)
|
||||||
{ m_usePrimary = primary; }
|
{ m_usePrimary = primary; }
|
||||||
|
|
||||||
|
|
||||||
// implementation from now on
|
// implementation from now on
|
||||||
|
// --------------------------
|
||||||
|
|
||||||
bool m_open;
|
bool m_open;
|
||||||
bool m_ownsClipboard;
|
bool m_ownsClipboard;
|
||||||
bool m_ownsPrimarySelection;
|
bool m_ownsPrimarySelection;
|
||||||
@@ -57,7 +60,6 @@ public:
|
|||||||
|
|
||||||
GtkWidget *m_clipboardWidget; /* for getting and offering data */
|
GtkWidget *m_clipboardWidget; /* for getting and offering data */
|
||||||
GtkWidget *m_targetsWidget; /* for getting list of supported formats */
|
GtkWidget *m_targetsWidget; /* for getting list of supported formats */
|
||||||
bool m_waiting; /* querying data or formats is asynchronous */
|
|
||||||
|
|
||||||
bool m_formatSupported;
|
bool m_formatSupported;
|
||||||
GdkAtom m_targetRequested;
|
GdkAtom m_targetRequested;
|
||||||
|
@@ -1,12 +1,20 @@
|
|||||||
/////////////////////////////////////////////////////////////////////////////
|
/////////////////////////////////////////////////////////////////////////////
|
||||||
// Name: src/gtk/clipbrd.cpp
|
// Name: src/gtk/clipbrd.cpp
|
||||||
// Purpose:
|
// Purpose: wxClipboard implementation for wxGTK
|
||||||
// Author: Robert Roebling
|
// Author: Robert Roebling
|
||||||
// Id: $Id$
|
// Id: $Id$
|
||||||
// Copyright: (c) 1998 Robert Roebling
|
// Copyright: (c) 1998 Robert Roebling
|
||||||
// Licence: wxWindows licence
|
// Licence: wxWindows licence
|
||||||
/////////////////////////////////////////////////////////////////////////////
|
/////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
|
// ============================================================================
|
||||||
|
// declarations
|
||||||
|
// ============================================================================
|
||||||
|
|
||||||
|
// ----------------------------------------------------------------------------
|
||||||
|
// headers
|
||||||
|
// ----------------------------------------------------------------------------
|
||||||
|
|
||||||
// For compilers that support precompilation, includes "wx.h".
|
// For compilers that support precompilation, includes "wx.h".
|
||||||
#include "wx/wxprec.h"
|
#include "wx/wxprec.h"
|
||||||
|
|
||||||
@@ -20,15 +28,17 @@
|
|||||||
#include "wx/dataobj.h"
|
#include "wx/dataobj.h"
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
#include "wx/scopeguard.h"
|
||||||
|
|
||||||
#include "wx/gtk/private.h"
|
#include "wx/gtk/private.h"
|
||||||
|
|
||||||
//-----------------------------------------------------------------------------
|
//-----------------------------------------------------------------------------
|
||||||
// data
|
// data
|
||||||
//-----------------------------------------------------------------------------
|
//-----------------------------------------------------------------------------
|
||||||
|
|
||||||
GdkAtom g_clipboardAtom = 0;
|
static GdkAtom g_clipboardAtom = 0;
|
||||||
GdkAtom g_targetsAtom = 0;
|
static GdkAtom g_targetsAtom = 0;
|
||||||
GdkAtom g_timestampAtom = 0;
|
static GdkAtom g_timestampAtom = 0;
|
||||||
|
|
||||||
#if wxUSE_UNICODE
|
#if wxUSE_UNICODE
|
||||||
extern GdkAtom g_altTextAtom;
|
extern GdkAtom g_altTextAtom;
|
||||||
@@ -39,30 +49,47 @@ extern GdkAtom g_altTextAtom;
|
|||||||
// (there will be a *lot* of them!)
|
// (there will be a *lot* of them!)
|
||||||
#define TRACE_CLIPBOARD _T("clipboard")
|
#define TRACE_CLIPBOARD _T("clipboard")
|
||||||
|
|
||||||
//-----------------------------------------------------------------------------
|
// ----------------------------------------------------------------------------
|
||||||
// reminder
|
// wxClipboardSync: used to perform clipboard operations synchronously
|
||||||
//-----------------------------------------------------------------------------
|
// ----------------------------------------------------------------------------
|
||||||
|
|
||||||
/* The contents of a selection are returned in a GtkSelectionData
|
// constructing this object on stack will wait wait until the latest clipboard
|
||||||
structure. selection/target identify the request.
|
// operation is finished on block exit
|
||||||
type specifies the type of the return; if length < 0, and
|
//
|
||||||
the data should be ignored. This structure has object semantics -
|
// notice that there can be no more than one such object alive at any moment,
|
||||||
no fields should be modified directly, they should not be created
|
// i.e. reentrancies are not allowed
|
||||||
directly, and pointers to them should not be stored beyond the duration of
|
class wxClipboardSync
|
||||||
a callback. (If the last is changed, we'll need to add reference
|
|
||||||
counting)
|
|
||||||
|
|
||||||
struct _GtkSelectionData
|
|
||||||
{
|
{
|
||||||
GdkAtom selection;
|
public:
|
||||||
GdkAtom target;
|
wxClipboardSync(wxClipboard& clipboard)
|
||||||
GdkAtom type;
|
{
|
||||||
gint format;
|
wxASSERT_MSG( !ms_clipboard, _T("reentrancy in clipboard code") );
|
||||||
guchar *data;
|
ms_clipboard = &clipboard;
|
||||||
gint length;
|
}
|
||||||
|
|
||||||
|
~wxClipboardSync()
|
||||||
|
{
|
||||||
|
while ( ms_clipboard )
|
||||||
|
gtk_main_iteration();
|
||||||
|
}
|
||||||
|
|
||||||
|
// this method must be called by GTK+ callbacks to indicate that we got the
|
||||||
|
// result for our clipboard operation
|
||||||
|
static void OnDone(wxClipboard *clipboard)
|
||||||
|
{
|
||||||
|
wxASSERT_MSG( clipboard == ms_clipboard,
|
||||||
|
_T("got notification for alien clipboard") );
|
||||||
|
|
||||||
|
ms_clipboard = NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
static wxClipboard *ms_clipboard;
|
||||||
|
|
||||||
|
DECLARE_NO_COPY_CLASS(wxClipboardSync)
|
||||||
};
|
};
|
||||||
|
|
||||||
*/
|
wxClipboard *wxClipboardSync::ms_clipboard = NULL;
|
||||||
|
|
||||||
//-----------------------------------------------------------------------------
|
//-----------------------------------------------------------------------------
|
||||||
// "selection_received" for targets
|
// "selection_received" for targets
|
||||||
@@ -75,6 +102,8 @@ targets_selection_received( GtkWidget *WXUNUSED(widget),
|
|||||||
guint32 WXUNUSED(time),
|
guint32 WXUNUSED(time),
|
||||||
wxClipboard *clipboard )
|
wxClipboard *clipboard )
|
||||||
{
|
{
|
||||||
|
wxON_BLOCK_EXIT1(wxClipboardSync::OnDone, clipboard);
|
||||||
|
|
||||||
if ( wxTheClipboard && selection_data->length > 0 )
|
if ( wxTheClipboard && selection_data->length > 0 )
|
||||||
{
|
{
|
||||||
// make sure we got the data in the correct form
|
// make sure we got the data in the correct form
|
||||||
@@ -86,7 +115,6 @@ targets_selection_received( GtkWidget *WXUNUSED(widget),
|
|||||||
wxLogTrace( TRACE_CLIPBOARD,
|
wxLogTrace( TRACE_CLIPBOARD,
|
||||||
_T("got unsupported clipboard target") );
|
_T("got unsupported clipboard target") );
|
||||||
|
|
||||||
clipboard->m_waiting = false;
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -115,14 +143,11 @@ targets_selection_received( GtkWidget *WXUNUSED(widget),
|
|||||||
|
|
||||||
if (format == clipboard->m_targetRequested)
|
if (format == clipboard->m_targetRequested)
|
||||||
{
|
{
|
||||||
clipboard->m_waiting = false;
|
|
||||||
clipboard->m_formatSupported = true;
|
clipboard->m_formatSupported = true;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
clipboard->m_waiting = false;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -137,50 +162,28 @@ selection_received( GtkWidget *WXUNUSED(widget),
|
|||||||
guint32 WXUNUSED(time),
|
guint32 WXUNUSED(time),
|
||||||
wxClipboard *clipboard )
|
wxClipboard *clipboard )
|
||||||
{
|
{
|
||||||
|
wxON_BLOCK_EXIT1(wxClipboardSync::OnDone, clipboard);
|
||||||
|
|
||||||
if (!wxTheClipboard)
|
if (!wxTheClipboard)
|
||||||
{
|
|
||||||
clipboard->m_waiting = false;
|
|
||||||
return;
|
return;
|
||||||
}
|
|
||||||
|
|
||||||
wxDataObject *data_object = clipboard->m_receivedData;
|
wxDataObject *data_object = clipboard->m_receivedData;
|
||||||
|
|
||||||
if (!data_object)
|
if (!data_object)
|
||||||
{
|
|
||||||
clipboard->m_waiting = false;
|
|
||||||
return;
|
return;
|
||||||
}
|
|
||||||
|
|
||||||
if (selection_data->length <= 0)
|
if (selection_data->length <= 0)
|
||||||
{
|
|
||||||
clipboard->m_waiting = false;
|
|
||||||
return;
|
return;
|
||||||
}
|
|
||||||
|
|
||||||
wxDataFormat format( selection_data->target );
|
wxDataFormat format( selection_data->target );
|
||||||
|
|
||||||
// make sure we got the data in the correct format
|
// make sure we got the data in the correct format
|
||||||
if (!data_object->IsSupportedFormat( format ) )
|
if (!data_object->IsSupportedFormat( format ) )
|
||||||
{
|
|
||||||
clipboard->m_waiting = false;
|
|
||||||
return;
|
return;
|
||||||
}
|
|
||||||
|
|
||||||
#if 0
|
|
||||||
This seems to cause problems somehow
|
|
||||||
// make sure we got the data in the correct form (selection type).
|
|
||||||
// if so, copy data to target object
|
|
||||||
if (selection_data->type != GDK_SELECTION_TYPE_STRING)
|
|
||||||
{
|
|
||||||
clipboard->m_waiting = false;
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
|
|
||||||
data_object->SetData( format, (size_t) selection_data->length, (const char*) selection_data->data );
|
data_object->SetData( format, (size_t) selection_data->length, (const char*) selection_data->data );
|
||||||
|
|
||||||
wxTheClipboard->m_formatSupported = true;
|
wxTheClipboard->m_formatSupported = true;
|
||||||
clipboard->m_waiting = false;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -192,6 +195,8 @@ extern "C" {
|
|||||||
static gint
|
static gint
|
||||||
selection_clear_clip( GtkWidget *WXUNUSED(widget), GdkEventSelection *event )
|
selection_clear_clip( GtkWidget *WXUNUSED(widget), GdkEventSelection *event )
|
||||||
{
|
{
|
||||||
|
wxON_BLOCK_EXIT1(wxClipboardSync::OnDone, wxTheClipboard);
|
||||||
|
|
||||||
if (!wxTheClipboard) return true;
|
if (!wxTheClipboard) return true;
|
||||||
|
|
||||||
if (event->selection == GDK_SELECTION_PRIMARY)
|
if (event->selection == GDK_SELECTION_PRIMARY)
|
||||||
@@ -205,7 +210,6 @@ selection_clear_clip( GtkWidget *WXUNUSED(widget), GdkEventSelection *event )
|
|||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
wxTheClipboard->m_waiting = false;
|
|
||||||
return FALSE;
|
return FALSE;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -222,7 +226,6 @@ selection_clear_clip( GtkWidget *WXUNUSED(widget), GdkEventSelection *event )
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
wxTheClipboard->m_waiting = false;
|
|
||||||
return TRUE;
|
return TRUE;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -320,7 +323,6 @@ IMPLEMENT_DYNAMIC_CLASS(wxClipboard,wxObject)
|
|||||||
wxClipboard::wxClipboard()
|
wxClipboard::wxClipboard()
|
||||||
{
|
{
|
||||||
m_open = false;
|
m_open = false;
|
||||||
m_waiting = false;
|
|
||||||
|
|
||||||
m_ownsClipboard = false;
|
m_ownsClipboard = false;
|
||||||
m_ownsPrimarySelection = false;
|
m_ownsPrimarySelection = false;
|
||||||
@@ -369,38 +371,26 @@ void wxClipboard::Clear()
|
|||||||
{
|
{
|
||||||
if (m_data)
|
if (m_data)
|
||||||
{
|
{
|
||||||
#if wxUSE_THREADS
|
|
||||||
/* disable GUI threads */
|
|
||||||
#endif
|
|
||||||
|
|
||||||
// As we have data we also own the clipboard. Once we no longer own
|
// As we have data we also own the clipboard. Once we no longer own
|
||||||
// it, clear_selection is called which will set m_data to zero
|
// it, clear_selection is called which will set m_data to zero
|
||||||
if (gdk_selection_owner_get( g_clipboardAtom ) == m_clipboardWidget->window)
|
if (gdk_selection_owner_get( g_clipboardAtom ) == m_clipboardWidget->window)
|
||||||
{
|
{
|
||||||
m_waiting = true;
|
wxClipboardSync sync(*this);
|
||||||
|
|
||||||
gtk_selection_owner_set( (GtkWidget*) NULL, g_clipboardAtom,
|
gtk_selection_owner_set( (GtkWidget*) NULL, g_clipboardAtom,
|
||||||
(guint32) GDK_CURRENT_TIME );
|
(guint32) GDK_CURRENT_TIME );
|
||||||
|
|
||||||
while (m_waiting) gtk_main_iteration();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (gdk_selection_owner_get( GDK_SELECTION_PRIMARY ) == m_clipboardWidget->window)
|
if (gdk_selection_owner_get( GDK_SELECTION_PRIMARY ) == m_clipboardWidget->window)
|
||||||
{
|
{
|
||||||
m_waiting = true;
|
wxClipboardSync sync(*this);
|
||||||
|
|
||||||
gtk_selection_owner_set( (GtkWidget*) NULL, GDK_SELECTION_PRIMARY,
|
gtk_selection_owner_set( (GtkWidget*) NULL, GDK_SELECTION_PRIMARY,
|
||||||
(guint32) GDK_CURRENT_TIME );
|
(guint32) GDK_CURRENT_TIME );
|
||||||
|
|
||||||
while (m_waiting) gtk_main_iteration();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
delete m_data;
|
delete m_data;
|
||||||
m_data = NULL;
|
m_data = NULL;
|
||||||
|
|
||||||
#if wxUSE_THREADS
|
|
||||||
/* re-enable GUI threads */
|
|
||||||
#endif
|
|
||||||
}
|
}
|
||||||
|
|
||||||
m_targetRequested = 0;
|
m_targetRequested = 0;
|
||||||
@@ -508,9 +498,6 @@ bool wxClipboard::IsOpened() const
|
|||||||
|
|
||||||
bool wxClipboard::IsSupported( const wxDataFormat& format )
|
bool wxClipboard::IsSupported( const wxDataFormat& format )
|
||||||
{
|
{
|
||||||
/* reentrance problems */
|
|
||||||
if (m_waiting) return false;
|
|
||||||
|
|
||||||
/* store requested format to be asked for by callbacks */
|
/* store requested format to be asked for by callbacks */
|
||||||
m_targetRequested = format;
|
m_targetRequested = format;
|
||||||
|
|
||||||
@@ -522,23 +509,17 @@ bool wxClipboard::IsSupported( const wxDataFormat& format )
|
|||||||
|
|
||||||
m_formatSupported = false;
|
m_formatSupported = false;
|
||||||
|
|
||||||
/* perform query. this will set m_formatSupported to
|
// block until m_formatSupported is set from targets_selection_received
|
||||||
true if m_targetRequested is supported.
|
// callback
|
||||||
also, we have to wait for the "answer" from the
|
{
|
||||||
clipboard owner which is an asynchronous process.
|
wxClipboardSync sync(*this);
|
||||||
therefore we set m_waiting = true here and wait
|
|
||||||
until the callback "targets_selection_received"
|
|
||||||
sets it to false */
|
|
||||||
|
|
||||||
m_waiting = true;
|
gtk_selection_convert( m_targetsWidget,
|
||||||
|
m_usePrimary ? (GdkAtom)GDK_SELECTION_PRIMARY
|
||||||
gtk_selection_convert( m_targetsWidget,
|
: g_clipboardAtom,
|
||||||
m_usePrimary ? (GdkAtom)GDK_SELECTION_PRIMARY
|
g_targetsAtom,
|
||||||
: g_clipboardAtom,
|
(guint32) GDK_CURRENT_TIME );
|
||||||
g_targetsAtom,
|
} // wait until we get the results
|
||||||
(guint32) GDK_CURRENT_TIME );
|
|
||||||
|
|
||||||
while (m_waiting) gtk_main_iteration();
|
|
||||||
|
|
||||||
#if wxUSE_UNICODE
|
#if wxUSE_UNICODE
|
||||||
if (!m_formatSupported && format == wxDataFormat(wxDF_UNICODETEXT))
|
if (!m_formatSupported && format == wxDataFormat(wxDF_UNICODETEXT))
|
||||||
@@ -577,23 +558,16 @@ bool wxClipboard::GetData( wxDataObject& data )
|
|||||||
|
|
||||||
m_formatSupported = false;
|
m_formatSupported = false;
|
||||||
|
|
||||||
/* perform query. this will set m_formatSupported to
|
// block until m_formatSupported is set by targets_selection_received
|
||||||
true if m_targetRequested is supported.
|
{
|
||||||
also, we have to wait for the "answer" from the
|
wxClipboardSync sync(*this);
|
||||||
clipboard owner which is an asynchronous process.
|
|
||||||
therefore we set m_waiting = true here and wait
|
|
||||||
until the callback "targets_selection_received"
|
|
||||||
sets it to false */
|
|
||||||
|
|
||||||
m_waiting = true;
|
gtk_selection_convert( m_targetsWidget,
|
||||||
|
m_usePrimary ? (GdkAtom)GDK_SELECTION_PRIMARY
|
||||||
gtk_selection_convert( m_targetsWidget,
|
: g_clipboardAtom,
|
||||||
m_usePrimary ? (GdkAtom)GDK_SELECTION_PRIMARY
|
g_targetsAtom,
|
||||||
: g_clipboardAtom,
|
(guint32) GDK_CURRENT_TIME );
|
||||||
g_targetsAtom,
|
} // wait until we get the results
|
||||||
(guint32) GDK_CURRENT_TIME );
|
|
||||||
|
|
||||||
while (m_waiting) gtk_main_iteration();
|
|
||||||
|
|
||||||
if (!m_formatSupported) continue;
|
if (!m_formatSupported) continue;
|
||||||
|
|
||||||
@@ -608,27 +582,18 @@ bool wxClipboard::GetData( wxDataObject& data )
|
|||||||
/* start query */
|
/* start query */
|
||||||
m_formatSupported = false;
|
m_formatSupported = false;
|
||||||
|
|
||||||
/* ask for clipboard contents. this will set
|
|
||||||
m_formatSupported to true if m_targetRequested
|
|
||||||
is supported.
|
|
||||||
also, we have to wait for the "answer" from the
|
|
||||||
clipboard owner which is an asynchronous process.
|
|
||||||
therefore we set m_waiting = true here and wait
|
|
||||||
until the callback "targets_selection_received"
|
|
||||||
sets it to false */
|
|
||||||
|
|
||||||
m_waiting = true;
|
|
||||||
|
|
||||||
wxLogTrace( TRACE_CLIPBOARD,
|
wxLogTrace( TRACE_CLIPBOARD,
|
||||||
wxT("wxClipboard::GetData: format found, start convert") );
|
wxT("wxClipboard::GetData: format found, start convert") );
|
||||||
|
|
||||||
gtk_selection_convert( m_clipboardWidget,
|
{
|
||||||
m_usePrimary ? (GdkAtom)GDK_SELECTION_PRIMARY
|
wxClipboardSync sync(*this);
|
||||||
: g_clipboardAtom,
|
|
||||||
m_targetRequested,
|
|
||||||
(guint32) GDK_CURRENT_TIME );
|
|
||||||
|
|
||||||
while (m_waiting) gtk_main_iteration();
|
gtk_selection_convert( m_clipboardWidget,
|
||||||
|
m_usePrimary ? (GdkAtom)GDK_SELECTION_PRIMARY
|
||||||
|
: g_clipboardAtom,
|
||||||
|
m_targetRequested,
|
||||||
|
(guint32) GDK_CURRENT_TIME );
|
||||||
|
} // wait until we get the results
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Normally this is a true error as we checked for the presence of such
|
Normally this is a true error as we checked for the presence of such
|
||||||
|
Reference in New Issue
Block a user