]> git.saurik.com Git - wxWidgets.git/blame - src/motif/clipbrd.cpp
Fix horizontal mouse wheel scrolling in wxGTK.
[wxWidgets.git] / src / motif / clipbrd.cpp
CommitLineData
4bb6408c 1/////////////////////////////////////////////////////////////////////////////
55034339 2// Name: src/motif/clipbrd.cpp
4bb6408c
JS
3// Purpose: Clipboard functionality
4// Author: Julian Smart
dd38c875 5// Modified by: Mattia Barbon (added support for generic wxDataObjects)
4bb6408c 6// Created: 17/09/98
4bb6408c 7// Copyright: (c) Julian Smart
55034339 8// Licence: wxWindows licence
4bb6408c
JS
9/////////////////////////////////////////////////////////////////////////////
10
1248b41f
MB
11// For compilers that support precompilation, includes "wx.h".
12#include "wx/wxprec.h"
13
c698eae5
JJ
14#ifdef __VMS
15#include "wx/vms_x_fix.h"
16#define XtWindow XTWINDOW
17#define XtScreen XTSCREEN
18#define XtParent XTPARENT
19#define XtIsRealized XTISREALIZED
20#define XtDisplay XTDISPLAY
21#endif
22
dfe1eee3
VZ
23#if wxUSE_CLIPBOARD
24
670f9935
WS
25#include "wx/clipbrd.h"
26
27#ifndef WX_PRECOMP
28 #include "wx/app.h"
de6185e2 29 #include "wx/utils.h"
0bca0373 30 #include "wx/bitmap.h"
28f92d74 31 #include "wx/dataobj.h"
670f9935
WS
32#endif
33
664e1314
VZ
34#include "wx/scopedarray.h"
35
36typedef wxScopedArray<wxDataFormat> wxDataFormatScopedArray;
2d120f83 37
338dd992
JJ
38#ifdef __VMS__
39#pragma message disable nosimpint
c698eae5 40
338dd992 41#endif
2d120f83
JS
42#include <Xm/Xm.h>
43#include <Xm/CutPaste.h>
338dd992
JJ
44#ifdef __VMS__
45#pragma message enable nosimpint
46#endif
4bb6408c 47
dd38c875 48#include "wx/motif/private.h"
2d120f83 49
4bb6408c
JS
50bool wxOpenClipboard()
51{
dd38c875 52 return wxTheClipboard->Open();
4bb6408c
JS
53}
54
55bool wxCloseClipboard()
56{
dd38c875
MB
57 wxTheClipboard->Close();
58
59 return true;
4bb6408c
JS
60}
61
62bool wxEmptyClipboard()
63{
dd38c875
MB
64 wxTheClipboard->Clear();
65 return true;
4bb6408c
JS
66}
67
68bool wxClipboardOpen()
69{
dd38c875 70 return wxTheClipboard->IsOpened();
4bb6408c
JS
71}
72
fbfb8bcc 73bool wxIsClipboardFormatAvailable(const wxDataFormat& dataFormat)
4bb6408c 74{
dd38c875
MB
75 return wxTheClipboard->IsSupported( dataFormat );
76}
2d120f83 77
dd38c875
MB
78bool wxSetClipboardData(wxDataFormat dataFormat, wxObject *obj,
79 int WXUNUSED(width), int WXUNUSED(height))
80{
81 wxDataObject* dobj = NULL;
2d120f83 82
dd38c875
MB
83 if( dataFormat == wxDF_TEXT )
84 {
85 wxChar* data = (wxChar*)obj;
86 dobj = new wxTextDataObject( data );
87 }
88 else if( dataFormat = wxDF_BITMAP )
89 {
90 wxBitmap* data = (wxBitmap*)obj;
91 dobj = new wxBitmapDataObject( *data );
92 }
2d120f83 93
dd38c875
MB
94 if( !dobj )
95 return false;
4bb6408c 96
dd38c875 97 return wxTheClipboard->SetData( dobj );
4bb6408c
JS
98}
99
2d120f83 100wxObject *wxGetClipboardData(wxDataFormat dataFormat, long *len)
4bb6408c 101{
dd38c875
MB
102 wxDataObject* dobj = NULL;
103 wxTextDataObject* tobj = NULL;
104 wxBitmapDataObject* bobj = NULL;
2d120f83 105
dd38c875
MB
106 if( dataFormat == wxDF_TEXT )
107 {
108 dobj = tobj = new wxTextDataObject;
109 }
110 else if( dataFormat = wxDF_BITMAP )
111 {
112 dobj = bobj = new wxBitmapDataObject;
113 }
2d120f83 114
dd38c875
MB
115 if( !dobj || !wxTheClipboard->GetData( *dobj ) )
116 return NULL;
2d120f83 117
dd38c875 118 if( tobj )
2d120f83 119 {
dd38c875
MB
120 wxString text = tobj->GetText();
121 wxChar* buf = new wxChar[text.length() + 1];
122
123 if( len ) *len = text.length();
124 return (wxObject*)wxStrcpy( buf, text.c_str() );
125 }
126 else if( bobj )
127 {
128 if( len ) *len = 0;
129 return new wxBitmap( bobj->GetBitmap() );
2d120f83
JS
130 }
131
dd38c875 132 return NULL; // just in case...
4bb6408c
JS
133}
134
fbfb8bcc 135wxDataFormat wxEnumClipboardFormats(const wxDataFormat& dataFormat)
4bb6408c 136{
2d120f83 137 // Only wxDF_TEXT supported
da175b2c 138 if (dataFormat == wxDF_TEXT)
2d120f83
JS
139 return wxDF_TEXT;
140 else
da175b2c 141 return wxDF_INVALID;
4bb6408c
JS
142}
143
dd38c875 144wxDataFormat wxRegisterClipboardFormat(char *WXUNUSED(formatName))
4bb6408c 145{
2d120f83 146 // Not supported
dd38c875 147 return wxDF_INVALID;
4bb6408c
JS
148}
149
fbfb8bcc 150bool wxGetClipboardFormatName(const wxDataFormat& dataFormat, char *formatName,
dd38c875 151 int maxCount)
4bb6408c 152{
e408bf52 153 wxStrlcpy( formatName, dataFormat.GetId().c_str(), maxCount );
dd38c875
MB
154
155 return true;
4bb6408c
JS
156}
157
2d120f83
JS
158//-----------------------------------------------------------------------------
159// wxClipboard
160//-----------------------------------------------------------------------------
4bb6408c 161
47be989f
MB
162struct wxDataIdToDataObject
163{
164 wxDataIdToDataObject( wxDataObject* o, long d, size_t s )
165 : object( o ), size( s ), dataId( d ) { }
166
167 wxDataObject* object;
168 size_t size;
169 long dataId;
170};
171
172#include "wx/listimpl.cpp"
173
e933b5bc
VZ
174WX_DEFINE_LIST(wxDataObjectList)
175WX_DEFINE_LIST(wxDataIdToDataObjectList)
47be989f 176
a13acfd8
VZ
177extern "C"
178{
f332a5a0
MB
179#if wxCHECK_LESSTIF()
180static void wxClipboardCallback( Widget widget, int* data_id,
181 int* priv, int* reason );
a13acfd8 182#else // Motif
47be989f
MB
183static void wxClipboardCallback( Widget widget, long* data_id,
184 long* priv, int* reason );
a13acfd8
VZ
185#endif // Less/Motif
186}
47be989f 187
2d120f83 188IMPLEMENT_DYNAMIC_CLASS(wxClipboard,wxObject)
4bb6408c 189
4bb6408c
JS
190wxClipboard::wxClipboard()
191{
dd38c875 192 m_open = false;
4bb6408c
JS
193}
194
195wxClipboard::~wxClipboard()
196{
55034339 197 Clear();
4bb6408c
JS
198}
199
2d120f83 200void wxClipboard::Clear()
4bb6408c 201{
ac32ba44 202 wxDataObjectList::compatibility_iterator node = m_data.GetFirst();
2d120f83
JS
203 while (node)
204 {
47be989f 205 delete node->GetData();
fd304d98 206 node = node->GetNext();
2d120f83
JS
207 }
208 m_data.Clear();
47be989f 209
ac32ba44 210 for( wxDataIdToDataObjectList::compatibility_iterator node2 = m_idToObject.GetFirst();
47be989f 211 node2; node2 = node2->GetNext() )
fa41b170
MB
212 {
213 delete node2->GetData();
214 }
47be989f 215 m_idToObject.Clear();
4bb6408c
JS
216}
217
2d120f83 218bool wxClipboard::Open()
4bb6408c 219{
dd38c875 220 wxCHECK_MSG( !m_open, false, "clipboard already open" );
55034339 221
dd38c875 222 m_open = true;
4bb6408c 223
dd38c875 224 return true;
2d120f83 225}
4bb6408c 226
2d120f83
JS
227bool wxClipboard::SetData( wxDataObject *data )
228{
dd38c875
MB
229 wxCHECK_MSG( data, false, "data is invalid" );
230 wxCHECK_MSG( m_open, false, "clipboard not open" );
2d120f83 231
12db77ca
VZ
232 Clear();
233
234 return AddData( data );
235}
236
f332a5a0
MB
237#if wxCHECK_LESSTIF()
238void wxClipboardCallback( Widget xwidget, int* data_id,
55034339 239 int* priv, int* WXUNUSED(reason) )
f332a5a0 240#else
47be989f 241void wxClipboardCallback( Widget xwidget, long* data_id,
55034339 242 long* priv, int* WXUNUSED(reason) )
f332a5a0 243#endif
47be989f
MB
244{
245 Display* xdisplay = XtDisplay( xwidget );
246 Window xwindow = XtWindow( xwidget );
247 wxDataObject* dobj = NULL;
248 size_t size = 0;
249
ac32ba44 250 for( wxDataIdToDataObjectList::compatibility_iterator node2 =
47be989f
MB
251 wxTheClipboard->m_idToObject.GetFirst();
252 node2; node2 = node2->GetNext() )
253 {
254 wxDataIdToDataObject* dido = node2->GetData();
255 if( dido->dataId == *data_id )
256 {
257 dobj = dido->object;
258 size = dido->size;
259 break;
260 }
261 }
262
263 if( !dobj ) return;
264
265 wxCharBuffer buffer(size);
266 size_t count = dobj->GetFormatCount( wxDataObject::Get );
267 wxDataFormatScopedArray dfarr( new wxDataFormat[count] );
268 dobj->GetAllFormats( dfarr.get(), wxDataObject::Get );
269
270 if( !dobj->GetDataHere( dfarr[*priv], buffer.data() ) )
271 return;
272
273 while( XmClipboardCopyByName( xdisplay, xwindow, *data_id,
274 buffer.data(), size, 0 )
275 == XmClipboardLocked );
276}
277
12db77ca
VZ
278bool wxClipboard::AddData( wxDataObject *data )
279{
dd38c875
MB
280 wxCHECK_MSG( data, false, "data is invalid" );
281 wxCHECK_MSG( m_open, false, "clipboard not open" );
282
283 m_data.Append( data );
284
285 Display* xdisplay = wxGlobalDisplay();
5a2e3d8c 286 Widget xwidget = (Widget)wxTheApp->GetTopLevelRealizedWidget();
47be989f 287 Window xwindow = XtWindow( xwidget );
9cf3d218 288 wxXmString label( wxTheApp->GetAppDisplayName() );
dd38c875
MB
289 Time timestamp = XtLastTimestampProcessed( xdisplay );
290 long itemId;
291
292 int retval;
12db77ca 293
dd38c875 294 while( ( retval = XmClipboardStartCopy( xdisplay, xwindow, label(),
47be989f
MB
295 timestamp, xwidget,
296 wxClipboardCallback,
dd38c875
MB
297 &itemId ) )
298 == XmClipboardLocked );
299 if( retval != XmClipboardSuccess )
300 return false;
301
302 size_t count = data->GetFormatCount( wxDataObject::Get );
303 wxDataFormatScopedArray dfarr( new wxDataFormat[count] );
304 data->GetAllFormats( dfarr.get(), wxDataObject::Get );
305
306 for( size_t i = 0; i < count; ++i )
2d120f83 307 {
dd38c875 308 size_t size = data->GetDataSize( dfarr[i] );
47be989f 309 long data_id;
dd38c875
MB
310 wxString id = dfarr[i].GetId();
311
312 while( ( retval = XmClipboardCopy( xdisplay, xwindow, itemId,
6991087b 313 id.char_str(),
47be989f 314 NULL, size, i, &data_id ) )
dd38c875 315 == XmClipboardLocked );
47be989f
MB
316
317 m_idToObject.Append( new wxDataIdToDataObject( data, data_id, size ) );
2d120f83 318 }
dd38c875
MB
319
320 while( XmClipboardEndCopy( xdisplay, xwindow, itemId )
321 == XmClipboardLocked );
322
323 return true;
2d120f83 324}
4bb6408c 325
2d120f83
JS
326void wxClipboard::Close()
327{
328 wxCHECK_RET( m_open, "clipboard not open" );
55034339 329
dd38c875 330 m_open = false;
4bb6408c
JS
331}
332
dd38c875 333bool wxClipboard::IsSupported(const wxDataFormat& format)
4bb6408c 334{
dd38c875 335 Display* xdisplay = wxGlobalDisplay();
5a2e3d8c 336 Window xwindow = XtWindow( (Widget)wxTheApp->GetTopLevelRealizedWidget() );
dd38c875
MB
337 bool isSupported = false;
338 int retval, count;
339 unsigned long max_name_length;
340 wxString id = format.GetId();
341
342 while( ( retval = XmClipboardLock( xdisplay, xwindow ) )
343 == XmClipboardLocked );
344 if( retval != XmClipboardSuccess )
345 return false;
346
347 if( XmClipboardInquireCount( xdisplay, xwindow, &count, &max_name_length )
348 == XmClipboardSuccess )
2d120f83 349 {
dd38c875
MB
350 wxCharBuffer buf( max_name_length + 1 );
351 unsigned long copied;
352
353 for( int i = 0; i < count; ++i )
2d120f83 354 {
dd38c875
MB
355 if( XmClipboardInquireFormat( xdisplay, xwindow, i + 1,
356 (XtPointer)buf.data(),
357 max_name_length, &copied )
358 != XmClipboardSuccess )
359 continue;
360
361 buf.data()[copied] = '\0';
362
363 if( buf == id )
2d120f83 364 {
dd38c875
MB
365 isSupported = true;
366 break;
2d120f83 367 }
2d120f83
JS
368 }
369 }
4bb6408c 370
dd38c875 371 XmClipboardUnlock( xdisplay, xwindow, False );
2d120f83 372
dd38c875 373 return isSupported;
2d120f83 374}
4bb6408c 375
dd38c875 376class wxClipboardEndRetrieve
2d120f83 377{
dd38c875
MB
378public:
379 wxClipboardEndRetrieve( Display* display, Window window )
380 : m_display( display ), m_window( window ) { }
381 ~wxClipboardEndRetrieve()
382 {
383 while( XmClipboardEndRetrieve( m_display, m_window )
384 == XmClipboardLocked );
385 }
386private:
387 Display* m_display;
388 Window m_window;
389};
2d120f83 390
dd38c875 391bool wxClipboard::GetData( wxDataObject& data )
4bb6408c 392{
dd38c875
MB
393 wxCHECK_MSG( m_open, false, "clipboard not open" );
394
395 Display* xdisplay = wxGlobalDisplay();
5a2e3d8c 396 Window xwindow = XtWindow( (Widget)wxTheApp->GetTopLevelRealizedWidget() );
dd38c875
MB
397 Time timestamp = XtLastTimestampProcessed( xdisplay );
398
399 wxDataFormat chosenFormat;
400 int retval;
401
402 ///////////////////////////////////////////////////////////////////////////
403 // determine if the cliboard holds any format we like
404 ///////////////////////////////////////////////////////////////////////////
405 while( ( retval = XmClipboardStartRetrieve( xdisplay, xwindow,
406 timestamp ) )
407 == XmClipboardLocked );
408 if( retval != XmClipboardSuccess )
409 return false;
410
411 wxClipboardEndRetrieve endRetrieve( xdisplay, xwindow );
412
413 int count;
414 unsigned long max_name_length;
415 size_t dfcount = data.GetFormatCount( wxDataObject::Set );
416 wxDataFormatScopedArray dfarr( new wxDataFormat[dfcount] );
417 data.GetAllFormats( dfarr.get(), wxDataObject::Set );
55034339 418
dd38c875
MB
419 if( XmClipboardInquireCount( xdisplay, xwindow, &count, &max_name_length )
420 == XmClipboardSuccess )
421 {
422 wxCharBuffer buf( max_name_length + 1 );
423 unsigned long copied;
4bb6408c 424
dd38c875
MB
425 for( int i = 0; i < count; ++i )
426 {
427 if( XmClipboardInquireFormat( xdisplay, xwindow, i + 1,
428 (XtPointer)buf.data(),
429 max_name_length, &copied )
430 != XmClipboardSuccess )
431 continue;
4bb6408c 432
dd38c875 433 buf.data()[copied] = '\0';
4bb6408c 434
dd38c875
MB
435 // try preferred format
436 if( buf == data.GetPreferredFormat( wxDataObject::Set ).GetId() )
437 {
438 chosenFormat = data.GetPreferredFormat( wxDataObject::Set );
439 break;
440 }
2d120f83 441
dd38c875
MB
442 // try all other formats
443 for( size_t i = 0; i < dfcount; ++i )
444 {
445 if( buf == dfarr[i].GetId() )
446 chosenFormat = dfarr[i];
447 }
448 }
2d120f83 449 }
4bb6408c 450
dd38c875
MB
451 if( chosenFormat == wxDF_INVALID )
452 return false;
453
454 ///////////////////////////////////////////////////////////////////////////
455 // now retrieve the data
456 ///////////////////////////////////////////////////////////////////////////
457 unsigned long length, dummy1;
458 long dummy2;
459 wxString id = chosenFormat.GetId();
460
461 while( ( retval = XmClipboardInquireLength( xdisplay, xwindow,
6991087b 462 id.char_str(),
dd38c875
MB
463 &length ) )
464 == XmClipboardLocked );
465 if( retval != XmClipboardSuccess )
466 return false;
467
468 wxCharBuffer buf(length);
469
470 while( ( retval = XmClipboardRetrieve( xdisplay, xwindow,
6991087b 471 id.char_str(),
dd38c875
MB
472 (XtPointer)buf.data(),
473 length, &dummy1, &dummy2 ) )
474 == XmClipboardLocked );
475 if( retval != XmClipboardSuccess )
476 return false;
477
478 if( !data.SetData( chosenFormat, length, buf.data() ) )
479 return false;
480
481 return true;
4bb6408c
JS
482}
483
dfe1eee3 484#endif // wxUSE_CLIPBOARD