1 ///////////////////////////////////////////////////////////////////////////////
2 // Name: src/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 // ----------------------------------------------------------------------------
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"
38 #include "wx/settings.h"
41 #include "wx/statline.h"
44 #include "wx/wizard.h"
46 // ----------------------------------------------------------------------------
48 // ----------------------------------------------------------------------------
50 class wxWizardSizer
: public wxSizer
53 wxWizardSizer(wxWizard
*owner
);
55 virtual wxSizerItem
*Insert(size_t index
, wxSizerItem
*item
);
57 virtual void RecalcSizes();
58 virtual wxSize
CalcMin();
60 // get the max size of all wizard pages
61 wxSize
GetMaxChildSize();
63 // return the border which can be either set using wxWizard::SetBorder() or
65 int GetBorder() const;
67 // hide the pages which we temporarily "show" when they're added to this
68 // sizer (see Insert())
72 wxSize
SiblingSize(wxSizerItem
*child
);
78 // ----------------------------------------------------------------------------
79 // event tables and such
80 // ----------------------------------------------------------------------------
82 DEFINE_EVENT_TYPE(wxEVT_WIZARD_PAGE_CHANGED
)
83 DEFINE_EVENT_TYPE(wxEVT_WIZARD_PAGE_CHANGING
)
84 DEFINE_EVENT_TYPE(wxEVT_WIZARD_CANCEL
)
85 DEFINE_EVENT_TYPE(wxEVT_WIZARD_FINISHED
)
86 DEFINE_EVENT_TYPE(wxEVT_WIZARD_HELP
)
88 BEGIN_EVENT_TABLE(wxWizard
, wxDialog
)
89 EVT_BUTTON(wxID_CANCEL
, wxWizard::OnCancel
)
90 EVT_BUTTON(wxID_BACKWARD
, wxWizard::OnBackOrNext
)
91 EVT_BUTTON(wxID_FORWARD
, wxWizard::OnBackOrNext
)
92 EVT_BUTTON(wxID_HELP
, wxWizard::OnHelp
)
94 EVT_WIZARD_PAGE_CHANGED(wxID_ANY
, wxWizard::OnWizEvent
)
95 EVT_WIZARD_PAGE_CHANGING(wxID_ANY
, wxWizard::OnWizEvent
)
96 EVT_WIZARD_CANCEL(wxID_ANY
, wxWizard::OnWizEvent
)
97 EVT_WIZARD_FINISHED(wxID_ANY
, wxWizard::OnWizEvent
)
98 EVT_WIZARD_HELP(wxID_ANY
, wxWizard::OnWizEvent
)
101 IMPLEMENT_DYNAMIC_CLASS(wxWizard
, wxDialog
)
110 IMPLEMENT_ABSTRACT_CLASS(wxWizardPage
, wxPanel
)
111 IMPLEMENT_DYNAMIC_CLASS(wxWizardPageSimple
, wxWizardPage
)
112 IMPLEMENT_DYNAMIC_CLASS(wxWizardEvent
, wxNotifyEvent
)
114 // ============================================================================
116 // ============================================================================
118 // ----------------------------------------------------------------------------
120 // ----------------------------------------------------------------------------
122 void wxWizardPage::Init()
124 m_bitmap
= wxNullBitmap
;
127 wxWizardPage::wxWizardPage(wxWizard
*parent
,
128 const wxBitmap
& bitmap
,
129 const wxChar
*resource
)
131 Create(parent
, bitmap
, resource
);
134 bool wxWizardPage::Create(wxWizard
*parent
,
135 const wxBitmap
& bitmap
,
136 const wxChar
*resource
)
138 if ( !wxPanel::Create(parent
, wxID_ANY
) )
141 if ( resource
!= NULL
)
143 #if wxUSE_WX_RESOURCES
145 if ( !LoadFromResource(this, resource
) )
147 wxFAIL_MSG(wxT("wxWizardPage LoadFromResource failed!!!!"));
150 #endif // wxUSE_RESOURCES
155 // initially the page is hidden, it's shown only when it becomes current
161 // ----------------------------------------------------------------------------
162 // wxWizardPageSimple
163 // ----------------------------------------------------------------------------
165 wxWizardPage
*wxWizardPageSimple::GetPrev() const
170 wxWizardPage
*wxWizardPageSimple::GetNext() const
175 // ----------------------------------------------------------------------------
177 // ----------------------------------------------------------------------------
179 wxWizardSizer::wxWizardSizer(wxWizard
*owner
)
181 m_childSize(wxDefaultSize
)
185 wxSizerItem
*wxWizardSizer::Insert(size_t index
, wxSizerItem
*item
)
187 m_owner
->m_usingSizer
= true;
189 if ( item
->IsWindow() )
191 // we must pretend that the window is shown as otherwise it wouldn't be
192 // taken into account for the layout -- but avoid really showing it, so
193 // just set the internal flag instead of calling wxWindow::Show()
194 item
->GetWindow()->wxWindowBase::Show();
197 return wxSizer::Insert(index
, item
);
200 void wxWizardSizer::HidePages()
202 for ( wxSizerItemList::compatibility_iterator node
= GetChildren().GetFirst();
204 node
= node
->GetNext() )
206 wxSizerItem
* const item
= node
->GetData();
207 if ( item
->IsWindow() )
208 item
->GetWindow()->wxWindowBase::Show(false);
212 void wxWizardSizer::RecalcSizes()
214 // Effect of this function depends on m_owner->m_page and
215 // it should be called whenever it changes (wxWizard::ShowPage)
216 if ( m_owner
->m_page
)
218 m_owner
->m_page
->SetSize(wxRect(m_position
, m_size
));
222 wxSize
wxWizardSizer::CalcMin()
224 return m_owner
->GetPageSize();
227 wxSize
wxWizardSizer::GetMaxChildSize()
229 #if !defined(__WXDEBUG__)
230 if ( m_childSize
.IsFullySpecified() )
236 for ( wxSizerItemList::compatibility_iterator childNode
= m_children
.GetFirst();
238 childNode
= childNode
->GetNext() )
240 wxSizerItem
*child
= childNode
->GetData();
241 maxOfMin
.IncTo(child
->CalcMin());
242 maxOfMin
.IncTo(SiblingSize(child
));
246 if ( m_childSize
.IsFullySpecified() && m_childSize
!= maxOfMin
)
248 wxFAIL_MSG( _T("Size changed in wxWizard::GetPageAreaSizer()")
249 _T("after RunWizard().\n")
250 _T("Did you forget to call GetSizer()->Fit(this) ")
251 _T("for some page?")) ;
255 #endif // __WXDEBUG__
257 if ( m_owner
->m_started
)
259 m_childSize
= maxOfMin
;
265 int wxWizardSizer::GetBorder() const
267 return m_owner
->m_border
;
270 wxSize
wxWizardSizer::SiblingSize(wxSizerItem
*child
)
274 if ( child
->IsWindow() )
276 wxWizardPage
*page
= wxDynamicCast(child
->GetWindow(), wxWizardPage
);
279 for ( wxWizardPage
*sibling
= page
->GetNext();
281 sibling
= sibling
->GetNext() )
283 if ( sibling
->GetSizer() )
285 maxSibling
.IncTo(sibling
->GetSizer()->CalcMin());
294 // ----------------------------------------------------------------------------
295 // generic wxWizard implementation
296 // ----------------------------------------------------------------------------
298 void wxWizard::Init()
300 m_posWizard
= wxDefaultPosition
;
301 m_page
= (wxWizardPage
*)NULL
;
302 m_btnPrev
= m_btnNext
= NULL
;
304 m_sizerBmpAndPage
= NULL
;
309 m_usingSizer
= false;
312 bool wxWizard::Create(wxWindow
*parent
,
314 const wxString
& title
,
315 const wxBitmap
& bitmap
,
319 bool result
= wxDialog::Create(parent
,id
,title
,pos
,wxDefaultSize
,style
);
329 void wxWizard::AddBitmapRow(wxBoxSizer
*mainColumn
)
331 m_sizerBmpAndPage
= new wxBoxSizer(wxHORIZONTAL
);
334 1, // Vertically stretchable
335 wxEXPAND
// Horizonal stretching, no border
338 0, // No vertical stretching
339 wxEXPAND
// No border, (mostly useless) horizontal stretching
345 m_statbmp
= new wxStaticBitmap(this, wxID_ANY
, m_bitmap
);
346 m_sizerBmpAndPage
->Add(
348 0, // No horizontal stretching
349 wxALL
, // Border all around, top alignment
352 m_sizerBmpAndPage
->Add(
354 0, // No horizontal stretching
355 wxEXPAND
// No border, (mostly useless) vertical stretching
360 // Added to m_sizerBmpAndPage later
361 m_sizerPage
= new wxWizardSizer(this);
364 void wxWizard::AddStaticLine(wxBoxSizer
*mainColumn
)
368 new wxStaticLine(this, wxID_ANY
),
369 0, // Vertically unstretchable
370 wxEXPAND
| wxALL
, // Border all around, horizontally stretchable
374 0, // No vertical stretching
375 wxEXPAND
// No border, (mostly useless) horizontal stretching
379 #endif // wxUSE_STATLINE
382 void wxWizard::AddBackNextPair(wxBoxSizer
*buttonRow
)
384 wxASSERT_MSG( m_btnNext
&& m_btnPrev
,
385 _T("You must create the buttons before calling ")
386 _T("wxWizard::AddBackNextPair") );
388 // margin between Back and Next buttons
390 static const int BACKNEXT_MARGIN
= 10;
392 static const int BACKNEXT_MARGIN
= 0;
395 wxBoxSizer
*backNextPair
= new wxBoxSizer(wxHORIZONTAL
);
398 0, // No horizontal stretching
399 wxALL
, // Border all around
403 backNextPair
->Add(m_btnPrev
);
404 backNextPair
->Add(BACKNEXT_MARGIN
,0,
405 0, // No horizontal stretching
406 wxEXPAND
// No border, (mostly useless) vertical stretching
408 backNextPair
->Add(m_btnNext
);
411 void wxWizard::AddButtonRow(wxBoxSizer
*mainColumn
)
413 // the order in which the buttons are created determines the TAB order - at least under MSWindows...
414 // although the 'back' button appears before the 'next' button, a more userfriendly tab order is
415 // to activate the 'next' button first (create the next button before the back button).
416 // The reason is: The user will repeatedly enter information in the wizard pages and then wants to
417 // press 'next'. If a user uses mostly the keyboard, he would have to skip the 'back' button
418 // everytime. This is annoying. There is a second reason: RETURN acts as TAB. If the 'next'
419 // button comes first in the TAB order, the user can enter information very fast using the RETURN
420 // key to TAB to the next entry field and page. This would not be possible, if the 'back' button
421 // was created before the 'next' button.
423 bool isPda
= (wxSystemSettings::GetScreenType() <= wxSYS_SCREEN_PDA
);
424 int buttonStyle
= isPda
? wxBU_EXACTFIT
: 0;
426 wxBoxSizer
*buttonRow
= new wxBoxSizer(wxHORIZONTAL
);
428 if (GetExtraStyle() & wxWIZARD_EX_HELPBUTTON
)
431 0, // Vertically unstretchable
432 wxGROW
|wxALIGN_CENTRE
438 0, // Vertically unstretchable
439 wxALIGN_RIGHT
// Right aligned, no border
442 // Desired TAB order is 'next', 'cancel', 'help', 'back'. This makes the 'back' button the last control on the page.
443 // Create the buttons in the right order...
446 if (GetExtraStyle() & wxWIZARD_EX_HELPBUTTON
)
447 btnHelp
=new wxButton(this, wxID_HELP
, _("&Help"), wxDefaultPosition
, wxDefaultSize
, buttonStyle
);
450 m_btnNext
= new wxButton(this, wxID_FORWARD
, _("&Next >"));
451 wxButton
*btnCancel
=new wxButton(this, wxID_CANCEL
, _("&Cancel"), wxDefaultPosition
, wxDefaultSize
, buttonStyle
);
453 if (GetExtraStyle() & wxWIZARD_EX_HELPBUTTON
)
454 btnHelp
=new wxButton(this, wxID_HELP
, _("&Help"), wxDefaultPosition
, wxDefaultSize
, buttonStyle
);
456 m_btnPrev
= new wxButton(this, wxID_BACKWARD
, _("< &Back"), wxDefaultPosition
, wxDefaultSize
, buttonStyle
);
462 0, // Horizontally unstretchable
463 wxALL
, // Border all around, top aligned
467 // Put stretchable space between help button and others
468 buttonRow
->Add(0, 0, 1, wxALIGN_CENTRE
, 0);
472 AddBackNextPair(buttonRow
);
476 0, // Horizontally unstretchable
477 wxALL
, // Border all around, top aligned
482 void wxWizard::DoCreateControls()
484 // do nothing if the controls were already created
488 bool isPda
= (wxSystemSettings::GetScreenType() <= wxSYS_SCREEN_PDA
);
490 // Horizontal stretching, and if not PDA, border all around
491 int mainColumnSizerFlags
= isPda
? wxEXPAND
: wxALL
|wxEXPAND
;
493 // wxWindow::SetSizer will be called at end
494 wxBoxSizer
*windowSizer
= new wxBoxSizer(wxVERTICAL
);
496 wxBoxSizer
*mainColumn
= new wxBoxSizer(wxVERTICAL
);
499 1, // Vertical stretching
500 mainColumnSizerFlags
,
504 AddBitmapRow(mainColumn
);
507 AddStaticLine(mainColumn
);
509 AddButtonRow(mainColumn
);
511 SetSizer(windowSizer
);
514 void wxWizard::SetPageSize(const wxSize
& size
)
516 wxCHECK_RET(!m_started
, wxT("wxWizard::SetPageSize after RunWizard"));
520 void wxWizard::FitToPage(const wxWizardPage
*page
)
522 wxCHECK_RET(!m_started
, wxT("wxWizard::FitToPage after RunWizard"));
526 wxSize size
= page
->GetBestSize();
528 m_sizePage
.IncTo(size
);
530 page
= page
->GetNext();
534 bool wxWizard::ShowPage(wxWizardPage
*page
, bool goingForward
)
536 wxASSERT_MSG( page
!= m_page
, wxT("this is useless") );
538 wxSizerFlags
flags(1);
539 flags
.Border(wxALL
, m_border
).Expand();
545 m_sizerBmpAndPage
->Add(m_sizerPage
, flags
);
547 // now that our layout is computed correctly, hide the pages
548 // artificially shown in wxWizardSizer::Insert() back again
549 m_sizerPage
->HidePages();
554 // we'll use this to decide whether we have to change the label of this
555 // button or not (initially the label is "Next")
556 bool btnLabelWasNext
= true;
558 // remember the old bitmap (if any) to compare with the new one later
561 // check for previous page
564 // send the event to the old page
565 wxWizardEvent
event(wxEVT_WIZARD_PAGE_CHANGING
, GetId(),
566 goingForward
, m_page
);
567 if ( m_page
->GetEventHandler()->ProcessEvent(event
) &&
570 // vetoed by the page
576 btnLabelWasNext
= HasNextPage(m_page
);
578 bmpPrev
= m_page
->GetBitmap();
581 m_sizerBmpAndPage
->Detach(m_page
);
590 // terminate successfully
597 SetReturnCode(wxID_OK
);
601 // and notify the user code (this is especially useful for modeless
603 wxWizardEvent
event(wxEVT_WIZARD_FINISHED
, GetId(), false, 0);
604 (void)GetEventHandler()->ProcessEvent(event
);
609 // position and show the new page
610 (void)m_page
->TransferDataToWindow();
614 // wxWizardSizer::RecalcSizes wants to be called when m_page changes
615 m_sizerPage
->RecalcSizes();
617 else // pages are not managed by the sizer
619 m_sizerBmpAndPage
->Add(m_page
, flags
);
620 m_sizerBmpAndPage
->SetItemMinSize(m_page
, GetPageSize());
624 // update the bitmap if:it changed
627 wxBitmap bmp
= m_page
->GetBitmap();
634 if ( bmp
!= bmpPrev
)
635 m_statbmp
->SetBitmap(bmp
);
637 #endif // wxUSE_STATBMP
640 // and update the buttons state
641 m_btnPrev
->Enable(HasPrevPage(m_page
));
643 bool hasNext
= HasNextPage(m_page
);
644 if ( btnLabelWasNext
!= hasNext
)
646 m_btnNext
->SetLabel(hasNext
? _("&Next >") : _("&Finish"));
648 // nothing to do: the label was already correct
650 m_btnNext
->SetDefault();
653 // send the change event to the new page now
654 wxWizardEvent
event(wxEVT_WIZARD_PAGE_CHANGED
, GetId(), goingForward
, m_page
);
655 (void)m_page
->GetEventHandler()->ProcessEvent(event
);
657 // and finally show it
662 m_sizerBmpAndPage
->Layout();
668 if ( wxSystemSettings::GetScreenType() > wxSYS_SCREEN_PDA
)
670 GetSizer()->SetSizeHints(this);
671 if ( m_posWizard
== wxDefaultPosition
)
679 bool wxWizard::RunWizard(wxWizardPage
*firstPage
)
681 wxCHECK_MSG( firstPage
, false, wxT("can't run empty wizard") );
683 // can't return false here because there is no old page
684 (void)ShowPage(firstPage
, true /* forward */);
688 return ShowModal() == wxID_OK
;
691 wxWizardPage
*wxWizard::GetCurrentPage() const
696 wxSize
wxWizard::GetPageSize() const
698 // default width and height of the page
699 int DEFAULT_PAGE_WIDTH
,
701 if ( wxSystemSettings::GetScreenType() <= wxSYS_SCREEN_PDA
)
703 // Make the default page size small enough to fit on screen
704 DEFAULT_PAGE_WIDTH
= wxSystemSettings::GetMetric(wxSYS_SCREEN_X
) / 2;
705 DEFAULT_PAGE_HEIGHT
= wxSystemSettings::GetMetric(wxSYS_SCREEN_Y
) / 2;
710 DEFAULT_PAGE_HEIGHT
= 270;
713 // start with default minimal size
714 wxSize
pageSize(DEFAULT_PAGE_WIDTH
, DEFAULT_PAGE_HEIGHT
);
716 // make the page at least as big as specified by user
717 pageSize
.IncTo(m_sizePage
);
721 // make the page at least as tall as the bitmap
722 pageSize
.IncTo(wxSize(0, m_bitmap
.GetHeight()));
727 // make it big enough to contain all pages added to the sizer
728 pageSize
.IncTo(m_sizerPage
->GetMaxChildSize());
734 wxSizer
*wxWizard::GetPageAreaSizer() const
739 void wxWizard::SetBorder(int border
)
741 wxCHECK_RET(!m_started
, wxT("wxWizard::SetBorder after RunWizard"));
746 void wxWizard::OnCancel(wxCommandEvent
& WXUNUSED(eventUnused
))
748 // this function probably can never be called when we don't have an active
749 // page, but a small extra check won't hurt
750 wxWindow
*win
= m_page
? (wxWindow
*)m_page
: (wxWindow
*)this;
752 wxWizardEvent
event(wxEVT_WIZARD_CANCEL
, GetId(), false, m_page
);
753 if ( !win
->GetEventHandler()->ProcessEvent(event
) || event
.IsAllowed() )
755 // no objections - close the dialog
758 EndModal(wxID_CANCEL
);
762 SetReturnCode(wxID_CANCEL
);
766 //else: request to Cancel ignored
769 void wxWizard::OnBackOrNext(wxCommandEvent
& event
)
771 wxASSERT_MSG( (event
.GetEventObject() == m_btnNext
) ||
772 (event
.GetEventObject() == m_btnPrev
),
773 wxT("unknown button") );
775 wxCHECK_RET( m_page
, _T("should have a valid current page") );
777 // ask the current page first: notice that we do it before calling
778 // GetNext/Prev() because the data transfered from the controls of the page
779 // may change the value returned by these methods
780 if ( !m_page
->Validate() || !m_page
->TransferDataFromWindow() )
782 // the page data is incorrect, don't do anything
786 bool forward
= event
.GetEventObject() == m_btnNext
;
791 page
= m_page
->GetNext();
795 page
= m_page
->GetPrev();
797 wxASSERT_MSG( page
, wxT("\"<Back\" button should have been disabled") );
800 // just pass to the new page (or maybe not - but we don't care here)
801 (void)ShowPage(page
, forward
);
804 void wxWizard::OnHelp(wxCommandEvent
& WXUNUSED(event
))
806 // this function probably can never be called when we don't have an active
807 // page, but a small extra check won't hurt
810 // Create and send the help event to the specific page handler
811 // event data contains the active page so that context-sensitive
813 wxWizardEvent
eventHelp(wxEVT_WIZARD_HELP
, GetId(), true, m_page
);
814 (void)m_page
->GetEventHandler()->ProcessEvent(eventHelp
);
818 void wxWizard::OnWizEvent(wxWizardEvent
& event
)
820 // the dialogs have wxWS_EX_BLOCK_EVENTS style on by default but we want to
821 // propagate wxEVT_WIZARD_XXX to the parent (if any), so do it manually
822 if ( !(GetExtraStyle() & wxWS_EX_BLOCK_EVENTS
) )
824 // the event will be propagated anyhow
829 wxWindow
*parent
= GetParent();
831 if ( !parent
|| !parent
->GetEventHandler()->ProcessEvent(event
) )
837 if ( ( !m_wasModal
) &&
839 ( event
.GetEventType() == wxEVT_WIZARD_FINISHED
||
840 event
.GetEventType() == wxEVT_WIZARD_CANCEL
848 // ----------------------------------------------------------------------------
850 // ----------------------------------------------------------------------------
852 wxWizardEvent::wxWizardEvent(wxEventType type
, int id
, bool direction
, wxWizardPage
* page
)
853 : wxNotifyEvent(type
, id
)
855 // Modified 10-20-2001 Robert Cavanaugh
856 // add the active page to the event data
857 m_direction
= direction
;
861 #endif // wxUSE_WIZARDDLG