gtkradiant/tools/urt/libs/gtkutil/clipboard.cpp

126 lines
3.9 KiB
C++
Raw Normal View History

#include "clipboard.h"
#include "stream/memstream.h"
#include "stream/textstream.h"
/// \file
/// \brief Platform-independent GTK clipboard support.
/// \todo Using GDK_SELECTION_CLIPBOARD fails on win32, so we use the win32 API directly for now.
#if defined ( __linux__ ) || defined ( __APPLE__ )
#include <gtk/gtkclipboard.h>
enum
{
RADIANT_CLIPPINGS = 23,
};
static const GtkTargetEntry clipboard_targets[] = {
{ "RADIANT_CLIPPINGS", 0, RADIANT_CLIPPINGS, },
};
static void clipboard_get( GtkClipboard *clipboard, GtkSelectionData *selection_data, guint info, gpointer data ){
std::size_t len = *reinterpret_cast<std::size_t*>( data );
const char* buffer = ( len != 0 ) ? reinterpret_cast<const char*>( data ) + sizeof( std::size_t ) : 0;
GdkAtom type = GDK_NONE;
if ( info == clipboard_targets[0].info ) {
type = gdk_atom_intern( clipboard_targets[0].target, FALSE );
}
gtk_selection_data_set( selection_data, type, 8, reinterpret_cast<const guchar*>( buffer ), static_cast<gint>( len ) );
}
static void clipboard_clear( GtkClipboard *clipboard, gpointer data ){
delete [] reinterpret_cast<const char*>( data );
}
static void clipboard_received( GtkClipboard *clipboard, GtkSelectionData *data, gpointer user_data ){
if ( data->length < 0 ) {
globalErrorStream() << "Error retrieving selection\n";
}
else if ( strcmp( gdk_atom_name( data->type ), clipboard_targets[0].target ) == 0 ) {
BufferInputStream istream( reinterpret_cast<const char*>( data->data ), data->length );
( *reinterpret_cast<ClipboardPasteFunc*>( user_data ) )( istream );
}
}
void clipboard_copy( ClipboardCopyFunc copy ){
GtkClipboard* clipboard = gtk_clipboard_get( GDK_SELECTION_CLIPBOARD );
BufferOutputStream ostream;
copy( ostream );
std::size_t length = ostream.size();
char* data = new char[length + sizeof( std::size_t )];
*reinterpret_cast<std::size_t*>( data ) = length;
memcpy( data + sizeof( std::size_t ), ostream.data(), length );
gtk_clipboard_set_with_data( clipboard, clipboard_targets, 1, clipboard_get, clipboard_clear, data );
}
ClipboardPasteFunc g_clipboardPasteFunc = 0;
void clipboard_paste( ClipboardPasteFunc paste ){
GtkClipboard* clipboard = gtk_clipboard_get( GDK_SELECTION_CLIPBOARD );
g_clipboardPasteFunc = paste;
gtk_clipboard_request_contents( clipboard, gdk_atom_intern( clipboard_targets[0].target, FALSE ), clipboard_received, &g_clipboardPasteFunc );
}
#elif defined( WIN32 )
const char* c_clipboard_format = "RadiantClippings";
#include <windows.h>
void clipboard_copy( ClipboardCopyFunc copy ){
BufferOutputStream ostream;
copy( ostream );
bool bClipped = false;
UINT nClipboard = ::RegisterClipboardFormat( c_clipboard_format );
if ( nClipboard > 0 ) {
if ( ::OpenClipboard( 0 ) ) {
EmptyClipboard();
std::size_t length = ostream.size();
HANDLE h = ::GlobalAlloc( GMEM_ZEROINIT | GMEM_MOVEABLE | GMEM_DDESHARE, length + sizeof( std::size_t ) );
if ( h != 0 ) {
char *buffer = reinterpret_cast<char*>( ::GlobalLock( h ) );
*reinterpret_cast<std::size_t*>( buffer ) = length;
buffer += sizeof( std::size_t );
memcpy( buffer, ostream.data(), length );
::GlobalUnlock( h );
::SetClipboardData( nClipboard, h );
::CloseClipboard();
bClipped = true;
}
}
}
if ( !bClipped ) {
globalOutputStream() << "Unable to register Windows clipboard formats, copy/paste between editors will not be possible\n";
}
}
void clipboard_paste( ClipboardPasteFunc paste ){
UINT nClipboard = ::RegisterClipboardFormat( c_clipboard_format );
if ( nClipboard > 0 && ::OpenClipboard( 0 ) ) {
if ( IsClipboardFormatAvailable( nClipboard ) ) {
HANDLE h = ::GetClipboardData( nClipboard );
if ( h ) {
const char *buffer = reinterpret_cast<const char*>( ::GlobalLock( h ) );
std::size_t length = *reinterpret_cast<const std::size_t*>( buffer );
buffer += sizeof( std::size_t );
BufferInputStream istream( buffer, length );
paste( istream );
::GlobalUnlock( h );
}
}
::CloseClipboard();
}
}
#endif