+ str.Printf( _T("%d files dropped"), (int)nFiles);
+
+ if (m_pOwner != NULL)
+ {
+ m_pOwner->Append(str);
+ for ( size_t n = 0; n < nFiles; n++ )
+ m_pOwner->Append(filenames[n]);
+ }
+
+ return true;
+}
+
+// ----------------------------------------------------------------------------
+// DnDShapeDialog
+// ----------------------------------------------------------------------------
+
+DnDShapeDialog::DnDShapeDialog(wxFrame *parent, DnDShape *shape)
+ :wxDialog( parent, 6001, wxT("Choose Shape"), wxPoint( 10, 10 ),
+ wxSize( 40, 40 ),
+ wxDEFAULT_DIALOG_STYLE | wxRAISED_BORDER | wxRESIZE_BORDER )
+{
+ m_shape = shape;
+ wxBoxSizer* topSizer = new wxBoxSizer( wxVERTICAL );
+
+ // radio box
+ wxBoxSizer* shapesSizer = new wxBoxSizer( wxHORIZONTAL );
+ const wxString choices[] = { wxT("None"), wxT("Triangle"),
+ wxT("Rectangle"), wxT("Ellipse") };
+
+ m_radio = new wxRadioBox( this, wxID_ANY, wxT("&Shape"),
+ wxDefaultPosition, wxDefaultSize, 4, choices, 4,
+ wxRA_SPECIFY_COLS );
+ shapesSizer->Add( m_radio, 0, wxGROW|wxALL, 5 );
+ topSizer->Add( shapesSizer, 0, wxALL, 2 );
+
+ // attributes
+ wxStaticBox* box = new wxStaticBox( this, wxID_ANY, wxT("&Attributes") );
+ wxStaticBoxSizer* attrSizer = new wxStaticBoxSizer( box, wxHORIZONTAL );
+ wxFlexGridSizer* xywhSizer = new wxFlexGridSizer( 4, 2 );
+
+ wxStaticText* st;
+
+ st = new wxStaticText( this, wxID_ANY, wxT("Position &X:") );
+ m_textX = new wxTextCtrl( this, wxID_ANY, wxEmptyString, wxDefaultPosition,
+ wxSize( 30, 20 ) );
+ xywhSizer->Add( st, 1, wxGROW|wxALL, 2 );
+ xywhSizer->Add( m_textX, 1, wxGROW|wxALL, 2 );
+
+ st = new wxStaticText( this, wxID_ANY, wxT("Size &width:") );
+ m_textW = new wxTextCtrl( this, wxID_ANY, wxEmptyString, wxDefaultPosition,
+ wxSize( 30, 20 ) );
+ xywhSizer->Add( st, 1, wxGROW|wxALL, 2 );
+ xywhSizer->Add( m_textW, 1, wxGROW|wxALL, 2 );
+
+ st = new wxStaticText( this, wxID_ANY, wxT("&Y:") );
+ m_textY = new wxTextCtrl( this, wxID_ANY, wxEmptyString, wxDefaultPosition,
+ wxSize( 30, 20 ) );
+ xywhSizer->Add( st, 1, wxALL|wxALIGN_RIGHT, 2 );
+ xywhSizer->Add( m_textY, 1, wxGROW|wxALL, 2 );
+
+ st = new wxStaticText( this, wxID_ANY, wxT("&height:") );
+ m_textH = new wxTextCtrl( this, wxID_ANY, wxEmptyString, wxDefaultPosition,
+ wxSize( 30, 20 ) );
+ xywhSizer->Add( st, 1, wxALL|wxALIGN_RIGHT, 2 );
+ xywhSizer->Add( m_textH, 1, wxGROW|wxALL, 2 );
+
+ wxButton* col = new wxButton( this, Button_Colour, wxT("&Colour...") );
+ attrSizer->Add( xywhSizer, 1, wxGROW );
+ attrSizer->Add( col, 0, wxALL|wxALIGN_CENTRE_VERTICAL, 2 );
+ topSizer->Add( attrSizer, 0, wxGROW|wxALL, 5 );
+
+ // buttons
+ wxBoxSizer* buttonSizer = new wxBoxSizer( wxHORIZONTAL );
+ wxButton* bt;
+ bt = new wxButton( this, wxID_OK, wxT("Ok") );
+ buttonSizer->Add( bt, 0, wxALL, 2 );
+ bt = new wxButton( this, wxID_CANCEL, wxT("Cancel") );
+ buttonSizer->Add( bt, 0, wxALL, 2 );
+ topSizer->Add( buttonSizer, 0, wxALL|wxALIGN_RIGHT, 2 );
+
+ SetSizerAndFit( topSizer );
+}
+
+DnDShape *DnDShapeDialog::GetShape() const
+{
+ switch ( m_shapeKind )
+ {
+ default:
+ case DnDShape::None: return NULL;
+ case DnDShape::Triangle: return new DnDTriangularShape(m_pos, m_size, m_col);
+ case DnDShape::Rectangle: return new DnDRectangularShape(m_pos, m_size, m_col);
+ case DnDShape::Ellipse: return new DnDEllipticShape(m_pos, m_size, m_col);
+ }
+}
+
+bool DnDShapeDialog::TransferDataToWindow()
+{
+
+ if ( m_shape )
+ {
+ m_radio->SetSelection(m_shape->GetKind());
+ m_pos = m_shape->GetPosition();
+ m_size = m_shape->GetSize();
+ m_col = m_shape->GetColour();
+ }
+ else
+ {
+ m_radio->SetSelection(DnDShape::None);
+ m_pos = wxPoint(1, 1);
+ m_size = wxSize(100, 100);
+ }
+
+ m_textX->SetValue(wxString() << m_pos.x);
+ m_textY->SetValue(wxString() << m_pos.y);
+ m_textW->SetValue(wxString() << m_size.x);
+ m_textH->SetValue(wxString() << m_size.y);
+
+ return true;
+}
+
+bool DnDShapeDialog::TransferDataFromWindow()
+{
+ m_shapeKind = (DnDShape::Kind)m_radio->GetSelection();
+
+ m_pos.x = wxAtoi(m_textX->GetValue());
+ m_pos.y = wxAtoi(m_textY->GetValue());
+ m_size.x = wxAtoi(m_textW->GetValue());
+ m_size.y = wxAtoi(m_textH->GetValue());
+
+ if ( !m_pos.x || !m_pos.y || !m_size.x || !m_size.y )
+ {
+ wxMessageBox(_T("All sizes and positions should be non null!"),
+ _T("Invalid shape"), wxICON_HAND | wxOK, this);
+
+ return false;
+ }
+
+ return true;
+}
+
+void DnDShapeDialog::OnColour(wxCommandEvent& WXUNUSED(event))
+{
+ wxColourData data;
+ data.SetChooseFull(true);
+ for (int i = 0; i < 16; i++)
+ {
+ wxColour colour((unsigned char)(i*16), (unsigned char)(i*16), (unsigned char)(i*16));
+ data.SetCustomColour(i, colour);
+ }
+
+ wxColourDialog dialog(this, &data);
+ if ( dialog.ShowModal() == wxID_OK )
+ {
+ m_col = dialog.GetColourData().GetColour();
+ }
+}
+
+// ----------------------------------------------------------------------------
+// DnDShapeFrame
+// ----------------------------------------------------------------------------
+
+DnDShapeFrame *DnDShapeFrame::ms_lastDropTarget = NULL;
+
+DnDShapeFrame::DnDShapeFrame(wxFrame *parent)
+ : wxFrame(parent, wxID_ANY, _T("Shape Frame"))
+{
+#if wxUSE_STATUSBAR
+ CreateStatusBar();
+#endif // wxUSE_STATUSBAR
+
+ wxMenu *menuShape = new wxMenu;
+ menuShape->Append(Menu_Shape_New, _T("&New default shape\tCtrl-S"));
+ menuShape->Append(Menu_Shape_Edit, _T("&Edit shape\tCtrl-E"));
+ menuShape->AppendSeparator();
+ menuShape->Append(Menu_Shape_Clear, _T("&Clear shape\tCtrl-L"));
+
+ wxMenu *menuClipboard = new wxMenu;
+ menuClipboard->Append(Menu_ShapeClipboard_Copy, _T("&Copy\tCtrl-C"));
+ menuClipboard->Append(Menu_ShapeClipboard_Paste, _T("&Paste\tCtrl-V"));
+
+ wxMenuBar *menubar = new wxMenuBar;
+ menubar->Append(menuShape, _T("&Shape"));
+ menubar->Append(menuClipboard, _T("&Clipboard"));
+
+ SetMenuBar(menubar);
+
+#if wxUSE_STATUSBAR
+ SetStatusText(_T("Press Ctrl-S to create a new shape"));
+#endif // wxUSE_STATUSBAR
+
+ SetDropTarget(new DnDShapeDropTarget(this));
+
+ m_shape = NULL;
+
+ SetBackgroundColour(*wxWHITE);
+}
+
+DnDShapeFrame::~DnDShapeFrame()
+{
+ if (m_shape)
+ delete m_shape;
+}
+
+void DnDShapeFrame::SetShape(DnDShape *shape)
+{
+ if (m_shape)
+ delete m_shape;
+ m_shape = shape;
+ Refresh();
+}
+
+// callbacks
+void DnDShapeFrame::OnDrag(wxMouseEvent& event)
+{
+ if ( !m_shape )
+ {
+ event.Skip();
+
+ return;
+ }
+
+ // start drag operation
+ DnDShapeDataObject shapeData(m_shape);
+ wxDropSource source(shapeData, this);
+
+ const wxChar *pc = NULL;
+ switch ( source.DoDragDrop(true) )
+ {
+ default:
+ case wxDragError:
+ wxLogError(wxT("An error occurred during drag and drop operation"));
+ break;
+
+ case wxDragNone:
+#if wxUSE_STATUSBAR
+ SetStatusText(_T("Nothing happened"));
+#endif // wxUSE_STATUSBAR
+ break;
+
+ case wxDragCopy:
+ pc = _T("copied");
+ break;
+
+ case wxDragMove:
+ pc = _T("moved");
+ if ( ms_lastDropTarget != this )
+ {
+ // don't delete the shape if we dropped it on ourselves!
+ SetShape(NULL);
+ }
+ break;
+
+ case wxDragCancel:
+#if wxUSE_STATUSBAR
+ SetStatusText(_T("Drag and drop operation cancelled"));
+#endif // wxUSE_STATUSBAR
+ break;
+ }
+
+ if ( pc )
+ {
+#if wxUSE_STATUSBAR
+ SetStatusText(wxString(_T("Shape successfully ")) + pc);
+#endif // wxUSE_STATUSBAR
+ }
+ //else: status text already set
+}
+
+void DnDShapeFrame::OnDrop(wxCoord x, wxCoord y, DnDShape *shape)
+{
+ ms_lastDropTarget = this;
+
+ wxPoint pt(x, y);
+
+#if wxUSE_STATUSBAR
+ wxString s;
+ s.Printf(wxT("Shape dropped at (%d, %d)"), pt.x, pt.y);
+ SetStatusText(s);
+#endif // wxUSE_STATUSBAR
+
+ shape->Move(pt);
+ SetShape(shape);
+}
+
+void DnDShapeFrame::OnEditShape(wxCommandEvent& WXUNUSED(event))
+{
+ DnDShapeDialog dlg(this, m_shape);
+ if ( dlg.ShowModal() == wxID_OK )
+ {
+ SetShape(dlg.GetShape());
+
+#if wxUSE_STATUSBAR
+ if ( m_shape )
+ {
+ SetStatusText(_T("You can now drag the shape to another frame"));
+ }
+#endif // wxUSE_STATUSBAR
+ }
+}
+
+void DnDShapeFrame::OnNewShape(wxCommandEvent& WXUNUSED(event))
+{
+ SetShape(new DnDEllipticShape(wxPoint(10, 10), wxSize(80, 60), *wxRED));
+
+#if wxUSE_STATUSBAR
+ SetStatusText(_T("You can now drag the shape to another frame"));
+#endif // wxUSE_STATUSBAR
+}
+
+void DnDShapeFrame::OnClearShape(wxCommandEvent& WXUNUSED(event))
+{
+ SetShape(NULL);
+}
+
+void DnDShapeFrame::OnCopyShape(wxCommandEvent& WXUNUSED(event))
+{
+ if ( m_shape )
+ {
+ wxClipboardLocker clipLocker;
+ if ( !clipLocker )
+ {
+ wxLogError(wxT("Can't open the clipboard"));
+
+ return;
+ }
+
+ wxTheClipboard->AddData(new DnDShapeDataObject(m_shape));
+ }
+}
+
+void DnDShapeFrame::OnPasteShape(wxCommandEvent& WXUNUSED(event))
+{
+ wxClipboardLocker clipLocker;
+ if ( !clipLocker )
+ {
+ wxLogError(wxT("Can't open the clipboard"));
+
+ return;
+ }
+
+ DnDShapeDataObject shapeDataObject(NULL);
+ if ( wxTheClipboard->GetData(shapeDataObject) )
+ {
+ SetShape(shapeDataObject.GetShape());
+ }
+ else
+ {
+ wxLogStatus(wxT("No shape on the clipboard"));
+ }
+}
+
+void DnDShapeFrame::OnUpdateUICopy(wxUpdateUIEvent& event)
+{
+ event.Enable( m_shape != NULL );
+}
+
+void DnDShapeFrame::OnUpdateUIPaste(wxUpdateUIEvent& event)
+{
+ event.Enable( wxTheClipboard->IsSupported(wxDataFormat(shapeFormatId)) );
+}
+
+void DnDShapeFrame::OnPaint(wxPaintEvent& event)
+{
+ if ( m_shape )
+ {
+ wxPaintDC dc(this);
+
+ m_shape->Draw(dc);
+ }
+ else
+ {
+ event.Skip();
+ }
+}
+
+// ----------------------------------------------------------------------------
+// DnDShape
+// ----------------------------------------------------------------------------
+
+DnDShape *DnDShape::New(const void *buf)
+{
+ const ShapeDump& dump = *(const ShapeDump *)buf;
+ switch ( dump.k )
+ {
+ case Triangle:
+ return new DnDTriangularShape(wxPoint(dump.x, dump.y),
+ wxSize(dump.w, dump.h),
+ wxColour(dump.r, dump.g, dump.b));
+
+ case Rectangle:
+ return new DnDRectangularShape(wxPoint(dump.x, dump.y),
+ wxSize(dump.w, dump.h),
+ wxColour(dump.r, dump.g, dump.b));
+
+ case Ellipse:
+ return new DnDEllipticShape(wxPoint(dump.x, dump.y),
+ wxSize(dump.w, dump.h),
+ wxColour(dump.r, dump.g, dump.b));
+
+ default:
+ wxFAIL_MSG(wxT("invalid shape!"));
+ return NULL;