]> git.saurik.com Git - wxWidgets.git/blame - samples/dnd/dnd.cpp
Minor changes to printing sample.
[wxWidgets.git] / samples / dnd / dnd.cpp
CommitLineData
43d811ea 1/////////////////////////////////////////////////////////////////////////////
457814b5
JS
2// Name: dnd.cpp
3// Purpose: Drag and drop sample
4// Author: Vadim Zeitlin
43d811ea
JS
5// Modified by:
6// Created: 04/01/98
8bbe427f 7// RCS-ID: $Id$
43d811ea 8// Copyright:
8bbe427f 9// Licence: wxWindows licence
43d811ea
JS
10/////////////////////////////////////////////////////////////////////////////
11
c50f1fb9 12#include "wx/wxprec.h"
457814b5
JS
13
14#ifdef __BORLANDC__
15#pragma hdrstop
16#endif
17
18#ifndef WX_PRECOMP
c50f1fb9 19#include "wx/wx.h"
457814b5
JS
20#endif
21
8e193f38
VZ
22#if !wxUSE_DRAG_AND_DROP
23 #error This sample requires drag and drop support in the library
e2acb9ae
RR
24#endif
25
5a1c877f
VZ
26// under Windows we also support data transfer of metafiles as an extra bonus,
27// but they're not available under other platforms
28#ifdef __WINDOWS__
29 #define USE_METAFILES
30#endif // Windows
31
4693b20c
MB
32#define USE_RESOURCES 0
33#if !wxUSE_RESOURCES
34#undef USE_RESOURCES
35#define USE_RESOURCES 0
36#endif
37
c50f1fb9
VZ
38#include "wx/intl.h"
39#include "wx/log.h"
457814b5 40
c50f1fb9 41#include "wx/dnd.h"
956dbab1 42#include "wx/dirdlg.h"
e2acb9ae
RR
43#include "wx/filedlg.h"
44#include "wx/image.h"
45#include "wx/clipbrd.h"
8e193f38 46#include "wx/colordlg.h"
4693b20c
MB
47#if USE_RESOURCES
48 #include "wx/resource.h"
49#else
50 #include "wx/sizer.h"
51#endif
acbd13a3 52
5a1c877f
VZ
53#ifdef USE_METAFILES
54 #include "wx/metafile.h"
55#endif // Windows
56
618f2efa 57#if defined(__WXGTK__) || defined(__WXX11__) || defined(__WXMOTIF__) || defined(__WXMAC__)
c50f1fb9 58 #include "mondrian.xpm"
f6bcfd97
BP
59
60 #include "dnd_copy.xpm"
61 #include "dnd_move.xpm"
62 #include "dnd_none.xpm"
47908e25
RR
63#endif
64
457814b5 65// ----------------------------------------------------------------------------
2845ddc2 66// Derive two simple classes which just put in the listbox the strings (text or
457814b5
JS
67// file names) we drop on them
68// ----------------------------------------------------------------------------
ab8884ac 69
457814b5
JS
70class DnDText : public wxTextDropTarget
71{
72public:
c50f1fb9 73 DnDText(wxListBox *pOwner) { m_pOwner = pOwner; }
457814b5 74
9e2896e5 75 virtual bool OnDropText(wxCoord x, wxCoord y, const wxString& text);
457814b5
JS
76
77private:
c50f1fb9 78 wxListBox *m_pOwner;
457814b5
JS
79};
80
81class DnDFile : public wxFileDropTarget
82{
83public:
c50f1fb9 84 DnDFile(wxListBox *pOwner) { m_pOwner = pOwner; }
457814b5 85
9e2896e5
VZ
86 virtual bool OnDropFiles(wxCoord x, wxCoord y,
87 const wxArrayString& filenames);
457814b5
JS
88
89private:
c50f1fb9 90 wxListBox *m_pOwner;
457814b5
JS
91};
92
444ad3a7
VZ
93// ----------------------------------------------------------------------------
94// Define a custom dtop target accepting URLs
95// ----------------------------------------------------------------------------
96
4693b20c 97class URLDropTarget : public wxDropTarget
444ad3a7
VZ
98{
99public:
100 URLDropTarget() { SetDataObject(new wxURLDataObject); }
101
102 void OnDropURL(wxCoord x, wxCoord y, const wxString& text)
103 {
104 // of course, a real program would do something more useful here...
105 wxMessageBox(text, _T("wxDnD sample: got URL"),
106 wxICON_INFORMATION | wxOK);
107 }
108
109 // URLs can't be moved, only copied
110 virtual wxDragResult OnDragOver(wxCoord WXUNUSED(x), wxCoord WXUNUSED(y),
111 wxDragResult def)
e6d318c2
RD
112 {
113 return wxDragLink; // At least IE 5.x needs wxDragLink, the
114 // other browsers on MSW seem okay with it too.
115 }
444ad3a7
VZ
116
117 // translate this to calls to OnDropURL() just for convenience
118 virtual wxDragResult OnData(wxCoord x, wxCoord y, wxDragResult def)
119 {
120 if ( !GetData() )
121 return wxDragNone;
122
123 OnDropURL(x, y, ((wxURLDataObject *)m_dataObject)->GetURL());
124
125 return def;
126 }
127};
128
457814b5
JS
129// ----------------------------------------------------------------------------
130// Define a new application type
131// ----------------------------------------------------------------------------
ab8884ac 132
457814b5 133class DnDApp : public wxApp
8bbe427f 134{
457814b5 135public:
8e193f38 136 virtual bool OnInit();
457814b5
JS
137};
138
139IMPLEMENT_APP(DnDApp);
140
ae3dd4a5
VZ
141// ----------------------------------------------------------------------------
142// Define canvas class to show a bitmap
143// ----------------------------------------------------------------------------
144
145class DnDCanvasBitmap : public wxScrolledWindow
146{
147public:
148 DnDCanvasBitmap(wxWindow *parent) : wxScrolledWindow(parent) { }
149
150 void SetBitmap(const wxBitmap& bitmap)
151 {
152 m_bitmap = bitmap;
153
154 SetScrollbars(10, 10,
155 m_bitmap.GetWidth() / 10, m_bitmap.GetHeight() / 10);
156
157 Refresh();
158 }
159
160 void OnPaint(wxPaintEvent& event)
161 {
162 wxPaintDC dc(this);
163
164 if ( m_bitmap.Ok() )
165 {
166 PrepareDC(dc);
167
168 dc.DrawBitmap(m_bitmap, 0, 0);
169 }
170 }
171
172private:
173 wxBitmap m_bitmap;
174
175 DECLARE_EVENT_TABLE()
176};
177
178#ifdef USE_METAFILES
179
180// and the same thing fo metafiles
181class DnDCanvasMetafile : public wxScrolledWindow
182{
183public:
184 DnDCanvasMetafile(wxWindow *parent) : wxScrolledWindow(parent) { }
185
186 void SetMetafile(const wxMetafile& metafile)
187 {
188 m_metafile = metafile;
189
190 SetScrollbars(10, 10,
191 m_metafile.GetWidth() / 10, m_metafile.GetHeight() / 10);
192
193 Refresh();
194 }
195
196 void OnPaint(wxPaintEvent& event)
197 {
198 wxPaintDC dc(this);
199
200 if ( m_metafile.Ok() )
201 {
202 PrepareDC(dc);
203
204 m_metafile.Play(&dc);
205 }
206 }
207
208private:
209 wxMetafile m_metafile;
210
211 DECLARE_EVENT_TABLE()
212};
213
214#endif // USE_METAFILES
215
457814b5 216// ----------------------------------------------------------------------------
8e193f38 217// Define a new frame type for the main frame
457814b5 218// ----------------------------------------------------------------------------
8e193f38 219
457814b5 220class DnDFrame : public wxFrame
8bbe427f 221{
457814b5 222public:
9f84eccd 223 DnDFrame(wxFrame *frame, wxChar *title, int x, int y, int w, int h);
2245b2b2 224 virtual ~DnDFrame();
457814b5 225
c50f1fb9 226 void OnPaint(wxPaintEvent& event);
fcf689db 227 void OnSize(wxSizeEvent& event);
2245b2b2 228 void OnQuit(wxCommandEvent& event);
c50f1fb9 229 void OnAbout(wxCommandEvent& event);
2245b2b2
VZ
230 void OnDrag(wxCommandEvent& event);
231 void OnDragMoveByDefault(wxCommandEvent& event);
232 void OnDragMoveAllow(wxCommandEvent& event);
8e193f38 233 void OnNewFrame(wxCommandEvent& event);
c50f1fb9
VZ
234 void OnHelp (wxCommandEvent& event);
235 void OnLogClear(wxCommandEvent& event);
51edda6a 236
c50f1fb9
VZ
237 void OnCopy(wxCommandEvent& event);
238 void OnPaste(wxCommandEvent& event);
51edda6a 239
e2acb9ae
RR
240 void OnCopyBitmap(wxCommandEvent& event);
241 void OnPasteBitmap(wxCommandEvent& event);
43d811ea 242
5a1c877f
VZ
243#ifdef USE_METAFILES
244 void OnPasteMetafile(wxCommandEvent& event);
245#endif // USE_METAFILES
246
51edda6a
VZ
247 void OnCopyFiles(wxCommandEvent& event);
248
c50f1fb9
VZ
249 void OnLeftDown(wxMouseEvent& event);
250 void OnRightDown(wxMouseEvent& event);
8bbe427f 251
2245b2b2
VZ
252 void OnUpdateUIMoveByDefault(wxUpdateUIEvent& event);
253
d59ceba5
VZ
254 void OnUpdateUIPasteText(wxUpdateUIEvent& event);
255 void OnUpdateUIPasteBitmap(wxUpdateUIEvent& event);
256
c50f1fb9 257 DECLARE_EVENT_TABLE()
457814b5
JS
258
259private:
2245b2b2 260 // GUI controls
e2acb9ae 261 wxListBox *m_ctrlFile,
90e12284 262 *m_ctrlText;
e2acb9ae 263 wxTextCtrl *m_ctrlLog;
457814b5 264
90e12284
VZ
265 wxLog *m_pLog,
266 *m_pLogPrev;
43d811ea 267
2245b2b2
VZ
268 // move the text by default (or copy)?
269 bool m_moveByDefault;
270
271 // allow moving the text at all?
272 bool m_moveAllow;
273
274 // the text we drag
e2acb9ae 275 wxString m_strText;
457814b5
JS
276};
277
8e193f38
VZ
278// ----------------------------------------------------------------------------
279// A shape is an example of application-specific data which may be transported
280// via drag-and-drop or clipboard: in our case, we have different geometric
281// shapes, each one with its own colour and position
282// ----------------------------------------------------------------------------
283
284class DnDShape
285{
286public:
287 enum Kind
288 {
289 None,
290 Triangle,
291 Rectangle,
292 Ellipse
293 };
294
295 DnDShape(const wxPoint& pos,
296 const wxSize& size,
297 const wxColour& col)
298 : m_pos(pos), m_size(size), m_col(col)
299 {
300 }
301
9e2896e5
VZ
302 // this is for debugging - lets us see when exactly an object is freed
303 // (this may be later than you think if it's on the clipboard, for example)
304 virtual ~DnDShape() { }
305
8e193f38 306 // the functions used for drag-and-drop: they dump and restore a shape into
d59ceba5 307 // some bitwise-copiable data (might use streams too...)
8e193f38
VZ
308 // ------------------------------------------------------------------------
309
310 // restore from buffer
311 static DnDShape *New(const void *buf);
312
313 virtual size_t GetDataSize() const
314 {
315 return sizeof(ShapeDump);
316 }
317
318 virtual void GetDataHere(void *buf) const
319 {
320 ShapeDump& dump = *(ShapeDump *)buf;
321 dump.x = m_pos.x;
322 dump.y = m_pos.y;
323 dump.w = m_size.x;
324 dump.h = m_size.y;
325 dump.r = m_col.Red();
326 dump.g = m_col.Green();
327 dump.b = m_col.Blue();
328 dump.k = GetKind();
329 }
330
331 // accessors
332 const wxPoint& GetPosition() const { return m_pos; }
333 const wxColour& GetColour() const { return m_col; }
334 const wxSize& GetSize() const { return m_size; }
335
d59ceba5
VZ
336 void Move(const wxPoint& pos) { m_pos = pos; }
337
8e193f38
VZ
338 // to implement in derived classes
339 virtual Kind GetKind() const = 0;
340
1dd989e1 341 virtual void Draw(wxDC& dc)
8e193f38
VZ
342 {
343 dc.SetPen(wxPen(m_col, 1, wxSOLID));
344 }
345
346protected:
f88c1a17 347 //get a point 1 up and 1 left, otherwise the mid-point of a triangle is on the line
8e193f38 348 wxPoint GetCentre() const
f88c1a17
JS
349 { return wxPoint(m_pos.x + m_size.x / 2 - 1, m_pos.y + m_size.y / 2 - 1); }
350
8e193f38
VZ
351 struct ShapeDump
352 {
353 int x, y, // position
354 w, h, // size
355 r, g, b, // colour
356 k; // kind
357 };
358
359 wxPoint m_pos;
360 wxSize m_size;
361 wxColour m_col;
362};
363
364class DnDTriangularShape : public DnDShape
365{
366public:
367 DnDTriangularShape(const wxPoint& pos,
368 const wxSize& size,
369 const wxColour& col)
370 : DnDShape(pos, size, col)
371 {
4693b20c 372 wxLogMessage(wxT("DnDTriangularShape is being created"));
8e193f38
VZ
373 }
374
9e2896e5
VZ
375 virtual ~DnDTriangularShape()
376 {
4693b20c 377 wxLogMessage(wxT("DnDTriangularShape is being deleted"));
9e2896e5
VZ
378 }
379
8e193f38
VZ
380 virtual Kind GetKind() const { return Triangle; }
381 virtual void Draw(wxDC& dc)
382 {
383 DnDShape::Draw(dc);
384
385 // well, it's a bit difficult to describe a triangle by position and
386 // size, but we're not doing geometry here, do we? ;-)
387 wxPoint p1(m_pos);
388 wxPoint p2(m_pos.x + m_size.x, m_pos.y);
389 wxPoint p3(m_pos.x, m_pos.y + m_size.y);
390
391 dc.DrawLine(p1, p2);
392 dc.DrawLine(p2, p3);
393 dc.DrawLine(p3, p1);
394
f88c1a17
JS
395 //works in multicolor modes; on GTK (at least) will fail in 16-bit color
396 dc.SetBrush(wxBrush(m_col, wxSOLID));
8e193f38
VZ
397 dc.FloodFill(GetCentre(), m_col, wxFLOOD_BORDER);
398 }
399};
400
401class DnDRectangularShape : public DnDShape
402{
403public:
404 DnDRectangularShape(const wxPoint& pos,
405 const wxSize& size,
406 const wxColour& col)
407 : DnDShape(pos, size, col)
408 {
4693b20c 409 wxLogMessage(wxT("DnDRectangularShape is being created"));
8e193f38
VZ
410 }
411
9e2896e5
VZ
412 virtual ~DnDRectangularShape()
413 {
4693b20c 414 wxLogMessage(wxT("DnDRectangularShape is being deleted"));
9e2896e5
VZ
415 }
416
8e193f38
VZ
417 virtual Kind GetKind() const { return Rectangle; }
418 virtual void Draw(wxDC& dc)
419 {
420 DnDShape::Draw(dc);
421
422 wxPoint p1(m_pos);
423 wxPoint p2(p1.x + m_size.x, p1.y);
424 wxPoint p3(p2.x, p2.y + m_size.y);
425 wxPoint p4(p1.x, p3.y);
426
427 dc.DrawLine(p1, p2);
428 dc.DrawLine(p2, p3);
429 dc.DrawLine(p3, p4);
430 dc.DrawLine(p4, p1);
431
f88c1a17 432 dc.SetBrush(wxBrush(m_col, wxSOLID));
8e193f38
VZ
433 dc.FloodFill(GetCentre(), m_col, wxFLOOD_BORDER);
434 }
435};
436
437class DnDEllipticShape : public DnDShape
438{
439public:
440 DnDEllipticShape(const wxPoint& pos,
441 const wxSize& size,
442 const wxColour& col)
443 : DnDShape(pos, size, col)
444 {
4693b20c 445 wxLogMessage(wxT("DnDEllipticShape is being created"));
8e193f38
VZ
446 }
447
9e2896e5
VZ
448 virtual ~DnDEllipticShape()
449 {
4693b20c 450 wxLogMessage(wxT("DnDEllipticShape is being deleted"));
9e2896e5
VZ
451 }
452
8e193f38
VZ
453 virtual Kind GetKind() const { return Ellipse; }
454 virtual void Draw(wxDC& dc)
455 {
456 DnDShape::Draw(dc);
457
458 dc.DrawEllipse(m_pos, m_size);
459
f88c1a17 460 dc.SetBrush(wxBrush(m_col, wxSOLID));
8e193f38
VZ
461 dc.FloodFill(GetCentre(), m_col, wxFLOOD_BORDER);
462 }
463};
464
465// ----------------------------------------------------------------------------
466// A wxDataObject specialisation for the application-specific data
467// ----------------------------------------------------------------------------
468
4693b20c 469static const wxChar *shapeFormatId = wxT("wxShape");
8e193f38
VZ
470
471class DnDShapeDataObject : public wxDataObject
472{
473public:
474 // ctor doesn't copy the pointer, so it shouldn't go away while this object
475 // is alive
9e2896e5 476 DnDShapeDataObject(DnDShape *shape = (DnDShape *)NULL)
8e193f38 477 {
9e2896e5
VZ
478 if ( shape )
479 {
480 // we need to copy the shape because the one we're handled may be
481 // deleted while it's still on the clipboard (for example) - and we
482 // reuse the serialisation methods here to copy it
483 void *buf = malloc(shape->DnDShape::GetDataSize());
484 shape->GetDataHere(buf);
485 m_shape = DnDShape::New(buf);
486
487 free(buf);
488 }
489 else
490 {
491 // nothing to copy
492 m_shape = NULL;
493 }
8e193f38
VZ
494
495 // this string should uniquely identify our format, but is otherwise
496 // arbitrary
497 m_formatShape.SetId(shapeFormatId);
498
499 // we don't draw the shape to a bitmap until it's really needed (i.e.
500 // we're asked to do so)
501 m_hasBitmap = FALSE;
5a1c877f
VZ
502#ifdef USE_METAFILES
503 m_hasMetaFile = FALSE;
504#endif // Windows
8e193f38
VZ
505 }
506
9e2896e5
VZ
507 virtual ~DnDShapeDataObject() { delete m_shape; }
508
90e12284
VZ
509 // after a call to this function, the shape is owned by the caller and it
510 // is responsible for deleting it!
511 //
512 // NB: a better solution would be to make DnDShapes ref counted and this
513 // is what should probably be done in a real life program, otherwise
514 // the ownership problems become too complicated really fast
515 DnDShape *GetShape()
516 {
517 DnDShape *shape = m_shape;
518
519 m_shape = (DnDShape *)NULL;
520 m_hasBitmap = FALSE;
5a1c877f
VZ
521#ifdef USE_METAFILES
522 m_hasMetaFile = FALSE;
523#endif // Windows
90e12284
VZ
524
525 return shape;
526 }
d59ceba5 527
8e193f38
VZ
528 // implement base class pure virtuals
529 // ----------------------------------
530
9e2896e5 531 virtual wxDataFormat GetPreferredFormat(Direction WXUNUSED(dir)) const
8e193f38
VZ
532 {
533 return m_formatShape;
534 }
535
9e2896e5 536 virtual size_t GetFormatCount(Direction dir) const
8e193f38 537 {
d59ceba5
VZ
538 // our custom format is supported by both GetData() and SetData()
539 size_t nFormats = 1;
9e2896e5 540 if ( dir == Get )
d59ceba5
VZ
541 {
542 // but the bitmap format(s) are only supported for output
5a1c877f
VZ
543 nFormats += m_dobjBitmap.GetFormatCount(dir);
544
545#ifdef USE_METAFILES
546 nFormats += m_dobjMetaFile.GetFormatCount(dir);
547#endif // Windows
d59ceba5
VZ
548 }
549
550 return nFormats;
8e193f38
VZ
551 }
552
9e2896e5 553 virtual void GetAllFormats(wxDataFormat *formats, Direction dir) const
8e193f38
VZ
554 {
555 formats[0] = m_formatShape;
9e2896e5 556 if ( dir == Get )
d59ceba5 557 {
5a1c877f
VZ
558 // in Get direction we additionally support bitmaps and metafiles
559 // under Windows
560 m_dobjBitmap.GetAllFormats(&formats[1], dir);
561
562#ifdef USE_METAFILES
563 // don't assume that m_dobjBitmap has only 1 format
564 m_dobjMetaFile.GetAllFormats(&formats[1 +
565 m_dobjBitmap.GetFormatCount(dir)], dir);
566#endif // Windows
d59ceba5 567 }
8e193f38
VZ
568 }
569
570 virtual size_t GetDataSize(const wxDataFormat& format) const
571 {
572 if ( format == m_formatShape )
573 {
574 return m_shape->GetDataSize();
575 }
5a1c877f 576#ifdef USE_METAFILES
ae3dd4a5 577 else if ( m_dobjMetaFile.IsSupported(format) )
5a1c877f
VZ
578 {
579 if ( !m_hasMetaFile )
580 CreateMetaFile();
581
ae3dd4a5 582 return m_dobjMetaFile.GetDataSize(format);
5a1c877f
VZ
583 }
584#endif // Windows
8e193f38
VZ
585 else
586 {
ae3dd4a5 587 wxASSERT_MSG( m_dobjBitmap.IsSupported(format),
4693b20c 588 wxT("unexpected format") );
ae3dd4a5 589
8e193f38
VZ
590 if ( !m_hasBitmap )
591 CreateBitmap();
592
5a1c877f 593 return m_dobjBitmap.GetDataSize();
8e193f38
VZ
594 }
595 }
596
d59ceba5 597 virtual bool GetDataHere(const wxDataFormat& format, void *pBuf) const
8e193f38
VZ
598 {
599 if ( format == m_formatShape )
600 {
601 m_shape->GetDataHere(pBuf);
d59ceba5
VZ
602
603 return TRUE;
8e193f38 604 }
5a1c877f 605#ifdef USE_METAFILES
ae3dd4a5 606 else if ( m_dobjMetaFile.IsSupported(format) )
5a1c877f
VZ
607 {
608 if ( !m_hasMetaFile )
609 CreateMetaFile();
610
ae3dd4a5 611 return m_dobjMetaFile.GetDataHere(format, pBuf);
5a1c877f
VZ
612 }
613#endif // Windows
8e193f38
VZ
614 else
615 {
ae3dd4a5 616 wxASSERT_MSG( m_dobjBitmap.IsSupported(format),
4693b20c 617 wxT("unexpected format") );
ae3dd4a5 618
8e193f38
VZ
619 if ( !m_hasBitmap )
620 CreateBitmap();
621
5a1c877f 622 return m_dobjBitmap.GetDataHere(pBuf);
8e193f38
VZ
623 }
624 }
625
9e2896e5
VZ
626 virtual bool SetData(const wxDataFormat& format,
627 size_t len, const void *buf)
d59ceba5 628 {
4693b20c
MB
629 wxCHECK_MSG( format == m_formatShape, FALSE,
630 wxT( "unsupported format") );
d59ceba5
VZ
631
632 delete m_shape;
633 m_shape = DnDShape::New(buf);
634
635 // the shape has changed
636 m_hasBitmap = FALSE;
637
5a1c877f
VZ
638#ifdef USE_METAFILES
639 m_hasMetaFile = FALSE;
640#endif // Windows
641
d59ceba5
VZ
642 return TRUE;
643 }
644
8e193f38 645private:
5a1c877f 646 // creates a bitmap and assigns it to m_dobjBitmap (also sets m_hasBitmap)
8e193f38 647 void CreateBitmap() const;
5a1c877f
VZ
648#ifdef USE_METAFILES
649 void CreateMetaFile() const;
650#endif // Windows
8e193f38
VZ
651
652 wxDataFormat m_formatShape; // our custom format
653
5a1c877f
VZ
654 wxBitmapDataObject m_dobjBitmap; // it handles bitmaps
655 bool m_hasBitmap; // true if m_dobjBitmap has valid bitmap
656
657#ifdef USE_METAFILES
658 wxMetaFileDataObject m_dobjMetaFile;// handles metafiles
659 bool m_hasMetaFile; // true if we have valid metafile
660#endif // Windows
8e193f38
VZ
661
662 DnDShape *m_shape; // our data
663};
664
665// ----------------------------------------------------------------------------
666// A dialog to edit shape properties
667// ----------------------------------------------------------------------------
668
669class DnDShapeDialog : public wxDialog
670{
671public:
672 DnDShapeDialog(wxFrame *parent, DnDShape *shape);
673
674 DnDShape *GetShape() const;
675
676 virtual bool TransferDataToWindow();
677 virtual bool TransferDataFromWindow();
678
679 void OnColour(wxCommandEvent& event);
680
681private:
682 // input
683 DnDShape *m_shape;
684
685 // output
686 DnDShape::Kind m_shapeKind;
687 wxPoint m_pos;
688 wxSize m_size;
689 wxColour m_col;
690
691 // controls
692 wxRadioBox *m_radio;
693 wxTextCtrl *m_textX,
694 *m_textY,
695 *m_textW,
696 *m_textH;
697
698 DECLARE_EVENT_TABLE()
699};
700
701// ----------------------------------------------------------------------------
702// A frame for the shapes which can be drag-and-dropped between frames
703// ----------------------------------------------------------------------------
704
705class DnDShapeFrame : public wxFrame
706{
707public:
708 DnDShapeFrame(wxFrame *parent);
709 ~DnDShapeFrame();
710
711 void SetShape(DnDShape *shape);
712
713 // callbacks
d59ceba5
VZ
714 void OnNewShape(wxCommandEvent& event);
715 void OnEditShape(wxCommandEvent& event);
716 void OnClearShape(wxCommandEvent& event);
717
718 void OnCopyShape(wxCommandEvent& event);
719 void OnPasteShape(wxCommandEvent& event);
720
721 void OnUpdateUICopy(wxUpdateUIEvent& event);
722 void OnUpdateUIPaste(wxUpdateUIEvent& event);
723
8e193f38 724 void OnDrag(wxMouseEvent& event);
8e193f38 725 void OnPaint(wxPaintEvent& event);
90e12284 726 void OnDrop(wxCoord x, wxCoord y, DnDShape *shape);
8e193f38
VZ
727
728private:
729 DnDShape *m_shape;
730
d59ceba5
VZ
731 static DnDShapeFrame *ms_lastDropTarget;
732
8e193f38
VZ
733 DECLARE_EVENT_TABLE()
734};
735
736// ----------------------------------------------------------------------------
737// wxDropTarget derivation for DnDShapes
738// ----------------------------------------------------------------------------
739
740class DnDShapeDropTarget : public wxDropTarget
741{
742public:
743 DnDShapeDropTarget(DnDShapeFrame *frame)
9e2896e5 744 : wxDropTarget(new DnDShapeDataObject)
8e193f38
VZ
745 {
746 m_frame = frame;
8e193f38
VZ
747 }
748
749 // override base class (pure) virtuals
72a7edf0 750 virtual wxDragResult OnEnter(wxCoord x, wxCoord y, wxDragResult def)
9f84eccd 751 { m_frame->SetStatusText(_T("Mouse entered the frame")); return OnDragOver(x, y, def); }
8e193f38 752 virtual void OnLeave()
9f84eccd 753 { m_frame->SetStatusText(_T("Mouse left the frame")); }
8ee9d618 754 virtual wxDragResult OnData(wxCoord x, wxCoord y, wxDragResult def)
8e193f38 755 {
9e2896e5
VZ
756 if ( !GetData() )
757 {
4693b20c 758 wxLogError(wxT("Failed to get drag and drop data"));
9e2896e5 759
8ee9d618 760 return wxDragNone;
9e2896e5
VZ
761 }
762
763 m_frame->OnDrop(x, y,
764 ((DnDShapeDataObject *)GetDataObject())->GetShape());
8e193f38 765
8ee9d618 766 return def;
8e193f38
VZ
767 }
768
8e193f38
VZ
769private:
770 DnDShapeFrame *m_frame;
8e193f38
VZ
771};
772
ae3dd4a5
VZ
773// ----------------------------------------------------------------------------
774// functions prototypes
775// ----------------------------------------------------------------------------
776
777static void ShowBitmap(const wxBitmap& bitmap);
778
779#ifdef USE_METAFILES
780static void ShowMetaFile(const wxMetaFile& metafile);
781#endif // USE_METAFILES
782
457814b5
JS
783// ----------------------------------------------------------------------------
784// IDs for the menu commands
785// ----------------------------------------------------------------------------
ab8884ac 786
457814b5
JS
787enum
788{
c50f1fb9
VZ
789 Menu_Quit = 1,
790 Menu_Drag,
2245b2b2
VZ
791 Menu_DragMoveDef,
792 Menu_DragMoveAllow,
8e193f38 793 Menu_NewFrame,
c50f1fb9
VZ
794 Menu_About = 101,
795 Menu_Help,
796 Menu_Clear,
797 Menu_Copy,
e2acb9ae
RR
798 Menu_Paste,
799 Menu_CopyBitmap,
800 Menu_PasteBitmap,
5a1c877f 801 Menu_PasteMFile,
51edda6a 802 Menu_CopyFiles,
d59ceba5
VZ
803 Menu_Shape_New = 500,
804 Menu_Shape_Edit,
805 Menu_Shape_Clear,
806 Menu_ShapeClipboard_Copy,
807 Menu_ShapeClipboard_Paste,
8e193f38 808 Button_Colour = 1001
457814b5
JS
809};
810
811BEGIN_EVENT_TABLE(DnDFrame, wxFrame)
e2acb9ae
RR
812 EVT_MENU(Menu_Quit, DnDFrame::OnQuit)
813 EVT_MENU(Menu_About, DnDFrame::OnAbout)
814 EVT_MENU(Menu_Drag, DnDFrame::OnDrag)
2245b2b2
VZ
815 EVT_MENU(Menu_DragMoveDef, DnDFrame::OnDragMoveByDefault)
816 EVT_MENU(Menu_DragMoveAllow,DnDFrame::OnDragMoveAllow)
8e193f38 817 EVT_MENU(Menu_NewFrame, DnDFrame::OnNewFrame)
e2acb9ae
RR
818 EVT_MENU(Menu_Help, DnDFrame::OnHelp)
819 EVT_MENU(Menu_Clear, DnDFrame::OnLogClear)
820 EVT_MENU(Menu_Copy, DnDFrame::OnCopy)
821 EVT_MENU(Menu_Paste, DnDFrame::OnPaste)
822 EVT_MENU(Menu_CopyBitmap, DnDFrame::OnCopyBitmap)
823 EVT_MENU(Menu_PasteBitmap,DnDFrame::OnPasteBitmap)
5a1c877f
VZ
824#ifdef USE_METAFILES
825 EVT_MENU(Menu_PasteMFile, DnDFrame::OnPasteMetafile)
826#endif // USE_METAFILES
51edda6a 827 EVT_MENU(Menu_CopyFiles, DnDFrame::OnCopyFiles)
d59ceba5 828
2245b2b2
VZ
829 EVT_UPDATE_UI(Menu_DragMoveDef, DnDFrame::OnUpdateUIMoveByDefault)
830
d59ceba5
VZ
831 EVT_UPDATE_UI(Menu_Paste, DnDFrame::OnUpdateUIPasteText)
832 EVT_UPDATE_UI(Menu_PasteBitmap, DnDFrame::OnUpdateUIPasteBitmap)
e2acb9ae
RR
833
834 EVT_LEFT_DOWN( DnDFrame::OnLeftDown)
835 EVT_RIGHT_DOWN( DnDFrame::OnRightDown)
836 EVT_PAINT( DnDFrame::OnPaint)
fcf689db 837 EVT_SIZE( DnDFrame::OnSize)
457814b5
JS
838END_EVENT_TABLE()
839
8e193f38 840BEGIN_EVENT_TABLE(DnDShapeFrame, wxFrame)
d59ceba5
VZ
841 EVT_MENU(Menu_Shape_New, DnDShapeFrame::OnNewShape)
842 EVT_MENU(Menu_Shape_Edit, DnDShapeFrame::OnEditShape)
843 EVT_MENU(Menu_Shape_Clear, DnDShapeFrame::OnClearShape)
844
845 EVT_MENU(Menu_ShapeClipboard_Copy, DnDShapeFrame::OnCopyShape)
846 EVT_MENU(Menu_ShapeClipboard_Paste, DnDShapeFrame::OnPasteShape)
847
848 EVT_UPDATE_UI(Menu_ShapeClipboard_Copy, DnDShapeFrame::OnUpdateUICopy)
849 EVT_UPDATE_UI(Menu_ShapeClipboard_Paste, DnDShapeFrame::OnUpdateUIPaste)
850
851 EVT_LEFT_DOWN(DnDShapeFrame::OnDrag)
852
8e193f38
VZ
853 EVT_PAINT(DnDShapeFrame::OnPaint)
854END_EVENT_TABLE()
855
856BEGIN_EVENT_TABLE(DnDShapeDialog, wxDialog)
b0a6c154 857 EVT_BUTTON(Button_Colour, DnDShapeDialog::OnColour)
8e193f38
VZ
858END_EVENT_TABLE()
859
ae3dd4a5
VZ
860BEGIN_EVENT_TABLE(DnDCanvasBitmap, wxScrolledWindow)
861 EVT_PAINT(DnDCanvasBitmap::OnPaint)
862END_EVENT_TABLE()
863
864#ifdef USE_METAFILES
865BEGIN_EVENT_TABLE(DnDCanvasMetafile, wxScrolledWindow)
866 EVT_PAINT(DnDCanvasMetafile::OnPaint)
867END_EVENT_TABLE()
868#endif // USE_METAFILES
869
8e193f38
VZ
870// ============================================================================
871// implementation
872// ============================================================================
873
874// `Main program' equivalent, creating windows and returning main app frame
8bbe427f 875bool DnDApp::OnInit()
457814b5 876{
4693b20c 877#if USE_RESOURCES
ae125753
VZ
878 // load our ressources
879 wxPathList pathList;
9f84eccd 880 pathList.Add(_T("."));
ae125753 881#ifdef __WXMSW__
9f84eccd
MB
882 pathList.Add(_T("./Debug"));
883 pathList.Add(_T("./Release"));
ae125753
VZ
884#endif // wxMSW
885
9f84eccd 886 wxString path = pathList.FindValidPath(_T("dnd.wxr"));
ae125753
VZ
887 if ( !path )
888 {
4693b20c
MB
889 wxLogError(wxT("Can't find the resource file dnd.wxr in the current ")
890 wxT("directory, aborting."));
ae125753
VZ
891
892 return FALSE;
893 }
894
895 wxDefaultResourceTable->ParseResourceFile(path);
4693b20c 896#endif
ae125753 897
61b04ac6
VZ
898 // switch on trace messages
899#if defined(__WXGTK__)
900 wxLog::AddTraceMask(_T("clipboard"));
901#elif defined(__WXMSW__)
902 wxLog::AddTraceMask(wxTRACE_OleCalls);
903#endif
904
e2acb9ae
RR
905#if wxUSE_LIBPNG
906 wxImage::AddHandler( new wxPNGHandler );
907#endif
908
74d38ad8
VZ
909 // under X we usually want to use the primary selection by default (which
910 // is shared with other apps)
911 wxTheClipboard->UsePrimarySelection();
912
c50f1fb9
VZ
913 // create the main frame window
914 DnDFrame *frame = new DnDFrame((wxFrame *) NULL,
9f84eccd 915 _T("Drag-and-Drop/Clipboard wxWindows Sample"),
5a1c877f 916 10, 100, 650, 340);
457814b5 917
c50f1fb9
VZ
918 // activate it
919 frame->Show(TRUE);
457814b5 920
c50f1fb9 921 SetTopWindow(frame);
457814b5 922
c50f1fb9 923 return TRUE;
457814b5
JS
924}
925
9f84eccd 926DnDFrame::DnDFrame(wxFrame *frame, wxChar *title, int x, int y, int w, int h)
43d811ea 927 : wxFrame(frame, -1, title, wxPoint(x, y), wxSize(w, h)),
9f84eccd 928 m_strText(_T("wxWindows drag & drop works :-)"))
43d811ea 929
457814b5 930{
c50f1fb9
VZ
931 // frame icon and status bar
932 SetIcon(wxICON(mondrian));
933
934 CreateStatusBar();
935
936 // construct menu
937 wxMenu *file_menu = new wxMenu;
9f84eccd
MB
938 file_menu->Append(Menu_Drag, _T("&Test drag..."));
939 file_menu->AppendCheckItem(Menu_DragMoveDef, _T("&Move by default"));
940 file_menu->AppendCheckItem(Menu_DragMoveAllow, _T("&Allow moving"));
c50f1fb9 941 file_menu->AppendSeparator();
9f84eccd 942 file_menu->Append(Menu_NewFrame, _T("&New frame\tCtrl-N"));
d1b15f03 943 file_menu->AppendSeparator();
9f84eccd 944 file_menu->Append(Menu_Quit, _T("E&xit\tCtrl-Q"));
c50f1fb9
VZ
945
946 wxMenu *log_menu = new wxMenu;
9f84eccd 947 log_menu->Append(Menu_Clear, _T("Clear\tCtrl-L"));
c50f1fb9
VZ
948
949 wxMenu *help_menu = new wxMenu;
9f84eccd 950 help_menu->Append(Menu_Help, _T("&Help..."));
c50f1fb9 951 help_menu->AppendSeparator();
9f84eccd 952 help_menu->Append(Menu_About, _T("&About"));
c50f1fb9
VZ
953
954 wxMenu *clip_menu = new wxMenu;
9f84eccd
MB
955 clip_menu->Append(Menu_Copy, _T("&Copy text\tCtrl-C"));
956 clip_menu->Append(Menu_Paste, _T("&Paste text\tCtrl-V"));
e2acb9ae 957 clip_menu->AppendSeparator();
9f84eccd
MB
958 clip_menu->Append(Menu_CopyBitmap, _T("Copy &bitmap\tCtrl-Shift-C"));
959 clip_menu->Append(Menu_PasteBitmap, _T("Paste b&itmap\tCtrl-Shift-V"));
5a1c877f 960#ifdef USE_METAFILES
51edda6a 961 clip_menu->AppendSeparator();
9f84eccd 962 clip_menu->Append(Menu_PasteMFile, _T("Paste &metafile\tCtrl-M"));
5a1c877f
VZ
963#endif // USE_METAFILES
964 clip_menu->AppendSeparator();
9f84eccd 965 clip_menu->Append(Menu_CopyFiles, _T("Copy &files\tCtrl-F"));
8e193f38 966
c50f1fb9 967 wxMenuBar *menu_bar = new wxMenuBar;
9f84eccd
MB
968 menu_bar->Append(file_menu, _T("&File"));
969 menu_bar->Append(log_menu, _T("&Log"));
970 menu_bar->Append(clip_menu, _T("&Clipboard"));
971 menu_bar->Append(help_menu, _T("&Help"));
c50f1fb9
VZ
972
973 SetMenuBar(menu_bar);
974
975 // make a panel with 3 subwindows
976 wxPoint pos(0, 0);
977 wxSize size(400, 200);
978
9f84eccd 979 wxString strFile(_T("Drop files here!")), strText(_T("Drop text on me"));
c50f1fb9 980
d59ceba5
VZ
981 m_ctrlFile = new wxListBox(this, -1, pos, size, 1, &strFile,
982 wxLB_HSCROLL | wxLB_ALWAYS_SB );
983 m_ctrlText = new wxListBox(this, -1, pos, size, 1, &strText,
984 wxLB_HSCROLL | wxLB_ALWAYS_SB );
c50f1fb9 985
9f84eccd 986 m_ctrlLog = new wxTextCtrl(this, -1, _T(""), pos, size,
d59ceba5
VZ
987 wxTE_MULTILINE | wxTE_READONLY |
988 wxSUNKEN_BORDER );
8e193f38 989
61b04ac6 990 // redirect log messages to the text window
e2acb9ae
RR
991 m_pLog = new wxLogTextCtrl(m_ctrlLog);
992 m_pLogPrev = wxLog::SetActiveTarget(m_pLog);
993
444ad3a7 994 // associate drop targets with the controls
e2acb9ae
RR
995 m_ctrlFile->SetDropTarget(new DnDFile(m_ctrlFile));
996 m_ctrlText->SetDropTarget(new DnDText(m_ctrlText));
444ad3a7 997 m_ctrlLog->SetDropTarget(new URLDropTarget);
e2acb9ae
RR
998
999 wxLayoutConstraints *c;
1000
1001 // Top-left listbox
1002 c = new wxLayoutConstraints;
1003 c->left.SameAs(this, wxLeft);
1004 c->top.SameAs(this, wxTop);
1005 c->right.PercentOf(this, wxRight, 50);
1006 c->height.PercentOf(this, wxHeight, 30);
1007 m_ctrlFile->SetConstraints(c);
1008
1009 // Top-right listbox
1010 c = new wxLayoutConstraints;
1011 c->left.SameAs (m_ctrlFile, wxRight);
1012 c->top.SameAs (this, wxTop);
1013 c->right.SameAs (this, wxRight);
1014 c->height.PercentOf(this, wxHeight, 30);
1015 m_ctrlText->SetConstraints(c);
1016
1017 // Lower text control
1018 c = new wxLayoutConstraints;
1019 c->left.SameAs (this, wxLeft);
1020 c->right.SameAs (this, wxRight);
d59ceba5 1021 c->height.PercentOf(this, wxHeight, 50);
e2acb9ae
RR
1022 c->top.SameAs(m_ctrlText, wxBottom);
1023 m_ctrlLog->SetConstraints(c);
1024
1025 SetAutoLayout(TRUE);
2245b2b2
VZ
1026
1027 // copy data by default but allow moving it as well
1028 m_moveByDefault = FALSE;
1029 m_moveAllow = TRUE;
1030 menu_bar->Check(Menu_DragMoveAllow, TRUE);
457814b5
JS
1031}
1032
e2acb9ae 1033void DnDFrame::OnQuit(wxCommandEvent& WXUNUSED(event))
457814b5 1034{
c50f1fb9 1035 Close(TRUE);
457814b5
JS
1036}
1037
fcf689db
VZ
1038void DnDFrame::OnSize(wxSizeEvent& event)
1039{
1040 Refresh();
1041
1042 event.Skip();
1043}
1044
e2acb9ae 1045void DnDFrame::OnPaint(wxPaintEvent& WXUNUSED(event))
b527aac5 1046{
c50f1fb9
VZ
1047 int w = 0;
1048 int h = 0;
1049 GetClientSize( &w, &h );
8bbe427f 1050
c50f1fb9 1051 wxPaintDC dc(this);
82cf15a4 1052 // dc.Clear(); -- this kills wxGTK
9f84eccd
MB
1053 dc.SetFont( wxFont( 24, wxDECORATIVE, wxNORMAL, wxNORMAL, FALSE, _T("charter") ) );
1054 dc.DrawText( _T("Drag text from here!"), 100, h-50 );
e2acb9ae
RR
1055}
1056
2245b2b2
VZ
1057void DnDFrame::OnUpdateUIMoveByDefault(wxUpdateUIEvent& event)
1058{
1059 // only can move by default if moving is allowed at all
1060 event.Enable(m_moveAllow);
1061}
1062
d59ceba5 1063void DnDFrame::OnUpdateUIPasteText(wxUpdateUIEvent& event)
e2acb9ae 1064{
61b04ac6
VZ
1065#ifdef __WXDEBUG__
1066 // too many trace messages if we don't do it - this function is called
1067 // very often
1068 wxLogNull nolog;
1069#endif
1070
d59ceba5
VZ
1071 event.Enable( wxTheClipboard->IsSupported(wxDF_TEXT) );
1072}
e2acb9ae 1073
d59ceba5
VZ
1074void DnDFrame::OnUpdateUIPasteBitmap(wxUpdateUIEvent& event)
1075{
61b04ac6
VZ
1076#ifdef __WXDEBUG__
1077 // too many trace messages if we don't do it - this function is called
1078 // very often
1079 wxLogNull nolog;
1080#endif
1081
d59ceba5 1082 event.Enable( wxTheClipboard->IsSupported(wxDF_BITMAP) );
e2acb9ae
RR
1083}
1084
8e193f38
VZ
1085void DnDFrame::OnNewFrame(wxCommandEvent& WXUNUSED(event))
1086{
1087 (new DnDShapeFrame(this))->Show(TRUE);
1088
4693b20c 1089 wxLogStatus(this, wxT("Double click the new frame to select a shape for it"));
8e193f38
VZ
1090}
1091
e2acb9ae 1092void DnDFrame::OnDrag(wxCommandEvent& WXUNUSED(event))
43d811ea 1093{
c50f1fb9
VZ
1094 wxString strText = wxGetTextFromUser
1095 (
9f84eccd
MB
1096 _T("After you enter text in this dialog, press any mouse\n")
1097 _T("button in the bottom (empty) part of the frame and \n")
1098 _T("drag it anywhere - you will be in fact dragging the\n")
1099 _T("text object containing this text"),
1100 _T("Please enter some text"), m_strText, this
c50f1fb9
VZ
1101 );
1102
1103 m_strText = strText;
43d811ea
JS
1104}
1105
2245b2b2
VZ
1106void DnDFrame::OnDragMoveByDefault(wxCommandEvent& event)
1107{
1108 m_moveByDefault = event.IsChecked();
1109}
1110
1111void DnDFrame::OnDragMoveAllow(wxCommandEvent& event)
1112{
1113 m_moveAllow = event.IsChecked();
1114}
1115
e2acb9ae 1116void DnDFrame::OnAbout(wxCommandEvent& WXUNUSED(event))
457814b5 1117{
9f84eccd
MB
1118 wxMessageBox(_T("Drag-&-Drop Demo\n")
1119 _T("Please see \"Help|Help...\" for details\n")
1120 _T("Copyright (c) 1998 Vadim Zeitlin"),
1121 _T("About wxDnD"),
c50f1fb9
VZ
1122 wxICON_INFORMATION | wxOK,
1123 this);
457814b5
JS
1124}
1125
1126void DnDFrame::OnHelp(wxCommandEvent& /* event */)
1127{
c50f1fb9 1128 wxMessageDialog dialog(this,
9f84eccd
MB
1129 _T("This small program demonstrates drag & drop support in wxWindows. The program window\n")
1130 _T("consists of 3 parts: the bottom pane is for debug messages, so that you can see what's\n")
1131 _T("going on inside. The top part is split into 2 listboxes, the left one accepts files\n")
1132 _T("and the right one accepts text.\n")
1133 _T("\n")
1134 _T("To test wxDropTarget: open wordpad (write.exe), select some text in it and drag it to\n")
1135 _T("the right listbox (you'll notice the usual visual feedback, i.e. the cursor will change).\n")
1136 _T("Also, try dragging some files (you can select several at once) from Windows Explorer (or \n")
1137 _T("File Manager) to the left pane. Hold down Ctrl/Shift keys when you drop text (doesn't \n")
1138 _T("work with files) and see what changes.\n")
1139 _T("\n")
1140 _T("To test wxDropSource: just press any mouse button on the empty zone of the window and drag\n")
1141 _T("it to wordpad or any other droptarget accepting text (and of course you can just drag it\n")
1142 _T("to the right pane). Due to a lot of trace messages, the cursor might take some time to \n")
1143 _T("change, don't release the mouse button until it does. You can change the string being\n")
1144 _T("dragged in in \"File|Test drag...\" dialog.\n")
1145 _T("\n")
1146 _T("\n")
1147 _T("Please send all questions/bug reports/suggestions &c to \n")
1148 _T("Vadim Zeitlin <zeitlin@dptmaths.ens-cachan.fr>"),
1149 _T("wxDnD Help"));
c50f1fb9
VZ
1150
1151 dialog.ShowModal();
457814b5
JS
1152}
1153
e3e65dac 1154void DnDFrame::OnLogClear(wxCommandEvent& /* event */ )
43d811ea 1155{
c50f1fb9 1156 m_ctrlLog->Clear();
810b5e1f
VZ
1157 m_ctrlText->Clear();
1158 m_ctrlFile->Clear();
43d811ea
JS
1159}
1160
30dea054 1161void DnDFrame::OnLeftDown(wxMouseEvent &WXUNUSED(event) )
43d811ea 1162{
8e193f38 1163 if ( !m_strText.IsEmpty() )
c50f1fb9
VZ
1164 {
1165 // start drag operation
c50f1fb9 1166 wxTextDataObject textData(m_strText);
2d68e1b4
RR
1167/*
1168 wxFileDataObject textData;
810b5e1f
VZ
1169 textData.AddFile( "/file1.txt" );
1170 textData.AddFile( "/file2.txt" );
2d68e1b4 1171*/
f6bcfd97
BP
1172 wxDropSource source(textData, this,
1173 wxDROP_ICON(dnd_copy),
1174 wxDROP_ICON(dnd_move),
1175 wxDROP_ICON(dnd_none));
8e193f38 1176
2245b2b2
VZ
1177 int flags = 0;
1178 if ( m_moveByDefault )
1179 flags |= wxDrag_DefaultMove;
1180 else if ( m_moveAllow )
1181 flags |= wxDrag_AllowMove;
c50f1fb9 1182
9f84eccd 1183 const wxChar *pc;
2245b2b2 1184 switch ( source.DoDragDrop(flags) )
c50f1fb9 1185 {
9f84eccd
MB
1186 case wxDragError: pc = _T("Error!"); break;
1187 case wxDragNone: pc = _T("Nothing"); break;
1188 case wxDragCopy: pc = _T("Copied"); break;
1189 case wxDragMove: pc = _T("Moved"); break;
1190 case wxDragCancel: pc = _T("Cancelled"); break;
1191 default: pc = _T("Huh?"); break;
c50f1fb9
VZ
1192 }
1193
9f84eccd 1194 SetStatusText(wxString(_T("Drag result: ")) + pc);
43d811ea 1195 }
43d811ea
JS
1196}
1197
30dea054
RR
1198void DnDFrame::OnRightDown(wxMouseEvent &event )
1199{
9f84eccd 1200 wxMenu menu(_T("Dnd sample menu"));
8bbe427f 1201
9f84eccd 1202 menu.Append(Menu_Drag, _T("&Test drag..."));
51edda6a 1203 menu.AppendSeparator();
9f84eccd 1204 menu.Append(Menu_About, _T("&About"));
8e193f38 1205
51edda6a 1206 PopupMenu( &menu, event.GetX(), event.GetY() );
30dea054
RR
1207}
1208
457814b5
JS
1209DnDFrame::~DnDFrame()
1210{
c50f1fb9
VZ
1211 if ( m_pLog != NULL ) {
1212 if ( wxLog::SetActiveTarget(m_pLogPrev) == m_pLog )
1213 delete m_pLog;
1214 }
1215}
1216
1217// ---------------------------------------------------------------------------
e2acb9ae
RR
1218// bitmap clipboard
1219// ---------------------------------------------------------------------------
1220
1221void DnDFrame::OnCopyBitmap(wxCommandEvent& WXUNUSED(event))
1222{
1bd1d102
VZ
1223 // PNG support is not always compiled in under Windows, so use BMP there
1224#ifdef __WXMSW__
9f84eccd 1225 wxFileDialog dialog(this, _T("Open a BMP file"), _T(""), _T(""), _T("BMP files (*.bmp)|*.bmp"), 0);
1bd1d102 1226#else
9f84eccd 1227 wxFileDialog dialog(this, _T("Open a PNG file"), _T(""), _T(""), _T("PNG files (*.png)|*.png"), 0);
1bd1d102 1228#endif
e2acb9ae
RR
1229
1230 if (dialog.ShowModal() != wxID_OK)
8e193f38 1231 {
e2acb9ae
RR
1232 wxLogMessage( _T("Aborted file open") );
1233 return;
1234 }
8e193f38 1235
e2acb9ae 1236 if (dialog.GetPath().IsEmpty())
8e193f38 1237 {
e2acb9ae
RR
1238 wxLogMessage( _T("Returned empty string.") );
1239 return;
1240 }
8e193f38 1241
e2acb9ae
RR
1242 if (!wxFileExists(dialog.GetPath()))
1243 {
1244 wxLogMessage( _T("File doesn't exist.") );
1245 return;
1246 }
8e193f38 1247
e2acb9ae 1248 wxImage image;
8e193f38 1249 image.LoadFile( dialog.GetPath(),
1bd1d102
VZ
1250#ifdef __WXMSW__
1251 wxBITMAP_TYPE_BMP
1252#else
1253 wxBITMAP_TYPE_PNG
1254#endif
1255 );
e2acb9ae
RR
1256 if (!image.Ok())
1257 {
8e193f38 1258 wxLogError( _T("Invalid image file...") );
e2acb9ae
RR
1259 return;
1260 }
8e193f38
VZ
1261
1262 wxLogStatus( _T("Decoding image file...") );
e2acb9ae 1263 wxYield();
8e193f38 1264
368d59f0 1265 wxBitmap bitmap( image );
e2acb9ae
RR
1266
1267 if ( !wxTheClipboard->Open() )
1268 {
1269 wxLogError(_T("Can't open clipboard."));
1270
1271 return;
1272 }
1273
1274 wxLogMessage( _T("Creating wxBitmapDataObject...") );
1275 wxYield();
8e193f38 1276
e2acb9ae
RR
1277 if ( !wxTheClipboard->AddData(new wxBitmapDataObject(bitmap)) )
1278 {
1279 wxLogError(_T("Can't copy image to the clipboard."));
1280 }
1281 else
1282 {
1283 wxLogMessage(_T("Image has been put on the clipboard.") );
1284 wxLogMessage(_T("You can paste it now and look at it.") );
1285 }
1286
1287 wxTheClipboard->Close();
1288}
1289
1290void DnDFrame::OnPasteBitmap(wxCommandEvent& WXUNUSED(event))
1291{
1292 if ( !wxTheClipboard->Open() )
1293 {
1294 wxLogError(_T("Can't open clipboard."));
1295
1296 return;
1297 }
1298
1299 if ( !wxTheClipboard->IsSupported(wxDF_BITMAP) )
1300 {
1301 wxLogWarning(_T("No bitmap on clipboard"));
1302
1303 wxTheClipboard->Close();
1304 return;
1305 }
1306
1307 wxBitmapDataObject data;
79ec2ce2 1308 if ( !wxTheClipboard->GetData(data) )
e2acb9ae
RR
1309 {
1310 wxLogError(_T("Can't paste bitmap from the clipboard"));
1311 }
1312 else
1313 {
ae3dd4a5
VZ
1314 const wxBitmap& bmp = data.GetBitmap();
1315
1316 wxLogMessage(_T("Bitmap %dx%d pasted from the clipboard"),
1317 bmp.GetWidth(), bmp.GetHeight());
1318 ShowBitmap(bmp);
e2acb9ae
RR
1319 }
1320
1321 wxTheClipboard->Close();
1322}
1323
5a1c877f
VZ
1324#ifdef USE_METAFILES
1325
1326void DnDFrame::OnPasteMetafile(wxCommandEvent& WXUNUSED(event))
1327{
1328 if ( !wxTheClipboard->Open() )
1329 {
1330 wxLogError(_T("Can't open clipboard."));
1331
1332 return;
1333 }
1334
1335 if ( !wxTheClipboard->IsSupported(wxDF_METAFILE) )
1336 {
1337 wxLogWarning(_T("No metafile on clipboard"));
1338 }
1339 else
1340 {
1341 wxMetaFileDataObject data;
1342 if ( !wxTheClipboard->GetData(data) )
1343 {
1344 wxLogError(_T("Can't paste metafile from the clipboard"));
1345 }
1346 else
1347 {
ae3dd4a5 1348 const wxMetaFile& mf = data.GetMetafile();
5a1c877f 1349
ae3dd4a5
VZ
1350 wxLogMessage(_T("Metafile %dx%d pasted from the clipboard"),
1351 mf.GetWidth(), mf.GetHeight());
1352
1353 ShowMetaFile(mf);
5a1c877f
VZ
1354 }
1355 }
1356
1357 wxTheClipboard->Close();
1358}
1359
1360#endif // USE_METAFILES
1361
51edda6a
VZ
1362// ----------------------------------------------------------------------------
1363// file clipboard
1364// ----------------------------------------------------------------------------
1365
1366void DnDFrame::OnCopyFiles(wxCommandEvent& WXUNUSED(event))
1367{
1368#ifdef __WXMSW__
9f84eccd
MB
1369 wxFileDialog dialog(this, _T("Select a file to copy"), _T(""), _T(""),
1370 _T("All files (*.*)|*.*"), 0);
51edda6a 1371
3f2711d5
VZ
1372 wxArrayString filenames;
1373 while ( dialog.ShowModal() == wxID_OK )
51edda6a 1374 {
3f2711d5
VZ
1375 filenames.Add(dialog.GetPath());
1376 }
1377
1378 if ( !filenames.IsEmpty() )
1379 {
1380 wxFileDataObject *dobj = new wxFileDataObject;
1381 size_t count = filenames.GetCount();
1382 for ( size_t n = 0; n < count; n++ )
1383 {
1384 dobj->AddFile(filenames[n]);
1385 }
51edda6a
VZ
1386
1387 wxClipboardLocker locker;
1388 if ( !locker )
1389 {
4693b20c 1390 wxLogError(wxT("Can't open clipboard"));
51edda6a
VZ
1391 }
1392 else
1393 {
1394 if ( !wxTheClipboard->AddData(dobj) )
1395 {
4693b20c 1396 wxLogError(wxT("Can't copy file(s) to the clipboard"));
51edda6a
VZ
1397 }
1398 else
1399 {
4693b20c
MB
1400 wxLogStatus(this, wxT("%d file%s copied to the clipboard"),
1401 count, count == 1 ? wxT("") : wxT("s"));
51edda6a
VZ
1402 }
1403 }
1404 }
1405 else
1406 {
4693b20c 1407 wxLogStatus(this, wxT("Aborted"));
51edda6a
VZ
1408 }
1409#else // !MSW
4693b20c 1410 wxLogError(wxT("Sorry, not implemented"));
51edda6a
VZ
1411#endif // MSW/!MSW
1412}
1413
e2acb9ae
RR
1414// ---------------------------------------------------------------------------
1415// text clipboard
c50f1fb9
VZ
1416// ---------------------------------------------------------------------------
1417
1418void DnDFrame::OnCopy(wxCommandEvent& WXUNUSED(event))
1419{
1420 if ( !wxTheClipboard->Open() )
1421 {
b56baa2d 1422 wxLogError(_T("Can't open clipboard."));
c50f1fb9
VZ
1423
1424 return;
1425 }
1426
1427 if ( !wxTheClipboard->AddData(new wxTextDataObject(m_strText)) )
1428 {
b56baa2d 1429 wxLogError(_T("Can't copy data to the clipboard"));
c50f1fb9
VZ
1430 }
1431 else
1432 {
b56baa2d 1433 wxLogMessage(_T("Text '%s' put on the clipboard"), m_strText.c_str());
c50f1fb9
VZ
1434 }
1435
1436 wxTheClipboard->Close();
1437}
1438
1439void DnDFrame::OnPaste(wxCommandEvent& WXUNUSED(event))
1440{
1441 if ( !wxTheClipboard->Open() )
1442 {
b56baa2d 1443 wxLogError(_T("Can't open clipboard."));
c50f1fb9
VZ
1444
1445 return;
1446 }
1447
1448 if ( !wxTheClipboard->IsSupported(wxDF_TEXT) )
1449 {
b56baa2d 1450 wxLogWarning(_T("No text data on clipboard"));
c50f1fb9 1451
e2acb9ae 1452 wxTheClipboard->Close();
c50f1fb9
VZ
1453 return;
1454 }
1455
1456 wxTextDataObject text;
79ec2ce2 1457 if ( !wxTheClipboard->GetData(text) )
c50f1fb9 1458 {
b56baa2d 1459 wxLogError(_T("Can't paste data from the clipboard"));
c50f1fb9
VZ
1460 }
1461 else
1462 {
b56baa2d 1463 wxLogMessage(_T("Text '%s' pasted from the clipboard"),
c50f1fb9
VZ
1464 text.GetText().c_str());
1465 }
1466
1467 wxTheClipboard->Close();
457814b5
JS
1468}
1469
1470// ----------------------------------------------------------------------------
1471// Notifications called by the base class
1472// ----------------------------------------------------------------------------
e2acb9ae 1473
9e2896e5 1474bool DnDText::OnDropText(wxCoord, wxCoord, const wxString& text)
457814b5 1475{
9e2896e5 1476 m_pOwner->Append(text);
457814b5 1477
c50f1fb9 1478 return TRUE;
457814b5
JS
1479}
1480
9e2896e5 1481bool DnDFile::OnDropFiles(wxCoord, wxCoord, const wxArrayString& filenames)
457814b5 1482{
9e2896e5 1483 size_t nFiles = filenames.GetCount();
c50f1fb9
VZ
1484 wxString str;
1485 str.Printf( _T("%d files dropped"), nFiles);
1486 m_pOwner->Append(str);
1487 for ( size_t n = 0; n < nFiles; n++ ) {
9e2896e5 1488 m_pOwner->Append(filenames[n]);
c50f1fb9
VZ
1489 }
1490
1491 return TRUE;
457814b5 1492}
8e193f38
VZ
1493
1494// ----------------------------------------------------------------------------
1495// DnDShapeDialog
1496// ----------------------------------------------------------------------------
1497
1498DnDShapeDialog::DnDShapeDialog(wxFrame *parent, DnDShape *shape)
4693b20c
MB
1499#if !USE_RESOURCES
1500 :wxDialog( parent, 6001, wxT("Choose Shape"), wxPoint( 10, 10 ),
1501 wxSize( 40, 40 ),
1502 wxRAISED_BORDER|wxCAPTION|wxTHICK_FRAME|wxSYSTEM_MENU )
1503#endif
8e193f38
VZ
1504{
1505 m_shape = shape;
4693b20c 1506#if USE_RESOURCES
9f84eccd 1507 LoadFromResource(parent, _T("dialogShape"));
8e193f38 1508
9f84eccd
MB
1509 m_textX = (wxTextCtrl *)wxFindWindowByName(_T("textX"), this);
1510 m_textY = (wxTextCtrl *)wxFindWindowByName(_T("textY"), this);
1511 m_textW = (wxTextCtrl *)wxFindWindowByName(_T("textW"), this);
1512 m_textH = (wxTextCtrl *)wxFindWindowByName(_T("textH"), this);
8e193f38 1513
9f84eccd 1514 m_radio = (wxRadioBox *)wxFindWindowByName(_T("radio"), this);
4693b20c
MB
1515#else
1516 wxBoxSizer* topSizer = new wxBoxSizer( wxVERTICAL );
1517
1518 // radio box
1519 wxBoxSizer* shapesSizer = new wxBoxSizer( wxHORIZONTAL );
1520 const wxString choices[] = { wxT("None"), wxT("Triangle"),
1521 wxT("Rectangle"), wxT("Ellipse") };
1522
e6d318c2 1523 m_radio = new wxRadioBox( this, -1, wxT("&Shape"),
4693b20c
MB
1524 wxDefaultPosition, wxDefaultSize, 4, choices, 4,
1525 wxRA_SPECIFY_COLS );
1526 shapesSizer->Add( m_radio, 0, wxGROW|wxALL, 5 );
1527 topSizer->Add( shapesSizer, 0, wxALL, 2 );
1528
1529 // attributes
1530 wxStaticBox* box = new wxStaticBox( this, -1, wxT("&Attributes") );
1531 wxStaticBoxSizer* attrSizer = new wxStaticBoxSizer( box, wxHORIZONTAL );
1532 wxFlexGridSizer* xywhSizer = new wxFlexGridSizer( 4, 2 );
e6d318c2 1533
4693b20c
MB
1534 wxStaticText* st;
1535
1536 st = new wxStaticText( this, -1, wxT("Position &X:") );
1537 m_textX = new wxTextCtrl( this, -1, wxEmptyString, wxDefaultPosition,
1538 wxSize( 30, 20 ) );
e6d318c2
RD
1539 xywhSizer->Add( st, 1, wxGROW|wxALL, 2 );
1540 xywhSizer->Add( m_textX, 1, wxGROW|wxALL, 2 );
4693b20c
MB
1541
1542 st = new wxStaticText( this, -1, wxT("Size &width:") );
1543 m_textW = new wxTextCtrl( this, -1, wxEmptyString, wxDefaultPosition,
1544 wxSize( 30, 20 ) );
e6d318c2
RD
1545 xywhSizer->Add( st, 1, wxGROW|wxALL, 2 );
1546 xywhSizer->Add( m_textW, 1, wxGROW|wxALL, 2 );
4693b20c
MB
1547
1548 st = new wxStaticText( this, -1, wxT("&Y:") );
1549 m_textY = new wxTextCtrl( this, -1, wxEmptyString, wxDefaultPosition,
1550 wxSize( 30, 20 ) );
e6d318c2
RD
1551 xywhSizer->Add( st, 1, wxALL|wxALIGN_RIGHT, 2 );
1552 xywhSizer->Add( m_textY, 1, wxGROW|wxALL, 2 );
4693b20c
MB
1553
1554 st = new wxStaticText( this, -1, wxT("&height:") );
1555 m_textH = new wxTextCtrl( this, -1, wxEmptyString, wxDefaultPosition,
1556 wxSize( 30, 20 ) );
e6d318c2
RD
1557 xywhSizer->Add( st, 1, wxALL|wxALIGN_RIGHT, 2 );
1558 xywhSizer->Add( m_textH, 1, wxGROW|wxALL, 2 );
4693b20c
MB
1559
1560 wxButton* col = new wxButton( this, Button_Colour, wxT("&Colour...") );
1561 attrSizer->Add( xywhSizer, 1, wxGROW );
1562 attrSizer->Add( col, 0, wxALL|wxALIGN_CENTRE_VERTICAL, 2 );
1563 topSizer->Add( attrSizer, 0, wxGROW|wxALL, 5 );
1564
1565 // buttons
1566 wxBoxSizer* buttonSizer = new wxBoxSizer( wxHORIZONTAL );
1567 wxButton* bt;
1568 bt = new wxButton( this, wxID_OK, wxT("Ok") );
1569 buttonSizer->Add( bt, 0, wxALL, 2 );
1570 bt = new wxButton( this, wxID_CANCEL, wxT("Cancel") );
1571 buttonSizer->Add( bt, 0, wxALL, 2 );
1572 topSizer->Add( buttonSizer, 0, wxALL|wxALIGN_RIGHT, 2 );
1573
1574 SetAutoLayout( TRUE );
1575 SetSizer( topSizer );
1576 topSizer->Fit( this );
1577#endif
8e193f38
VZ
1578}
1579
1580DnDShape *DnDShapeDialog::GetShape() const
1581{
1582 switch ( m_shapeKind )
1583 {
1584 default:
1585 case DnDShape::None: return NULL;
1586 case DnDShape::Triangle: return new DnDTriangularShape(m_pos, m_size, m_col);
1587 case DnDShape::Rectangle: return new DnDRectangularShape(m_pos, m_size, m_col);
1588 case DnDShape::Ellipse: return new DnDEllipticShape(m_pos, m_size, m_col);
1589 }
1590}
1591
1592bool DnDShapeDialog::TransferDataToWindow()
1593{
a3e7d24d 1594
8e193f38
VZ
1595 if ( m_shape )
1596 {
1597 m_radio->SetSelection(m_shape->GetKind());
1598 m_pos = m_shape->GetPosition();
1599 m_size = m_shape->GetSize();
1600 m_col = m_shape->GetColour();
1601 }
1602 else
1603 {
1604 m_radio->SetSelection(DnDShape::None);
1605 m_pos = wxPoint(1, 1);
1606 m_size = wxSize(100, 100);
1607 }
1608
1609 m_textX->SetValue(wxString() << m_pos.x);
1610 m_textY->SetValue(wxString() << m_pos.y);
1611 m_textW->SetValue(wxString() << m_size.x);
1612 m_textH->SetValue(wxString() << m_size.y);
1613
1614 return TRUE;
1615}
1616
1617bool DnDShapeDialog::TransferDataFromWindow()
1618{
1619 m_shapeKind = (DnDShape::Kind)m_radio->GetSelection();
1620
4693b20c
MB
1621 m_pos.x = wxAtoi(m_textX->GetValue());
1622 m_pos.y = wxAtoi(m_textY->GetValue());
1623 m_size.x = wxAtoi(m_textW->GetValue());
1624 m_size.y = wxAtoi(m_textH->GetValue());
8e193f38
VZ
1625
1626 if ( !m_pos.x || !m_pos.y || !m_size.x || !m_size.y )
1627 {
9f84eccd
MB
1628 wxMessageBox(_T("All sizes and positions should be non null!"),
1629 _T("Invalid shape"), wxICON_HAND | wxOK, this);
8e193f38
VZ
1630
1631 return FALSE;
1632 }
1633
1634 return TRUE;
1635}
1636
1637void DnDShapeDialog::OnColour(wxCommandEvent& WXUNUSED(event))
1638{
1639 wxColourData data;
1640 data.SetChooseFull(TRUE);
1641 for (int i = 0; i < 16; i++)
1642 {
1643 wxColour colour(i*16, i*16, i*16);
1644 data.SetCustomColour(i, colour);
1645 }
1646
1647 wxColourDialog dialog(this, &data);
1648 if ( dialog.ShowModal() == wxID_OK )
1649 {
1650 m_col = dialog.GetColourData().GetColour();
1651 }
1652}
1653
1654// ----------------------------------------------------------------------------
1655// DnDShapeFrame
1656// ----------------------------------------------------------------------------
1657
d59ceba5
VZ
1658DnDShapeFrame *DnDShapeFrame::ms_lastDropTarget = NULL;
1659
8e193f38 1660DnDShapeFrame::DnDShapeFrame(wxFrame *parent)
9f84eccd 1661 : wxFrame(parent, -1, _T("Shape Frame"),
8e193f38
VZ
1662 wxDefaultPosition, wxSize(250, 150))
1663{
8e193f38
VZ
1664 CreateStatusBar();
1665
d59ceba5 1666 wxMenu *menuShape = new wxMenu;
9f84eccd
MB
1667 menuShape->Append(Menu_Shape_New, _T("&New default shape\tCtrl-S"));
1668 menuShape->Append(Menu_Shape_Edit, _T("&Edit shape\tCtrl-E"));
d59ceba5 1669 menuShape->AppendSeparator();
9f84eccd 1670 menuShape->Append(Menu_Shape_Clear, _T("&Clear shape\tCtrl-L"));
d59ceba5
VZ
1671
1672 wxMenu *menuClipboard = new wxMenu;
9f84eccd
MB
1673 menuClipboard->Append(Menu_ShapeClipboard_Copy, _T("&Copy\tCtrl-C"));
1674 menuClipboard->Append(Menu_ShapeClipboard_Paste, _T("&Paste\tCtrl-V"));
d59ceba5
VZ
1675
1676 wxMenuBar *menubar = new wxMenuBar;
9f84eccd
MB
1677 menubar->Append(menuShape, _T("&Shape"));
1678 menubar->Append(menuClipboard, _T("&Clipboard"));
d59ceba5
VZ
1679
1680 SetMenuBar(menubar);
1681
9f84eccd 1682 SetStatusText(_T("Press Ctrl-S to create a new shape"));
8e193f38
VZ
1683
1684 SetDropTarget(new DnDShapeDropTarget(this));
1685
1686 m_shape = NULL;
f6bcfd97 1687
572c6194 1688 SetBackgroundColour(*wxWHITE);
8e193f38
VZ
1689}
1690
1691DnDShapeFrame::~DnDShapeFrame()
1692{
f6bcfd97 1693 if (m_shape)
572c6194 1694 delete m_shape;
8e193f38
VZ
1695}
1696
1697void DnDShapeFrame::SetShape(DnDShape *shape)
1698{
f6bcfd97 1699 if (m_shape)
572c6194 1700 delete m_shape;
8e193f38
VZ
1701 m_shape = shape;
1702 Refresh();
1703}
1704
1705// callbacks
1706void DnDShapeFrame::OnDrag(wxMouseEvent& event)
1707{
1708 if ( !m_shape )
1709 {
1710 event.Skip();
1711
1712 return;
1713 }
1714
1715 // start drag operation
1716 DnDShapeDataObject shapeData(m_shape);
2d93e133 1717 wxDropSource source(shapeData, this);
8e193f38 1718
9f84eccd 1719 const wxChar *pc = NULL;
8e193f38
VZ
1720 switch ( source.DoDragDrop(TRUE) )
1721 {
1722 default:
1723 case wxDragError:
4693b20c 1724 wxLogError(wxT("An error occured during drag and drop operation"));
8e193f38
VZ
1725 break;
1726
1727 case wxDragNone:
9f84eccd 1728 SetStatusText(_T("Nothing happened"));
8e193f38
VZ
1729 break;
1730
1731 case wxDragCopy:
9f84eccd 1732 pc = _T("copied");
8e193f38
VZ
1733 break;
1734
1735 case wxDragMove:
9f84eccd 1736 pc = _T("moved");
d59ceba5
VZ
1737 if ( ms_lastDropTarget != this )
1738 {
1739 // don't delete the shape if we dropped it on ourselves!
1740 SetShape(NULL);
1741 }
8e193f38
VZ
1742 break;
1743
1744 case wxDragCancel:
9f84eccd 1745 SetStatusText(_T("Drag and drop operation cancelled"));
8e193f38
VZ
1746 break;
1747 }
1748
1749 if ( pc )
1750 {
9f84eccd 1751 SetStatusText(wxString(_T("Shape successfully ")) + pc);
8e193f38
VZ
1752 }
1753 //else: status text already set
1754}
1755
90e12284 1756void DnDShapeFrame::OnDrop(wxCoord x, wxCoord y, DnDShape *shape)
9e2896e5
VZ
1757{
1758 ms_lastDropTarget = this;
1759
90e12284 1760 wxPoint pt(x, y);
90e12284 1761
9e2896e5 1762 wxString s;
508d586e 1763 s.Printf(wxT("Shape dropped at (%d, %d)"), pt.x, pt.y);
9e2896e5
VZ
1764 SetStatusText(s);
1765
90e12284 1766 shape->Move(pt);
9e2896e5
VZ
1767 SetShape(shape);
1768}
1769
d59ceba5 1770void DnDShapeFrame::OnEditShape(wxCommandEvent& event)
8e193f38
VZ
1771{
1772 DnDShapeDialog dlg(this, m_shape);
1773 if ( dlg.ShowModal() == wxID_OK )
1774 {
1775 SetShape(dlg.GetShape());
1776
1777 if ( m_shape )
1778 {
9f84eccd 1779 SetStatusText(_T("You can now drag the shape to another frame"));
8e193f38
VZ
1780 }
1781 }
1782}
1783
d59ceba5
VZ
1784void DnDShapeFrame::OnNewShape(wxCommandEvent& event)
1785{
1786 SetShape(new DnDEllipticShape(wxPoint(10, 10), wxSize(80, 60), *wxRED));
1787
9f84eccd 1788 SetStatusText(_T("You can now drag the shape to another frame"));
d59ceba5
VZ
1789}
1790
1791void DnDShapeFrame::OnClearShape(wxCommandEvent& event)
1792{
1793 SetShape(NULL);
1794}
1795
1796void DnDShapeFrame::OnCopyShape(wxCommandEvent& event)
1797{
1798 if ( m_shape )
ae125753
VZ
1799 {
1800 wxClipboardLocker clipLocker;
1801 if ( !clipLocker )
1802 {
4693b20c 1803 wxLogError(wxT("Can't open the clipboard"));
ae125753
VZ
1804
1805 return;
1806 }
1807
d59ceba5 1808 wxTheClipboard->AddData(new DnDShapeDataObject(m_shape));
ae125753 1809 }
d59ceba5
VZ
1810}
1811
1812void DnDShapeFrame::OnPasteShape(wxCommandEvent& event)
1813{
ae125753
VZ
1814 wxClipboardLocker clipLocker;
1815 if ( !clipLocker )
1816 {
4693b20c 1817 wxLogError(wxT("Can't open the clipboard"));
ae125753
VZ
1818
1819 return;
1820 }
1821
d59ceba5 1822 DnDShapeDataObject shapeDataObject(NULL);
79ec2ce2 1823 if ( wxTheClipboard->GetData(shapeDataObject) )
d59ceba5
VZ
1824 {
1825 SetShape(shapeDataObject.GetShape());
1826 }
1827 else
1828 {
4693b20c 1829 wxLogStatus(wxT("No shape on the clipboard"));
d59ceba5
VZ
1830 }
1831}
1832
1833void DnDShapeFrame::OnUpdateUICopy(wxUpdateUIEvent& event)
1834{
1835 event.Enable( m_shape != NULL );
1836}
1837
1838void DnDShapeFrame::OnUpdateUIPaste(wxUpdateUIEvent& event)
1839{
1840 event.Enable( wxTheClipboard->IsSupported(wxDataFormat(shapeFormatId)) );
1841}
1842
8e193f38
VZ
1843void DnDShapeFrame::OnPaint(wxPaintEvent& event)
1844{
1845 if ( m_shape )
79ec2ce2
VZ
1846 {
1847 wxPaintDC dc(this);
1848
1849 m_shape->Draw(dc);
1850 }
8e193f38 1851 else
79ec2ce2 1852 {
8e193f38 1853 event.Skip();
79ec2ce2 1854 }
8e193f38
VZ
1855}
1856
8e193f38
VZ
1857// ----------------------------------------------------------------------------
1858// DnDShape
1859// ----------------------------------------------------------------------------
1860
1861DnDShape *DnDShape::New(const void *buf)
1862{
1863 const ShapeDump& dump = *(const ShapeDump *)buf;
1864 switch ( dump.k )
1865 {
1866 case Triangle:
1867 return new DnDTriangularShape(wxPoint(dump.x, dump.y),
1868 wxSize(dump.w, dump.h),
1869 wxColour(dump.r, dump.g, dump.b));
1870
1871 case Rectangle:
1872 return new DnDRectangularShape(wxPoint(dump.x, dump.y),
1873 wxSize(dump.w, dump.h),
1874 wxColour(dump.r, dump.g, dump.b));
1875
1876 case Ellipse:
1877 return new DnDEllipticShape(wxPoint(dump.x, dump.y),
1878 wxSize(dump.w, dump.h),
1879 wxColour(dump.r, dump.g, dump.b));
1880
1881 default:
4693b20c 1882 wxFAIL_MSG(wxT("invalid shape!"));
8e193f38
VZ
1883 return NULL;
1884 }
1885}
1886
1887// ----------------------------------------------------------------------------
1888// DnDShapeDataObject
1889// ----------------------------------------------------------------------------
1890
5a1c877f
VZ
1891#ifdef USE_METAFILES
1892
1893void DnDShapeDataObject::CreateMetaFile() const
1894{
ae3dd4a5
VZ
1895 wxPoint pos = m_shape->GetPosition();
1896 wxSize size = m_shape->GetSize();
1897
1898 wxMetaFileDC dcMF(wxEmptyString, pos.x + size.x, pos.y + size.y);
5a1c877f
VZ
1899
1900 m_shape->Draw(dcMF);
1901
1902 wxMetafile *mf = dcMF.Close();
1903
5a1c877f
VZ
1904 DnDShapeDataObject *self = (DnDShapeDataObject *)this; // const_cast
1905 self->m_dobjMetaFile.SetMetafile(*mf);
1906 self->m_hasMetaFile = TRUE;
ae3dd4a5
VZ
1907
1908 delete mf;
5a1c877f
VZ
1909}
1910
1911#endif // Windows
1912
8e193f38
VZ
1913void DnDShapeDataObject::CreateBitmap() const
1914{
d59ceba5
VZ
1915 wxPoint pos = m_shape->GetPosition();
1916 wxSize size = m_shape->GetSize();
1917 int x = pos.x + size.x,
1918 y = pos.y + size.y;
1919 wxBitmap bitmap(x, y);
8e193f38
VZ
1920 wxMemoryDC dc;
1921 dc.SelectObject(bitmap);
a60b1f5d 1922 dc.SetBrush(wxBrush(wxT("white"), wxSOLID));
d59ceba5 1923 dc.Clear();
8e193f38
VZ
1924 m_shape->Draw(dc);
1925 dc.SelectObject(wxNullBitmap);
1926
1927 DnDShapeDataObject *self = (DnDShapeDataObject *)this; // const_cast
5a1c877f 1928 self->m_dobjBitmap.SetBitmap(bitmap);
8e193f38
VZ
1929 self->m_hasBitmap = TRUE;
1930}
1931
ae3dd4a5
VZ
1932// ----------------------------------------------------------------------------
1933// global functions
1934// ----------------------------------------------------------------------------
1935
1936static void ShowBitmap(const wxBitmap& bitmap)
1937{
1938 wxFrame *frame = new wxFrame(NULL, -1, _T("Bitmap view"));
1939 frame->CreateStatusBar();
1940 DnDCanvasBitmap *canvas = new DnDCanvasBitmap(frame);
1941 canvas->SetBitmap(bitmap);
1942
1943 int w = bitmap.GetWidth(),
1944 h = bitmap.GetHeight();
1945 frame->SetStatusText(wxString::Format(_T("%dx%d"), w, h));
1946
1947 frame->SetClientSize(w > 100 ? 100 : w, h > 100 ? 100 : h);
e680a378 1948 frame->Show(TRUE);
ae3dd4a5
VZ
1949}
1950
1951#ifdef USE_METAFILES
1952
1953static void ShowMetaFile(const wxMetaFile& metafile)
1954{
1955 wxFrame *frame = new wxFrame(NULL, -1, _T("Metafile view"));
1956 frame->CreateStatusBar();
1957 DnDCanvasMetafile *canvas = new DnDCanvasMetafile(frame);
1958 canvas->SetMetafile(metafile);
1959
1960 wxSize size = metafile.GetSize();
1961 frame->SetStatusText(wxString::Format(_T("%dx%d"), size.x, size.y));
1962
1963 frame->SetClientSize(size.x > 100 ? 100 : size.x,
1964 size.y > 100 ? 100 : size.y);
1965 frame->Show();
1966}
1967
1968#endif // USE_METAFILES