]> git.saurik.com Git - wxWidgets.git/blame_incremental - src/msw/clipbrd.cpp
More OS/2 stuff
[wxWidgets.git] / src / msw / clipbrd.cpp
... / ...
CommitLineData
1/////////////////////////////////////////////////////////////////////////////
2// Name: clipbrd.cpp
3// Purpose: Clipboard functionality
4// Author: Julian Smart
5// Modified by:
6// Created: 04/01/98
7// RCS-ID: $Id$
8// Copyright: (c) Julian Smart and Markus Holzem
9// Licence: wxWindows license
10/////////////////////////////////////////////////////////////////////////////
11
12// ===========================================================================
13// declarations
14// ===========================================================================
15
16// ---------------------------------------------------------------------------
17// headers
18// ---------------------------------------------------------------------------
19
20#ifdef __GNUG__
21 #pragma implementation "clipbrd.h"
22#endif
23
24// For compilers that support precompilation, includes "wx.h".
25#include "wx/wxprec.h"
26
27#ifdef __BORLANDC__
28 #pragma hdrstop
29#endif
30
31#ifndef WX_PRECOMP
32 #include "wx/setup.h"
33#endif
34
35#if wxUSE_CLIPBOARD
36
37#ifndef WX_PRECOMP
38 #include "wx/object.h"
39 #include "wx/event.h"
40 #include "wx/app.h"
41 #include "wx/frame.h"
42 #include "wx/bitmap.h"
43 #include "wx/utils.h"
44 #include "wx/intl.h"
45#endif
46
47#if wxUSE_METAFILE
48 #include "wx/metafile.h"
49#endif
50
51#include "wx/log.h"
52#include "wx/clipbrd.h"
53
54#include <string.h>
55#include <windows.h>
56
57#include "wx/msw/private.h"
58#include "wx/msw/dib.h"
59
60// wxDataObject is tied to OLE/drag and drop implementation,
61// therefore so is wxClipboard :-(
62#if wxUSE_DRAG_AND_DROP
63 #include "wx/dataobj.h"
64
65 static bool wxSetClipboardData(wxDataObject *data);
66#endif
67
68#ifdef __WIN16__
69 #define memcpy hmemcpy
70#endif
71
72// ===========================================================================
73// implementation
74// ===========================================================================
75
76// ---------------------------------------------------------------------------
77// old-style clipboard functions using Windows API
78// ---------------------------------------------------------------------------
79
80static bool gs_wxClipboardIsOpen = FALSE;
81
82bool wxOpenClipboard()
83{
84 wxCHECK_MSG( !gs_wxClipboardIsOpen, TRUE, wxT("clipboard already opened.") );
85
86 wxWindow *win = wxTheApp->GetTopWindow();
87 if ( win )
88 {
89 gs_wxClipboardIsOpen = ::OpenClipboard((HWND)win->GetHWND()) != 0;
90
91 if ( !gs_wxClipboardIsOpen )
92 wxLogSysError(_("Failed to open the clipboard."));
93
94 return gs_wxClipboardIsOpen;
95 }
96 else
97 {
98 wxLogDebug(wxT("Can not open clipboard without a main window."));
99
100 return FALSE;
101 }
102}
103
104bool wxCloseClipboard()
105{
106 wxCHECK_MSG( gs_wxClipboardIsOpen, FALSE, wxT("clipboard is not opened") );
107
108 gs_wxClipboardIsOpen = FALSE;
109
110 if ( ::CloseClipboard() == 0 )
111 {
112 wxLogSysError(_("Failed to close the clipboard."));
113
114 return FALSE;
115 }
116
117 return TRUE;
118}
119
120bool wxEmptyClipboard()
121{
122 if ( ::EmptyClipboard() == 0 )
123 {
124 wxLogSysError(_("Failed to empty the clipboard."));
125
126 return FALSE;
127 }
128
129 return TRUE;
130}
131
132bool wxIsClipboardOpened()
133{
134 return gs_wxClipboardIsOpen;
135}
136
137bool wxIsClipboardFormatAvailable(wxDataFormat dataFormat)
138{
139 return ::IsClipboardFormatAvailable(dataFormat) != 0;
140}
141
142#if wxUSE_DRAG_AND_DROP
143static bool wxSetClipboardData(wxDataObject *data)
144{
145 size_t size = data->GetDataSize();
146 HANDLE hGlobal = ::GlobalAlloc(GMEM_MOVEABLE | GMEM_DDESHARE, size);
147 if ( !hGlobal )
148 {
149 wxLogSysError(_("Failed to allocate %dKb of memory for clipboard "
150 "transfer."), size / 1024);
151
152 return FALSE;
153 }
154
155 LPVOID lpGlobalMemory = ::GlobalLock(hGlobal);
156
157 data->GetDataHere(lpGlobalMemory);
158
159 GlobalUnlock(hGlobal);
160
161 wxDataFormat format = data->GetPreferredFormat();
162 if ( !::SetClipboardData(format, hGlobal) )
163 {
164 wxLogSysError(_("Failed to set clipboard data in format %s"),
165 wxDataObject::GetFormatName(format));
166
167 return FALSE;
168 }
169
170 return TRUE;
171}
172#endif // wxUSE_DRAG_AND_DROP
173
174bool wxSetClipboardData(wxDataFormat dataFormat,
175 const void *data,
176 int width, int height)
177{
178 HANDLE handle = 0; // return value of SetClipboardData
179
180 switch (dataFormat)
181 {
182 case wxDF_BITMAP:
183 {
184 wxBitmap *bitmap = (wxBitmap *)data;
185
186 HDC hdcMem = CreateCompatibleDC((HDC) NULL);
187 HDC hdcSrc = CreateCompatibleDC((HDC) NULL);
188 HBITMAP old = (HBITMAP)
189 ::SelectObject(hdcSrc, (HBITMAP)bitmap->GetHBITMAP());
190 HBITMAP hBitmap = CreateCompatibleBitmap(hdcSrc,
191 bitmap->GetWidth(),
192 bitmap->GetHeight());
193 if (!hBitmap)
194 {
195 SelectObject(hdcSrc, old);
196 DeleteDC(hdcMem);
197 DeleteDC(hdcSrc);
198 return FALSE;
199 }
200
201 HBITMAP old1 = (HBITMAP) SelectObject(hdcMem, hBitmap);
202 BitBlt(hdcMem, 0, 0, bitmap->GetWidth(), bitmap->GetHeight(),
203 hdcSrc, 0, 0, SRCCOPY);
204
205 // Select new bitmap out of memory DC
206 SelectObject(hdcMem, old1);
207
208 // Set the data
209 handle = ::SetClipboardData(CF_BITMAP, hBitmap);
210
211 // Clean up
212 SelectObject(hdcSrc, old);
213 DeleteDC(hdcSrc);
214 DeleteDC(hdcMem);
215 break;
216 }
217
218 case wxDF_DIB:
219 {
220#if wxUSE_IMAGE_LOADING_IN_MSW
221 wxBitmap *bitmap = (wxBitmap *)data;
222 HBITMAP hBitmap = (HBITMAP)bitmap->GetHBITMAP();
223 // NULL palette means to use the system one
224 HANDLE hDIB = wxBitmapToDIB(hBitmap, (HPALETTE)NULL);
225 handle = SetClipboardData(CF_DIB, hDIB);
226#endif
227 break;
228 }
229
230#if wxUSE_METAFILE
231 case wxDF_METAFILE:
232 {
233 wxMetafile *wxMF = (wxMetafile *)data;
234 HANDLE data = GlobalAlloc(GHND, sizeof(METAFILEPICT) + 1);
235 METAFILEPICT *mf = (METAFILEPICT *)GlobalLock(data);
236
237 mf->mm = wxMF->GetWindowsMappingMode();
238 mf->xExt = width;
239 mf->yExt = height;
240 mf->hMF = (HMETAFILE) wxMF->GetHMETAFILE();
241 GlobalUnlock(data);
242 wxMF->SetHMETAFILE((WXHANDLE) NULL);
243
244 handle = SetClipboardData(CF_METAFILEPICT, data);
245 break;
246 }
247#endif
248 case CF_SYLK:
249 case CF_DIF:
250 case CF_TIFF:
251 case CF_PALETTE:
252 default:
253 {
254 wxLogError(_("Unsupported clipboard format."));
255 return FALSE;
256 }
257
258 case wxDF_OEMTEXT:
259 dataFormat = wxDF_TEXT;
260 // fall through
261
262 case wxDF_TEXT:
263 {
264 char *s = (char *)data;
265
266 width = strlen(s) + 1;
267 height = 1;
268 DWORD l = (width * height);
269 HANDLE hGlobalMemory = GlobalAlloc(GHND, l);
270 if ( hGlobalMemory )
271 {
272 LPSTR lpGlobalMemory = (LPSTR)GlobalLock(hGlobalMemory);
273
274 memcpy(lpGlobalMemory, s, l);
275
276 GlobalUnlock(hGlobalMemory);
277 }
278
279 handle = SetClipboardData(dataFormat, hGlobalMemory);
280 break;
281 }
282 }
283
284 if ( handle == 0 )
285 {
286 wxLogSysError(_("Failed to set clipboard data."));
287
288 return FALSE;
289 }
290
291 return TRUE;
292}
293
294void *wxGetClipboardData(wxDataFormat dataFormat, long *len)
295{
296 void *retval = NULL;
297
298 switch ( dataFormat )
299 {
300 case wxDF_BITMAP:
301 {
302 BITMAP bm;
303 HBITMAP hBitmap = (HBITMAP) GetClipboardData(CF_BITMAP);
304 if (!hBitmap)
305 break;
306
307 HDC hdcMem = CreateCompatibleDC((HDC) NULL);
308 HDC hdcSrc = CreateCompatibleDC((HDC) NULL);
309
310 HBITMAP old = (HBITMAP) ::SelectObject(hdcSrc, hBitmap);
311 GetObject(hBitmap, sizeof(BITMAP), (LPSTR)&bm);
312
313 HBITMAP hNewBitmap = CreateBitmapIndirect(&bm);
314
315 if (!hNewBitmap)
316 {
317 SelectObject(hdcSrc, old);
318 DeleteDC(hdcMem);
319 DeleteDC(hdcSrc);
320 break;
321 }
322
323 HBITMAP old1 = (HBITMAP) SelectObject(hdcMem, hNewBitmap);
324 BitBlt(hdcMem, 0, 0, bm.bmWidth, bm.bmHeight,
325 hdcSrc, 0, 0, SRCCOPY);
326
327 // Select new bitmap out of memory DC
328 SelectObject(hdcMem, old1);
329
330 // Clean up
331 SelectObject(hdcSrc, old);
332 DeleteDC(hdcSrc);
333 DeleteDC(hdcMem);
334
335 // Create and return a new wxBitmap
336 wxBitmap *wxBM = new wxBitmap;
337 wxBM->SetHBITMAP((WXHBITMAP) hNewBitmap);
338 wxBM->SetWidth(bm.bmWidth);
339 wxBM->SetHeight(bm.bmHeight);
340 wxBM->SetDepth(bm.bmPlanes);
341 wxBM->SetOk(TRUE);
342 retval = wxBM;
343 break;
344 }
345
346 case wxDF_METAFILE:
347 case CF_SYLK:
348 case CF_DIF:
349 case CF_TIFF:
350 case CF_PALETTE:
351 case wxDF_DIB:
352 {
353 wxLogError(_("Unsupported clipboard format."));
354 return FALSE;
355 }
356
357 case wxDF_OEMTEXT:
358 dataFormat = wxDF_TEXT;
359 // fall through
360
361 case wxDF_TEXT:
362 {
363 HANDLE hGlobalMemory = ::GetClipboardData(dataFormat);
364 if (!hGlobalMemory)
365 break;
366
367 DWORD hsize = ::GlobalSize(hGlobalMemory);
368 if (len)
369 *len = hsize;
370
371 char *s = new char[hsize];
372 if (!s)
373 break;
374
375 LPSTR lpGlobalMemory = (LPSTR)::GlobalLock(hGlobalMemory);
376
377 memcpy(s, lpGlobalMemory, hsize);
378
379 ::GlobalUnlock(hGlobalMemory);
380
381 retval = s;
382 break;
383 }
384
385 default:
386 {
387 HANDLE hGlobalMemory = ::GetClipboardData(dataFormat);
388 if ( !hGlobalMemory )
389 break;
390
391 DWORD size = ::GlobalSize(hGlobalMemory);
392 if ( len )
393 *len = size;
394
395 void *buf = malloc(size);
396 if ( !buf )
397 break;
398
399 LPSTR lpGlobalMemory = (LPSTR)::GlobalLock(hGlobalMemory);
400
401 memcpy(buf, lpGlobalMemory, size);
402
403 ::GlobalUnlock(hGlobalMemory);
404
405 retval = buf;
406 break;
407 }
408 }
409
410 if ( !retval )
411 {
412 wxLogSysError(_("Failed to retrieve data from the clipboard."));
413 }
414
415 return retval;
416}
417
418wxDataFormat wxEnumClipboardFormats(wxDataFormat dataFormat)
419{
420 return ::EnumClipboardFormats(dataFormat);
421}
422
423int wxRegisterClipboardFormat(wxChar *formatName)
424{
425 return ::RegisterClipboardFormat(formatName);
426}
427
428bool wxGetClipboardFormatName(wxDataFormat dataFormat,
429 wxChar *formatName,
430 int maxCount)
431{
432 return ::GetClipboardFormatName((int)dataFormat, formatName, maxCount) > 0;
433}
434
435// ---------------------------------------------------------------------------
436// wxClipboard
437// ---------------------------------------------------------------------------
438
439IMPLEMENT_DYNAMIC_CLASS(wxClipboard, wxObject)
440
441wxClipboard* wxTheClipboard = (wxClipboard *)NULL;
442
443wxClipboard::wxClipboard()
444{
445}
446
447wxClipboard::~wxClipboard()
448{
449 Clear();
450}
451
452void wxClipboard::Clear()
453{
454}
455
456bool wxClipboard::Open()
457{
458 return wxOpenClipboard();
459}
460
461bool wxClipboard::SetData( wxDataObject *data )
462{
463 (void)wxEmptyClipboard();
464
465 if ( data )
466 return AddData(data);
467 else
468 return TRUE;
469}
470
471bool wxClipboard::AddData( wxDataObject *data )
472{
473 wxCHECK_MSG( data, FALSE, wxT("data is invalid") );
474
475#if wxUSE_DRAG_AND_DROP
476 wxCHECK_MSG( wxIsClipboardOpened(), FALSE, wxT("clipboard not open") );
477
478 wxDataFormat format = data->GetFormat();
479
480 switch ( format )
481 {
482 case wxDF_TEXT:
483 case wxDF_OEMTEXT:
484 {
485 wxTextDataObject* textDataObject = (wxTextDataObject*) data;
486 wxString str(textDataObject->GetText());
487 return wxSetClipboardData(format, str.c_str());
488 }
489
490 case wxDF_BITMAP:
491 case wxDF_DIB:
492 {
493 wxBitmapDataObject* bitmapDataObject = (wxBitmapDataObject*) data;
494 wxBitmap bitmap(bitmapDataObject->GetBitmap());
495 return wxSetClipboardData(data->GetFormat(), &bitmap);
496 }
497
498#if wxUSE_METAFILE
499 case wxDF_METAFILE:
500 {
501 wxMetafileDataObject* metaFileDataObject =
502 (wxMetafileDataObject*) data;
503 wxMetafile metaFile = metaFileDataObject->GetMetafile();
504 return wxSetClipboardData(wxDF_METAFILE, &metaFile,
505 metaFileDataObject->GetWidth(),
506 metaFileDataObject->GetHeight());
507 }
508#endif // wxUSE_METAFILE
509
510 default:
511 return wxSetClipboardData(data);
512 }
513#else // !wxUSE_DRAG_AND_DROP
514 return FALSE;
515#endif // wxUSE_DRAG_AND_DROP/!wxUSE_DRAG_AND_DROP
516}
517
518void wxClipboard::Close()
519{
520 wxCloseClipboard();
521}
522
523bool wxClipboard::IsSupported( wxDataFormat format )
524{
525 return wxIsClipboardFormatAvailable(format);
526}
527
528bool wxClipboard::GetData( wxDataObject *data )
529{
530 wxCHECK_MSG( wxIsClipboardOpened(), FALSE, wxT("clipboard not open") );
531
532#if wxUSE_DRAG_AND_DROP
533 wxDataFormat format = data->GetFormat();
534 switch ( format )
535 {
536 case wxDF_TEXT:
537 case wxDF_OEMTEXT:
538 {
539 wxTextDataObject* textDataObject = (wxTextDataObject*) data;
540 char* s = (char*) wxGetClipboardData(format);
541 if ( s )
542 {
543 textDataObject->SetText(s);
544 delete[] s;
545 return TRUE;
546 }
547 else
548 return FALSE;
549 }
550
551 case wxDF_BITMAP:
552 case wxDF_DIB:
553 {
554 wxBitmapDataObject* bitmapDataObject = (wxBitmapDataObject *)data;
555 wxBitmap* bitmap = (wxBitmap *)wxGetClipboardData(data->GetFormat());
556 if (bitmap)
557 {
558 bitmapDataObject->SetBitmap(* bitmap);
559 delete bitmap;
560 return TRUE;
561 }
562 else
563 return FALSE;
564 }
565#if wxUSE_METAFILE
566 case wxDF_METAFILE:
567 {
568 wxMetafileDataObject* metaFileDataObject = (wxMetafileDataObject *)data;
569 wxMetafile* metaFile = (wxMetafile *)wxGetClipboardData(wxDF_METAFILE);
570 if (metaFile)
571 {
572 metaFileDataObject->SetMetafile(*metaFile);
573 delete metaFile;
574 return TRUE;
575 }
576 else
577 return FALSE;
578 }
579#endif
580 default:
581 {
582 long len;
583 void *buf = wxGetClipboardData(format, &len);
584 if ( buf )
585 {
586 // FIXME this is for testing only!!
587 ((wxPrivateDataObject *)data)->SetData(buf, len);
588 free(buf);
589
590 return TRUE;
591 }
592 }
593
594 return FALSE;
595 }
596#else
597 return FALSE;
598#endif
599}
600
601//-----------------------------------------------------------------------------
602// wxClipboardModule
603//-----------------------------------------------------------------------------
604
605IMPLEMENT_DYNAMIC_CLASS(wxClipboardModule,wxModule)
606
607bool wxClipboardModule::OnInit()
608{
609 wxTheClipboard = new wxClipboard();
610
611 return TRUE;
612}
613
614void wxClipboardModule::OnExit()
615{
616 if (wxTheClipboard) delete wxTheClipboard;
617 wxTheClipboard = (wxClipboard*) NULL;
618}
619
620#else
621 #error "Please turn wxUSE_CLIPBOARD on to compile this file."
622#endif // wxUSE_CLIPBOARD
623