1 ///////////////////////////////////////////////////////////////////////////////
2 // Name: generic/wizard.cpp
3 // Purpose: generic implementation of wxWizard class
4 // Author: Vadim Zeitlin
8 // Copyright: (c) 1999 Vadim Zeitlin <zeitlin@dptmaths.ens-cachan.fr>
9 // Licence: wxWindows license
10 ///////////////////////////////////////////////////////////////////////////////
12 // ============================================================================
14 // ============================================================================
16 // ----------------------------------------------------------------------------
18 // ----------------------------------------------------------------------------
21 #pragma implementation ".h"
24 // For compilers that support precompilation, includes "wx.h".
25 #include "wx/wxprec.h"
34 #include "wx/dynarray.h"
36 #include "wx/statbmp.h"
37 #include "wx/button.h"
40 #include "wx/statline.h"
42 #include "wx/wizard.h"
44 // ----------------------------------------------------------------------------
46 // ----------------------------------------------------------------------------
48 WX_DEFINE_ARRAY(wxPanel
*, wxArrayPages
);
50 // ----------------------------------------------------------------------------
51 // event tables and such
52 // ----------------------------------------------------------------------------
54 DEFINE_EVENT_TYPE(wxEVT_WIZARD_PAGE_CHANGED
)
55 DEFINE_EVENT_TYPE(wxEVT_WIZARD_PAGE_CHANGING
)
56 DEFINE_EVENT_TYPE(wxEVT_WIZARD_CANCEL
)
58 BEGIN_EVENT_TABLE(wxWizard
, wxDialog
)
59 EVT_BUTTON(wxID_CANCEL
, wxWizard::OnCancel
)
60 EVT_BUTTON(-1, wxWizard::OnBackOrNext
)
63 IMPLEMENT_DYNAMIC_CLASS(wxWizard
, wxDialog
)
64 IMPLEMENT_ABSTRACT_CLASS(wxWizardPage
, wxPanel
)
65 IMPLEMENT_DYNAMIC_CLASS(wxWizardPageSimple
, wxWizardPage
)
66 IMPLEMENT_DYNAMIC_CLASS(wxWizardEvent
, wxNotifyEvent
)
68 // ============================================================================
70 // ============================================================================
72 // ----------------------------------------------------------------------------
74 // ----------------------------------------------------------------------------
76 wxWizardPage::wxWizardPage(wxWizard
*parent
, const wxBitmap
& bitmap
)
77 : wxPanel(parent
), m_bitmap(bitmap
)
79 // initially the page is hidden, it's shown only when it becomes current
83 // ----------------------------------------------------------------------------
85 // ----------------------------------------------------------------------------
87 wxWizardPage
*wxWizardPageSimple::GetPrev() const
92 wxWizardPage
*wxWizardPageSimple::GetNext() const
96 // ----------------------------------------------------------------------------
97 // generic wxWizard implementation
98 // ----------------------------------------------------------------------------
100 wxWizard::wxWizard(wxWindow
*parent
,
102 const wxString
& title
,
103 const wxBitmap
& bitmap
,
105 : m_posWizard(pos
), m_bitmap(bitmap
)
107 // just create the dialog itself here, the controls will be created in
108 // DoCreateControls() called later when we know our final size
109 m_page
= (wxWizardPage
*)NULL
;
110 m_btnPrev
= m_btnNext
= NULL
;
113 (void)wxDialog::Create(parent
, id
, title
, pos
);
116 void wxWizard::DoCreateControls()
118 // do nothing if the controls were already created
122 // constants defining the dialog layout
123 // ------------------------------------
125 // these constants define the position of the upper left corner of the
126 // bitmap or the page in the wizard
127 static const int X_MARGIN
= 10;
128 static const int Y_MARGIN
= 10;
130 // margin between the bitmap and the panel
131 static const int BITMAP_X_MARGIN
= 15;
133 // margin between the bitmap and the static line
134 static const int BITMAP_Y_MARGIN
= 15;
136 // margin between the static line and the buttons
137 static const int SEPARATOR_LINE_MARGIN
= 15;
139 // margin between "Next >" and "Cancel" buttons
140 static const int BUTTON_MARGIN
= 10;
142 // default width and height of the page
143 static const int DEFAULT_PAGE_WIDTH
= 270;
144 static const int DEFAULT_PAGE_HEIGHT
= 290;
149 wxSize sizeBtn
= wxButton::GetDefaultSize();
151 // the global dialog layout is: a row of buttons at the bottom (aligned to
152 // the right), the static line above them, the bitmap (if any) on the left
153 // of the upper part of the dialog and the panel in the remaining space
160 m_statbmp
= new wxStaticBitmap(this, -1, m_bitmap
, wxPoint(m_x
, m_y
));
162 m_x
+= m_bitmap
.GetWidth() + BITMAP_X_MARGIN
;
164 defaultHeight
= m_bitmap
.GetHeight();
168 m_statbmp
= (wxStaticBitmap
*)NULL
;
170 defaultHeight
= DEFAULT_PAGE_HEIGHT
;
173 // use default size if none given and also make sure that the dialog is
174 // not less than the default size
175 m_height
= m_sizePage
.y
== -1 ? defaultHeight
: m_sizePage
.y
;
176 m_width
= m_sizePage
.x
== -1 ? DEFAULT_PAGE_WIDTH
: m_sizePage
.x
;
177 if ( m_height
< defaultHeight
)
178 m_height
= defaultHeight
;
179 if ( m_width
< DEFAULT_PAGE_WIDTH
)
180 m_width
= DEFAULT_PAGE_WIDTH
;
183 int y
= m_y
+ m_height
+ BITMAP_Y_MARGIN
;
186 (void)new wxStaticLine(this, -1, wxPoint(x
, y
),
187 wxSize(m_x
+ m_width
- x
, 2));
188 #endif // wxUSE_STATLINE
190 x
= m_x
+ m_width
- 3*sizeBtn
.x
- BUTTON_MARGIN
;
191 y
+= SEPARATOR_LINE_MARGIN
;
192 m_btnPrev
= new wxButton(this, wxID_BACKWARD
, _("< &Back"), wxPoint(x
, y
), sizeBtn
);
195 m_btnNext
= new wxButton(this, wxID_FORWARD
, _("&Next >"), wxPoint(x
, y
), sizeBtn
);
197 x
+= sizeBtn
.x
+ BUTTON_MARGIN
;
198 (void)new wxButton(this, wxID_CANCEL
, _("Cancel"), wxPoint(x
, y
), sizeBtn
);
200 // position and size the dialog
201 // ----------------------------
203 SetClientSize(m_x
+ m_width
+ X_MARGIN
,
204 m_y
+ m_height
+ BITMAP_Y_MARGIN
+
205 SEPARATOR_LINE_MARGIN
+ sizeBtn
.y
+ Y_MARGIN
);
207 if ( m_posWizard
== wxDefaultPosition
)
213 void wxWizard::SetPageSize(const wxSize
& size
)
215 // otherwise it will have no effect now as it's too late...
216 wxASSERT_MSG( !WasCreated(), _T("should be called before RunWizard()!") );
221 bool wxWizard::ShowPage(wxWizardPage
*page
, bool goingForward
)
223 wxASSERT_MSG( page
!= m_page
, wxT("this is useless") );
225 // we'll use this to decide whether we have to change the label of this
226 // button or not (initially the label is "Next")
227 bool btnLabelWasNext
= TRUE
;
229 // and this tells us whether we already had the default bitmap before
234 // send the event to the old page
235 wxWizardEvent
event(wxEVT_WIZARD_PAGE_CHANGING
, GetId(), goingForward
);
236 if ( m_page
->GetEventHandler()->ProcessEvent(event
) &&
239 // vetoed by the page
245 btnLabelWasNext
= m_page
->GetNext() != (wxWizardPage
*)NULL
;
246 bmpWasDefault
= !m_page
->GetBitmap().Ok();
248 else // no previous page
250 // always set the bitmap
260 // terminate successfully
266 // send the event to the new page now
267 wxWizardEvent
event(wxEVT_WIZARD_PAGE_CHANGED
, GetId(), goingForward
);
268 (void)m_page
->GetEventHandler()->ProcessEvent(event
);
270 // position and show the new page
271 (void)m_page
->TransferDataToWindow();
272 m_page
->SetSize(m_x
, m_y
, m_width
, m_height
);
275 // change the bitmap if necessary (and if we have it at all)
276 int bmpIsDefault
= !m_page
->GetBitmap().Ok();
277 if ( m_statbmp
&& (bmpIsDefault
!= bmpWasDefault
) )
283 bmp
= m_page
->GetBitmap();
284 m_statbmp
->SetBitmap(bmp
);
287 // and update the buttons state
288 m_btnPrev
->Enable(m_page
->GetPrev() != (wxWizardPage
*)NULL
);
290 bool hasNext
= m_page
->GetNext() != (wxWizardPage
*)NULL
;
291 if ( btnLabelWasNext
!= hasNext
)
295 m_btnNext
->SetLabel(_("&Finish"));
297 m_btnNext
->SetLabel(_("&Next >"));
299 // nothing to do: the label was already correct
304 bool wxWizard::RunWizard(wxWizardPage
*firstPage
)
306 wxCHECK_MSG( firstPage
, FALSE
, wxT("can't run empty wizard") );
310 // can't return FALSE here because there is no old page
311 (void)ShowPage(firstPage
, TRUE
/* forward */);
313 return ShowModal() == wxID_OK
;
316 wxWizardPage
*wxWizard::GetCurrentPage() const
321 wxSize
wxWizard::GetPageSize() const
323 // make sure that the controls are created because otherwise m_width and
324 // m_height would be both still -1
325 wxConstCast(this, wxWizard
)->DoCreateControls();
327 return wxSize(m_width
, m_height
);
330 void wxWizard::OnCancel(wxCommandEvent
& WXUNUSED(event
))
332 // this function probably can never be called when we don't have an active
333 // page, but a small extra check won't hurt
334 wxWindow
*win
= m_page
? (wxWindow
*)m_page
: (wxWindow
*)this;
336 wxWizardEvent
event(wxEVT_WIZARD_CANCEL
, GetId());
337 if ( !win
->GetEventHandler()->ProcessEvent(event
) || event
.IsAllowed() )
339 // no objections - close the dialog
340 EndModal(wxID_CANCEL
);
342 //else: request to Cancel ignored
345 void wxWizard::OnBackOrNext(wxCommandEvent
& event
)
347 wxASSERT_MSG( (event
.GetEventObject() == m_btnNext
) ||
348 (event
.GetEventObject() == m_btnPrev
),
349 wxT("unknown button") );
351 // ask the current page first: notice that we do it before calling
352 // GetNext/Prev() because the data transfered from the controls of the page
353 // may change the value returned by these methods
354 if ( m_page
&& !m_page
->TransferDataFromWindow() )
356 // the page data is incorrect, don't do anything
360 bool forward
= event
.GetEventObject() == m_btnNext
;
365 page
= m_page
->GetNext();
369 page
= m_page
->GetPrev();
371 wxASSERT_MSG( page
, wxT("\"<Back\" button should have been disabled") );
374 // just pass to the new page (or may be not - but we don't care here)
375 (void)ShowPage(page
, forward
);
378 // ----------------------------------------------------------------------------
379 // our public interface
380 // ----------------------------------------------------------------------------
383 wxWizard
*wxWizardBase::Create(wxWindow
*parent
,
385 const wxString
& title
,
386 const wxBitmap
& bitmap
,
388 const wxSize
& WXUNUSED(size
))
390 return new wxWizard(parent
, id
, title
, bitmap
, pos
);
393 // ----------------------------------------------------------------------------
395 // ----------------------------------------------------------------------------
397 wxWizardEvent::wxWizardEvent(wxEventType type
, int id
, bool direction
)
398 : wxNotifyEvent(type
, id
)
400 m_direction
= direction
;
403 #endif // wxUSE_WIZARDDLG