1 ///////////////////////////////////////////////////////////////////////////////
2 // Name: generic/wizard.cpp
3 // Purpose: generic implementation of wxWizard class
4 // Author: Vadim Zeitlin
5 // Modified by: Robert Cavanaugh
6 // 1) Added capability for wxWizardPage to accept resources
7 // 2) Added "Help" button handler stub
8 // 3) Fixed ShowPage() bug on displaying bitmaps
9 // Robert Vazan (sizers)
12 // Copyright: (c) 1999 Vadim Zeitlin <zeitlin@dptmaths.ens-cachan.fr>
13 // Licence: wxWindows licence
14 ///////////////////////////////////////////////////////////////////////////////
16 // ============================================================================
18 // ============================================================================
20 // ----------------------------------------------------------------------------
22 // ----------------------------------------------------------------------------
25 #pragma implementation "wizardg.h"
28 // For compilers that support precompilation, includes "wx.h".
29 #include "wx/wxprec.h"
38 #include "wx/dynarray.h"
40 #include "wx/statbmp.h"
41 #include "wx/button.h"
44 #include "wx/statline.h"
47 #include "wx/wizard.h"
49 // ----------------------------------------------------------------------------
50 // event tables and such
51 // ----------------------------------------------------------------------------
53 DEFINE_EVENT_TYPE(wxEVT_WIZARD_PAGE_CHANGED
)
54 DEFINE_EVENT_TYPE(wxEVT_WIZARD_PAGE_CHANGING
)
55 DEFINE_EVENT_TYPE(wxEVT_WIZARD_CANCEL
)
56 DEFINE_EVENT_TYPE(wxEVT_WIZARD_FINISHED
)
57 DEFINE_EVENT_TYPE(wxEVT_WIZARD_HELP
)
59 BEGIN_EVENT_TABLE(wxWizard
, wxDialog
)
60 EVT_BUTTON(wxID_CANCEL
, wxWizard::OnCancel
)
61 EVT_BUTTON(wxID_BACKWARD
, wxWizard::OnBackOrNext
)
62 EVT_BUTTON(wxID_FORWARD
, wxWizard::OnBackOrNext
)
63 EVT_BUTTON(wxID_HELP
, wxWizard::OnHelp
)
65 EVT_WIZARD_PAGE_CHANGED(-1, wxWizard::OnWizEvent
)
66 EVT_WIZARD_PAGE_CHANGING(-1, wxWizard::OnWizEvent
)
67 EVT_WIZARD_CANCEL(-1, wxWizard::OnWizEvent
)
68 EVT_WIZARD_FINISHED(-1, wxWizard::OnWizEvent
)
69 EVT_WIZARD_HELP(-1, wxWizard::OnWizEvent
)
72 IMPLEMENT_DYNAMIC_CLASS(wxWizard
, wxDialog
)
73 IMPLEMENT_ABSTRACT_CLASS(wxWizardPage
, wxPanel
)
74 IMPLEMENT_DYNAMIC_CLASS(wxWizardPageSimple
, wxWizardPage
)
75 IMPLEMENT_DYNAMIC_CLASS(wxWizardEvent
, wxNotifyEvent
)
77 // ============================================================================
79 // ============================================================================
81 // ----------------------------------------------------------------------------
83 // ----------------------------------------------------------------------------
85 void wxWizardPage::Init()
87 m_bitmap
= wxNullBitmap
;
90 wxWizardPage::wxWizardPage(wxWizard
*parent
,
91 const wxBitmap
& bitmap
,
92 const wxChar
*resource
)
94 Create(parent
, bitmap
, resource
);
97 bool wxWizardPage::Create(wxWizard
*parent
,
98 const wxBitmap
& bitmap
,
99 const wxChar
*resource
)
101 if ( !wxPanel::Create(parent
, -1) )
104 if ( resource
!= NULL
)
106 #if wxUSE_WX_RESOURCES
108 if ( !LoadFromResource(this, resource
) )
110 wxFAIL_MSG(wxT("wxWizardPage LoadFromResource failed!!!!"));
113 #endif // wxUSE_RESOURCES
118 // initially the page is hidden, it's shown only when it becomes current
124 // ----------------------------------------------------------------------------
125 // wxWizardPageSimple
126 // ----------------------------------------------------------------------------
128 wxWizardPage
*wxWizardPageSimple::GetPrev() const
133 wxWizardPage
*wxWizardPageSimple::GetNext() const
138 // ----------------------------------------------------------------------------
140 // ----------------------------------------------------------------------------
142 class wxWizardSizer
: public wxSizer
145 wxWizardSizer(wxWizard
*owner
);
150 wxSize
GetMaxChildSize();
154 wxSize
SiblingSize(wxSizerItem
*child
);
157 bool m_childSizeValid
;
160 DECLARE_CLASS(wxWizardSizer
);
163 IMPLEMENT_CLASS(wxWizardSizer
, wxSizer
)
165 wxWizardSizer::wxWizardSizer(wxWizard
*owner
)
168 m_childSizeValid
= false;
171 void wxWizardSizer::RecalcSizes()
173 // Effect of this function depends on m_owner->m_page and
174 // it should be called whenever it changes (wxWizard::ShowPage)
175 if ( m_owner
->m_page
)
177 m_owner
->m_page
->SetSize(m_position
.x
,m_position
.y
, m_size
.x
,m_size
.y
);
181 wxSize
wxWizardSizer::CalcMin()
183 return m_owner
->GetPageSize();
186 wxSize
wxWizardSizer::GetMaxChildSize()
188 #if !defined(__WXDEBUG__)
189 if ( m_childSizeValid
)
194 wxSizerItemList::Node
*childNode
;
196 for(childNode
= m_children
.GetFirst(); childNode
;
197 childNode
= childNode
->GetNext())
199 wxSizerItem
*child
= childNode
->GetData();
200 maxOfMin
.IncTo(child
->CalcMin());
201 maxOfMin
.IncTo(SiblingSize(child
));
205 if ( m_childSizeValid
&& m_childSize
!= maxOfMin
)
207 wxFAIL_MSG( _T("Size changed in wxWizard::GetPageAreaSizer()")
208 _T("after RunWizard().\n")
209 _T("Did you forget to call GetSizer()->Fit(this) ")
210 _T("for some page?")) ;
214 #endif // __WXDEBUG__
216 if ( m_owner
->m_started
)
218 m_childSizeValid
= true;
219 m_childSize
= maxOfMin
;
225 int wxWizardSizer::Border() const
227 if ( m_owner
->m_calledSetBorder
)
228 return m_owner
->m_border
;
230 return m_children
.IsEmpty() ? 5 : 0;
233 wxSize
wxWizardSizer::SiblingSize(wxSizerItem
*child
)
237 if ( child
->IsWindow() )
239 wxWizardPage
*page
= wxDynamicCast(child
->GetWindow(), wxWizardPage
);
242 for ( wxWizardPage
*sibling
= page
->GetNext();
244 sibling
= sibling
->GetNext() )
246 if ( sibling
->GetSizer() )
248 maxSibling
.IncTo(sibling
->GetSizer()->CalcMin());
257 // ----------------------------------------------------------------------------
258 // generic wxWizard implementation
259 // ----------------------------------------------------------------------------
261 void wxWizard::Init()
263 m_posWizard
= wxDefaultPosition
;
264 m_page
= (wxWizardPage
*)NULL
;
265 m_btnPrev
= m_btnNext
= NULL
;
268 m_calledSetBorder
= false;
273 bool wxWizard::Create(wxWindow
*parent
,
275 const wxString
& title
,
276 const wxBitmap
& bitmap
,
280 bool result
= wxDialog::Create(parent
,id
,title
,pos
,wxDefaultSize
,style
);
290 void wxWizard::AddBitmapRow(wxBoxSizer
*mainColumn
)
292 m_sizerBmpAndPage
= new wxBoxSizer(wxHORIZONTAL
);
295 1, // Vertically stretchable
296 wxEXPAND
// Horizonal stretching, no border
299 0, // No vertical stretching
300 wxEXPAND
// No border, (mostly useless) horizontal stretching
305 m_statbmp
= new wxStaticBitmap(this, -1, m_bitmap
);
306 m_sizerBmpAndPage
->Add(
308 0, // No horizontal stretching
309 wxALL
, // Border all around, top alignment
312 m_sizerBmpAndPage
->Add(
314 0, // No horizontal stretching
315 wxEXPAND
// No border, (mostly useless) vertical stretching
319 m_statbmp
= (wxStaticBitmap
*)NULL
;
321 // Added to m_sizerBmpAndPage in FinishLayout
322 m_sizerPage
= new wxWizardSizer(this);
325 void wxWizard::AddStaticLine(wxBoxSizer
*mainColumn
)
329 new wxStaticLine(this, -1),
330 0, // Vertically unstretchable
331 wxEXPAND
| wxALL
, // Border all around, horizontally stretchable
335 0, // No vertical stretching
336 wxEXPAND
// No border, (mostly useless) horizontal stretching
340 #endif // wxUSE_STATLINE
343 void wxWizard::AddBackNextPair(wxBoxSizer
*buttonRow
)
345 // margin between Back and Next buttons
347 static const int BACKNEXT_MARGIN
= 10;
349 static const int BACKNEXT_MARGIN
= 0;
352 wxBoxSizer
*backNextPair
= new wxBoxSizer(wxHORIZONTAL
);
355 0, // No horizontal stretching
356 wxALL
, // Border all around
360 m_btnPrev
= new wxButton(this, wxID_BACKWARD
, _("< &Back"));
361 backNextPair
->Add(m_btnPrev
);
362 backNextPair
->Add(BACKNEXT_MARGIN
,0,
363 0, // No horizontal stretching
364 wxEXPAND
// No border, (mostly useless) vertical stretching
366 m_btnNext
= new wxButton(this, wxID_FORWARD
, _("&Next >"));
367 backNextPair
->Add(m_btnNext
);
370 void wxWizard::AddButtonRow(wxBoxSizer
*mainColumn
)
372 wxBoxSizer
*buttonRow
= new wxBoxSizer(wxHORIZONTAL
);
375 0, // Vertically unstretchable
376 wxALIGN_RIGHT
// Right aligned, no border
379 if (GetExtraStyle() & wxWIZARD_EX_HELPBUTTON
)
381 new wxButton(this, wxID_HELP
, _("&Help")),
382 0, // Horizontally unstretchable
383 wxALL
, // Border all around, top aligned
387 AddBackNextPair(buttonRow
);
390 new wxButton(this, wxID_CANCEL
, _("&Cancel")),
391 0, // Horizontally unstretchable
392 wxALL
, // Border all around, top aligned
397 void wxWizard::DoCreateControls()
399 // do nothing if the controls were already created
403 // wxWindow::SetSizer will be called at end
404 wxBoxSizer
*windowSizer
= new wxBoxSizer(wxVERTICAL
);
406 wxBoxSizer
*mainColumn
= new wxBoxSizer(wxVERTICAL
);
409 1, // Vertical stretching
410 wxALL
| wxEXPAND
, // Border all around, horizontal stretching
414 AddBitmapRow(mainColumn
);
415 AddStaticLine(mainColumn
);
416 AddButtonRow(mainColumn
);
418 // wxWindow::SetSizer should be followed by wxWindow::Fit, but
419 // this is done in FinishLayout anyway so why duplicate it
420 SetSizer(windowSizer
);
423 void wxWizard::SetPageSize(const wxSize
& size
)
425 wxCHECK_RET(!m_started
,wxT("wxWizard::SetPageSize after RunWizard"));
429 void wxWizard::FinishLayout()
431 m_sizerBmpAndPage
->Add(
433 1, // Horizontal stretching
434 wxEXPAND
| wxALL
, // Vertically stretchable
435 m_sizerPage
->Border()
438 GetSizer()->SetSizeHints(this);
439 if ( m_posWizard
== wxDefaultPosition
)
443 void wxWizard::FitToPage(const wxWizardPage
*page
)
445 wxCHECK_RET(!m_started
,wxT("wxWizard::FitToPage after RunWizard"));
449 wxSize size
= page
->GetBestSize();
451 m_sizePage
.IncTo(size
);
453 page
= page
->GetNext();
457 bool wxWizard::ShowPage(wxWizardPage
*page
, bool goingForward
)
459 wxASSERT_MSG( page
!= m_page
, wxT("this is useless") );
461 // we'll use this to decide whether we have to change the label of this
462 // button or not (initially the label is "Next")
463 bool btnLabelWasNext
= TRUE
;
465 // Modified 10-20-2001 Robert Cavanaugh.
466 // Fixed bug for displaying a new bitmap
467 // in each *consecutive* page
469 // flag to indicate if this page uses a new bitmap
470 bool bmpIsDefault
= TRUE
;
472 // use these labels to determine if we need to change the bitmap
474 wxBitmap bmpPrev
, bmpCur
;
476 // check for previous page
479 // send the event to the old page
480 wxWizardEvent
event(wxEVT_WIZARD_PAGE_CHANGING
, GetId(), goingForward
, m_page
);
481 if ( m_page
->GetEventHandler()->ProcessEvent(event
) &&
484 // vetoed by the page
490 btnLabelWasNext
= HasNextPage(m_page
);
492 // Get the bitmap of the previous page (if it exists)
493 if ( m_page
->GetBitmap().Ok() )
495 bmpPrev
= m_page
->GetBitmap();
505 // terminate successfully
509 wxWizardEvent
event(wxEVT_WIZARD_FINISHED
, GetId(),FALSE
, 0);
510 (void)GetEventHandler()->ProcessEvent(event
);
515 // position and show the new page
516 (void)m_page
->TransferDataToWindow();
518 // wxWizardSizer::RecalcSizes wants to be called when m_page changes
519 m_sizerPage
->RecalcSizes();
521 // check if bitmap needs to be updated
522 // update default flag as well
523 if ( m_page
->GetBitmap().Ok() )
525 bmpCur
= m_page
->GetBitmap();
526 bmpIsDefault
= FALSE
;
529 // change the bitmap if:
530 // 1) a default bitmap was selected in constructor
531 // 2) this page was constructed with a bitmap
532 // 3) this bitmap is not the previous bitmap
533 if ( m_statbmp
&& (bmpCur
!= bmpPrev
) )
539 bmp
= m_page
->GetBitmap();
540 m_statbmp
->SetBitmap(bmp
);
543 // and update the buttons state
544 m_btnPrev
->Enable(HasPrevPage(m_page
));
546 bool hasNext
= HasNextPage(m_page
);
547 if ( btnLabelWasNext
!= hasNext
)
551 m_btnNext
->SetLabel(_("&Finish"));
553 m_btnNext
->SetLabel(_("&Next >"));
555 // nothing to do: the label was already correct
557 // send the change event to the new page now
558 wxWizardEvent
event(wxEVT_WIZARD_PAGE_CHANGED
, GetId(), goingForward
, m_page
);
559 (void)m_page
->GetEventHandler()->ProcessEvent(event
);
561 // and finally show it
568 bool wxWizard::RunWizard(wxWizardPage
*firstPage
)
570 wxCHECK_MSG( firstPage
, FALSE
, wxT("can't run empty wizard") );
572 // Set before FinishLayout to enable wxWizardSizer::GetMaxChildSize
575 // This cannot be done sooner, because user can change layout options
579 // can't return FALSE here because there is no old page
580 (void)ShowPage(firstPage
, TRUE
/* forward */);
582 return ShowModal() == wxID_OK
;
585 wxWizardPage
*wxWizard::GetCurrentPage() const
590 wxSize
wxWizard::GetPageSize() const
592 wxSize
pageSize(GetManualPageSize());
593 pageSize
.IncTo(m_sizerPage
->GetMaxChildSize());
597 wxSizer
*wxWizard::GetPageAreaSizer() const
602 void wxWizard::SetBorder(int border
)
604 wxCHECK_RET(!m_started
,wxT("wxWizard::SetBorder after RunWizard"));
605 m_calledSetBorder
= true;
609 wxSize
wxWizard::GetManualPageSize() const
611 // default width and height of the page
612 static const int DEFAULT_PAGE_WIDTH
= 270;
613 static const int DEFAULT_PAGE_HEIGHT
= 290;
615 wxSize
totalPageSize(DEFAULT_PAGE_WIDTH
,DEFAULT_PAGE_HEIGHT
);
617 totalPageSize
.IncTo(m_sizePage
);
620 totalPageSize
.IncTo(wxSize(0,m_bitmap
.GetHeight()));
622 return totalPageSize
;
625 void wxWizard::OnCancel(wxCommandEvent
& WXUNUSED(eventUnused
))
627 // this function probably can never be called when we don't have an active
628 // page, but a small extra check won't hurt
629 wxWindow
*win
= m_page
? (wxWindow
*)m_page
: (wxWindow
*)this;
631 wxWizardEvent
event(wxEVT_WIZARD_CANCEL
, GetId(), FALSE
, m_page
);
632 if ( !win
->GetEventHandler()->ProcessEvent(event
) || event
.IsAllowed() )
634 // no objections - close the dialog
635 EndModal(wxID_CANCEL
);
637 //else: request to Cancel ignored
640 void wxWizard::OnBackOrNext(wxCommandEvent
& event
)
642 wxASSERT_MSG( (event
.GetEventObject() == m_btnNext
) ||
643 (event
.GetEventObject() == m_btnPrev
),
644 wxT("unknown button") );
646 // ask the current page first: notice that we do it before calling
647 // GetNext/Prev() because the data transfered from the controls of the page
648 // may change the value returned by these methods
649 if ( m_page
&& (!m_page
->Validate() || !m_page
->TransferDataFromWindow()) )
651 // the page data is incorrect, don't do anything
655 bool forward
= event
.GetEventObject() == m_btnNext
;
660 page
= m_page
->GetNext();
664 page
= m_page
->GetPrev();
666 wxASSERT_MSG( page
, wxT("\"<Back\" button should have been disabled") );
669 // just pass to the new page (or may be not - but we don't care here)
670 (void)ShowPage(page
, forward
);
673 void wxWizard::OnHelp(wxCommandEvent
& WXUNUSED(event
))
675 // this function probably can never be called when we don't have an active
676 // page, but a small extra check won't hurt
679 // Create and send the help event to the specific page handler
680 // event data contains the active page so that context-sensitive
682 wxWizardEvent
eventHelp(wxEVT_WIZARD_HELP
, GetId(), TRUE
, m_page
);
683 (void)m_page
->GetEventHandler()->ProcessEvent(eventHelp
);
687 void wxWizard::OnWizEvent(wxWizardEvent
& event
)
689 // the dialogs have wxWS_EX_BLOCK_EVENTS style on by default but we want to
690 // propagate wxEVT_WIZARD_XXX to the parent (if any), so do it manually
691 if ( !(GetExtraStyle() & wxWS_EX_BLOCK_EVENTS
) )
693 // the event will be propagated anyhow
697 wxWindow
*parent
= GetParent();
699 if ( !parent
|| !parent
->GetEventHandler()->ProcessEvent(event
) )
705 // ----------------------------------------------------------------------------
706 // our public interface
707 // ----------------------------------------------------------------------------
709 #ifdef WXWIN_COMPATIBILITY_2_2
712 wxWizard
*wxWizardBase::Create(wxWindow
*parent
,
714 const wxString
& title
,
715 const wxBitmap
& bitmap
,
717 const wxSize
& WXUNUSED(size
))
719 return new wxWizard(parent
, id
, title
, bitmap
, pos
);
722 #endif // WXWIN_COMPATIBILITY_2_2
724 // ----------------------------------------------------------------------------
726 // ----------------------------------------------------------------------------
728 wxWizardEvent::wxWizardEvent(wxEventType type
, int id
, bool direction
, wxWizardPage
* page
)
729 : wxNotifyEvent(type
, id
)
731 // Modified 10-20-2001 Robert Cavanaugh
732 // add the active page to the event data
733 m_direction
= direction
;
737 #endif // wxUSE_WIZARDDLG