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 // ----------------------------------------------------------------------------
24 #if defined(__GNUG__) && !defined(NO_GCC_PRAGMA)
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 // ----------------------------------------------------------------------------
51 // ----------------------------------------------------------------------------
53 class wxWizardSizer
: public wxSizer
56 wxWizardSizer(wxWizard
*owner
);
61 wxSize
GetMaxChildSize();
65 wxSize
SiblingSize(wxSizerItem
*child
);
68 bool m_childSizeValid
;
72 // ----------------------------------------------------------------------------
73 // event tables and such
74 // ----------------------------------------------------------------------------
76 DEFINE_EVENT_TYPE(wxEVT_WIZARD_PAGE_CHANGED
)
77 DEFINE_EVENT_TYPE(wxEVT_WIZARD_PAGE_CHANGING
)
78 DEFINE_EVENT_TYPE(wxEVT_WIZARD_CANCEL
)
79 DEFINE_EVENT_TYPE(wxEVT_WIZARD_FINISHED
)
80 DEFINE_EVENT_TYPE(wxEVT_WIZARD_HELP
)
82 BEGIN_EVENT_TABLE(wxWizard
, wxDialog
)
83 EVT_BUTTON(wxID_CANCEL
, wxWizard::OnCancel
)
84 EVT_BUTTON(wxID_BACKWARD
, wxWizard::OnBackOrNext
)
85 EVT_BUTTON(wxID_FORWARD
, wxWizard::OnBackOrNext
)
86 EVT_BUTTON(wxID_HELP
, wxWizard::OnHelp
)
88 EVT_WIZARD_PAGE_CHANGED(-1, wxWizard::OnWizEvent
)
89 EVT_WIZARD_PAGE_CHANGING(-1, wxWizard::OnWizEvent
)
90 EVT_WIZARD_CANCEL(-1, wxWizard::OnWizEvent
)
91 EVT_WIZARD_FINISHED(-1, wxWizard::OnWizEvent
)
92 EVT_WIZARD_HELP(-1, wxWizard::OnWizEvent
)
95 IMPLEMENT_DYNAMIC_CLASS(wxWizard
, wxDialog
)
104 IMPLEMENT_ABSTRACT_CLASS(wxWizardPage
, wxPanel
)
105 IMPLEMENT_DYNAMIC_CLASS(wxWizardPageSimple
, wxWizardPage
)
106 IMPLEMENT_DYNAMIC_CLASS(wxWizardEvent
, wxNotifyEvent
)
108 // ============================================================================
110 // ============================================================================
112 // ----------------------------------------------------------------------------
114 // ----------------------------------------------------------------------------
116 void wxWizardPage::Init()
118 m_bitmap
= wxNullBitmap
;
121 wxWizardPage::wxWizardPage(wxWizard
*parent
,
122 const wxBitmap
& bitmap
,
123 const wxChar
*resource
)
125 Create(parent
, bitmap
, resource
);
128 bool wxWizardPage::Create(wxWizard
*parent
,
129 const wxBitmap
& bitmap
,
130 const wxChar
*resource
)
132 if ( !wxPanel::Create(parent
, -1) )
135 if ( resource
!= NULL
)
137 #if wxUSE_WX_RESOURCES
139 if ( !LoadFromResource(this, resource
) )
141 wxFAIL_MSG(wxT("wxWizardPage LoadFromResource failed!!!!"));
144 #endif // wxUSE_RESOURCES
149 // initially the page is hidden, it's shown only when it becomes current
155 // ----------------------------------------------------------------------------
156 // wxWizardPageSimple
157 // ----------------------------------------------------------------------------
159 wxWizardPage
*wxWizardPageSimple::GetPrev() const
164 wxWizardPage
*wxWizardPageSimple::GetNext() const
169 // ----------------------------------------------------------------------------
171 // ----------------------------------------------------------------------------
173 wxWizardSizer::wxWizardSizer(wxWizard
*owner
)
176 m_childSizeValid
= false;
179 void wxWizardSizer::RecalcSizes()
181 // Effect of this function depends on m_owner->m_page and
182 // it should be called whenever it changes (wxWizard::ShowPage)
183 if ( m_owner
->m_page
)
185 m_owner
->m_page
->SetSize(m_position
.x
,m_position
.y
, m_size
.x
,m_size
.y
);
189 wxSize
wxWizardSizer::CalcMin()
191 return m_owner
->GetPageSize();
194 wxSize
wxWizardSizer::GetMaxChildSize()
196 #if !defined(__WXDEBUG__)
197 if ( m_childSizeValid
)
202 wxSizerItemList::compatibility_iterator childNode
;
204 for(childNode
= m_children
.GetFirst(); childNode
;
205 childNode
= childNode
->GetNext())
207 wxSizerItem
*child
= childNode
->GetData();
208 maxOfMin
.IncTo(child
->CalcMin());
209 maxOfMin
.IncTo(SiblingSize(child
));
213 if ( m_childSizeValid
&& m_childSize
!= maxOfMin
)
215 wxFAIL_MSG( _T("Size changed in wxWizard::GetPageAreaSizer()")
216 _T("after RunWizard().\n")
217 _T("Did you forget to call GetSizer()->Fit(this) ")
218 _T("for some page?")) ;
222 #endif // __WXDEBUG__
224 if ( m_owner
->m_started
)
226 m_childSizeValid
= true;
227 m_childSize
= maxOfMin
;
233 int wxWizardSizer::Border() const
235 if ( m_owner
->m_calledSetBorder
)
236 return m_owner
->m_border
;
238 return m_children
.IsEmpty() ? 5 : 0;
241 wxSize
wxWizardSizer::SiblingSize(wxSizerItem
*child
)
245 if ( child
->IsWindow() )
247 wxWizardPage
*page
= wxDynamicCast(child
->GetWindow(), wxWizardPage
);
250 for ( wxWizardPage
*sibling
= page
->GetNext();
252 sibling
= sibling
->GetNext() )
254 if ( sibling
->GetSizer() )
256 maxSibling
.IncTo(sibling
->GetSizer()->CalcMin());
265 // ----------------------------------------------------------------------------
266 // generic wxWizard implementation
267 // ----------------------------------------------------------------------------
269 void wxWizard::Init()
271 m_posWizard
= wxDefaultPosition
;
272 m_page
= (wxWizardPage
*)NULL
;
273 m_btnPrev
= m_btnNext
= NULL
;
275 m_sizerBmpAndPage
= NULL
;
277 m_calledSetBorder
= false;
282 bool wxWizard::Create(wxWindow
*parent
,
284 const wxString
& title
,
285 const wxBitmap
& bitmap
,
289 bool result
= wxDialog::Create(parent
,id
,title
,pos
,wxDefaultSize
,style
);
299 void wxWizard::AddBitmapRow(wxBoxSizer
*mainColumn
)
301 m_sizerBmpAndPage
= new wxBoxSizer(wxHORIZONTAL
);
304 1, // Vertically stretchable
305 wxEXPAND
// Horizonal stretching, no border
308 0, // No vertical stretching
309 wxEXPAND
// No border, (mostly useless) horizontal stretching
314 m_statbmp
= new wxStaticBitmap(this, -1, m_bitmap
);
315 m_sizerBmpAndPage
->Add(
317 0, // No horizontal stretching
318 wxALL
, // Border all around, top alignment
321 m_sizerBmpAndPage
->Add(
323 0, // No horizontal stretching
324 wxEXPAND
// No border, (mostly useless) vertical stretching
328 // Added to m_sizerBmpAndPage in FinishLayout
329 m_sizerPage
= new wxWizardSizer(this);
332 void wxWizard::AddStaticLine(wxBoxSizer
*mainColumn
)
336 new wxStaticLine(this, -1),
337 0, // Vertically unstretchable
338 wxEXPAND
| wxALL
, // Border all around, horizontally stretchable
342 0, // No vertical stretching
343 wxEXPAND
// No border, (mostly useless) horizontal stretching
347 #endif // wxUSE_STATLINE
350 void wxWizard::AddBackNextPair(wxBoxSizer
*buttonRow
)
352 wxASSERT_MSG(m_btnNext
!=0 && m_btnPrev
!=0, "You must create the buttons first before calling wxWizard::AddBackNextPair");
353 // margin between Back and Next buttons
355 static const int BACKNEXT_MARGIN
= 10;
357 static const int BACKNEXT_MARGIN
= 0;
360 wxBoxSizer
*backNextPair
= new wxBoxSizer(wxHORIZONTAL
);
363 0, // No horizontal stretching
364 wxALL
, // Border all around
368 backNextPair
->Add(m_btnPrev
);
369 backNextPair
->Add(BACKNEXT_MARGIN
,0,
370 0, // No horizontal stretching
371 wxEXPAND
// No border, (mostly useless) vertical stretching
373 backNextPair
->Add(m_btnNext
);
376 void wxWizard::AddButtonRow(wxBoxSizer
*mainColumn
)
378 // the order in which the buttons are created determines the TAB order - at least under MSWindows...
379 // although the 'back' button appears before the 'next' button, a more userfriendly tab order is
380 // to activate the 'next' button first (create the next button before the back button).
381 // The reason is: The user will repeatedly enter information in the wizard pages and then wants to
382 // press 'next'. If a user uses mostly the keyboard, he would have to skip the 'back' button
383 // everytime. This is annoying. There is a second reason: RETURN acts as TAB. If the 'next'
384 // button comes first in the TAB order, the user can enter information very fast using the RETURN
385 // key to TAB to the next entry field and page. This would not be possible, if the 'back' button
386 // was created before the 'next' button.
388 wxBoxSizer
*buttonRow
= new wxBoxSizer(wxHORIZONTAL
);
391 0, // Vertically unstretchable
392 wxALIGN_RIGHT
// Right aligned, no border
395 // Desired TAB order is 'next', 'cancel', 'help', 'back'. This makes the 'back' button the last control on the page.
396 // Create the buttons in the right order...
397 m_btnNext
= new wxButton(this, wxID_FORWARD
, _("&Next >"));
398 wxButton
*btnCancel
=new wxButton(this, wxID_CANCEL
, _("&Cancel"));
400 if (GetExtraStyle() & wxWIZARD_EX_HELPBUTTON
)
401 btnHelp
=new wxButton(this, wxID_HELP
, _("&Help"));
402 m_btnPrev
= new wxButton(this, wxID_BACKWARD
, _("< &Back"));
407 0, // Horizontally unstretchable
408 wxALL
, // Border all around, top aligned
412 AddBackNextPair(buttonRow
);
416 0, // Horizontally unstretchable
417 wxALL
, // Border all around, top aligned
422 void wxWizard::DoCreateControls()
424 // do nothing if the controls were already created
428 // wxWindow::SetSizer will be called at end
429 wxBoxSizer
*windowSizer
= new wxBoxSizer(wxVERTICAL
);
431 wxBoxSizer
*mainColumn
= new wxBoxSizer(wxVERTICAL
);
434 1, // Vertical stretching
435 wxALL
| wxEXPAND
, // Border all around, horizontal stretching
439 AddBitmapRow(mainColumn
);
440 AddStaticLine(mainColumn
);
441 AddButtonRow(mainColumn
);
443 // wxWindow::SetSizer should be followed by wxWindow::Fit, but
444 // this is done in FinishLayout anyway so why duplicate it
445 SetSizer(windowSizer
);
448 void wxWizard::SetPageSize(const wxSize
& size
)
450 wxCHECK_RET(!m_started
,wxT("wxWizard::SetPageSize after RunWizard"));
454 void wxWizard::FinishLayout()
456 m_sizerBmpAndPage
->Add(
458 1, // Horizontal stretching
459 wxEXPAND
| wxALL
, // Vertically stretchable
460 m_sizerPage
->Border()
463 GetSizer()->SetSizeHints(this);
464 if ( m_posWizard
== wxDefaultPosition
)
468 void wxWizard::FitToPage(const wxWizardPage
*page
)
470 wxCHECK_RET(!m_started
,wxT("wxWizard::FitToPage after RunWizard"));
474 wxSize size
= page
->GetBestSize();
476 m_sizePage
.IncTo(size
);
478 page
= page
->GetNext();
482 bool wxWizard::ShowPage(wxWizardPage
*page
, bool goingForward
)
484 wxASSERT_MSG( page
!= m_page
, wxT("this is useless") );
486 // we'll use this to decide whether we have to change the label of this
487 // button or not (initially the label is "Next")
488 bool btnLabelWasNext
= TRUE
;
490 // Modified 10-20-2001 Robert Cavanaugh.
491 // Fixed bug for displaying a new bitmap
492 // in each *consecutive* page
494 // flag to indicate if this page uses a new bitmap
495 bool bmpIsDefault
= TRUE
;
497 // use these labels to determine if we need to change the bitmap
499 wxBitmap bmpPrev
, bmpCur
;
501 // check for previous page
504 // send the event to the old page
505 wxWizardEvent
event(wxEVT_WIZARD_PAGE_CHANGING
, GetId(), goingForward
, m_page
);
506 if ( m_page
->GetEventHandler()->ProcessEvent(event
) &&
509 // vetoed by the page
515 btnLabelWasNext
= HasNextPage(m_page
);
517 // Get the bitmap of the previous page (if it exists)
518 if ( m_page
->GetBitmap().Ok() )
520 bmpPrev
= m_page
->GetBitmap();
530 // terminate successfully
534 wxWizardEvent
event(wxEVT_WIZARD_FINISHED
, GetId(),FALSE
, 0);
535 (void)GetEventHandler()->ProcessEvent(event
);
540 // position and show the new page
541 (void)m_page
->TransferDataToWindow();
543 // wxWizardSizer::RecalcSizes wants to be called when m_page changes
544 m_sizerPage
->RecalcSizes();
546 // check if bitmap needs to be updated
547 // update default flag as well
548 if ( m_page
->GetBitmap().Ok() )
550 bmpCur
= m_page
->GetBitmap();
551 bmpIsDefault
= FALSE
;
554 // change the bitmap if:
555 // 1) a default bitmap was selected in constructor
556 // 2) this page was constructed with a bitmap
557 // 3) this bitmap is not the previous bitmap
558 if ( m_statbmp
&& (bmpCur
!= bmpPrev
) )
564 bmp
= m_page
->GetBitmap();
565 m_statbmp
->SetBitmap(bmp
);
568 // and update the buttons state
569 m_btnPrev
->Enable(HasPrevPage(m_page
));
571 bool hasNext
= HasNextPage(m_page
);
572 if ( btnLabelWasNext
!= hasNext
)
576 m_btnNext
->SetLabel(_("&Finish"));
578 m_btnNext
->SetLabel(_("&Next >"));
580 // nothing to do: the label was already correct
582 // send the change event to the new page now
583 wxWizardEvent
event(wxEVT_WIZARD_PAGE_CHANGED
, GetId(), goingForward
, m_page
);
584 (void)m_page
->GetEventHandler()->ProcessEvent(event
);
586 // and finally show it
593 bool wxWizard::RunWizard(wxWizardPage
*firstPage
)
595 wxCHECK_MSG( firstPage
, FALSE
, wxT("can't run empty wizard") );
597 // Set before FinishLayout to enable wxWizardSizer::GetMaxChildSize
600 // This cannot be done sooner, because user can change layout options
604 // can't return FALSE here because there is no old page
605 (void)ShowPage(firstPage
, TRUE
/* forward */);
607 return ShowModal() == wxID_OK
;
610 wxWizardPage
*wxWizard::GetCurrentPage() const
615 wxSize
wxWizard::GetPageSize() const
617 wxSize
pageSize(GetManualPageSize());
618 pageSize
.IncTo(m_sizerPage
->GetMaxChildSize());
622 wxSizer
*wxWizard::GetPageAreaSizer() const
627 void wxWizard::SetBorder(int border
)
629 wxCHECK_RET(!m_started
,wxT("wxWizard::SetBorder after RunWizard"));
631 m_calledSetBorder
= true;
635 wxSize
wxWizard::GetManualPageSize() const
637 // default width and height of the page
638 static const int DEFAULT_PAGE_WIDTH
= 270;
639 static const int DEFAULT_PAGE_HEIGHT
= 290;
641 wxSize
totalPageSize(DEFAULT_PAGE_WIDTH
,DEFAULT_PAGE_HEIGHT
);
643 totalPageSize
.IncTo(m_sizePage
);
647 totalPageSize
.IncTo(wxSize(0, m_bitmap
.GetHeight()));
650 return totalPageSize
;
653 void wxWizard::OnCancel(wxCommandEvent
& WXUNUSED(eventUnused
))
655 // this function probably can never be called when we don't have an active
656 // page, but a small extra check won't hurt
657 wxWindow
*win
= m_page
? (wxWindow
*)m_page
: (wxWindow
*)this;
659 wxWizardEvent
event(wxEVT_WIZARD_CANCEL
, GetId(), FALSE
, m_page
);
660 if ( !win
->GetEventHandler()->ProcessEvent(event
) || event
.IsAllowed() )
662 // no objections - close the dialog
663 EndModal(wxID_CANCEL
);
665 //else: request to Cancel ignored
668 void wxWizard::OnBackOrNext(wxCommandEvent
& event
)
670 wxASSERT_MSG( (event
.GetEventObject() == m_btnNext
) ||
671 (event
.GetEventObject() == m_btnPrev
),
672 wxT("unknown button") );
674 // ask the current page first: notice that we do it before calling
675 // GetNext/Prev() because the data transfered from the controls of the page
676 // may change the value returned by these methods
677 if ( m_page
&& (!m_page
->Validate() || !m_page
->TransferDataFromWindow()) )
679 // the page data is incorrect, don't do anything
683 bool forward
= event
.GetEventObject() == m_btnNext
;
688 page
= m_page
->GetNext();
692 page
= m_page
->GetPrev();
694 wxASSERT_MSG( page
, wxT("\"<Back\" button should have been disabled") );
697 // just pass to the new page (or may be not - but we don't care here)
698 (void)ShowPage(page
, forward
);
701 void wxWizard::OnHelp(wxCommandEvent
& WXUNUSED(event
))
703 // this function probably can never be called when we don't have an active
704 // page, but a small extra check won't hurt
707 // Create and send the help event to the specific page handler
708 // event data contains the active page so that context-sensitive
710 wxWizardEvent
eventHelp(wxEVT_WIZARD_HELP
, GetId(), TRUE
, m_page
);
711 (void)m_page
->GetEventHandler()->ProcessEvent(eventHelp
);
715 void wxWizard::OnWizEvent(wxWizardEvent
& event
)
717 // the dialogs have wxWS_EX_BLOCK_EVENTS style on by default but we want to
718 // propagate wxEVT_WIZARD_XXX to the parent (if any), so do it manually
719 if ( !(GetExtraStyle() & wxWS_EX_BLOCK_EVENTS
) )
721 // the event will be propagated anyhow
725 wxWindow
*parent
= GetParent();
727 if ( !parent
|| !parent
->GetEventHandler()->ProcessEvent(event
) )
733 // ----------------------------------------------------------------------------
734 // our public interface
735 // ----------------------------------------------------------------------------
737 #ifdef WXWIN_COMPATIBILITY_2_2
740 wxWizard
*wxWizardBase::Create(wxWindow
*parent
,
742 const wxString
& title
,
743 const wxBitmap
& bitmap
,
745 const wxSize
& WXUNUSED(size
))
747 return new wxWizard(parent
, id
, title
, bitmap
, pos
);
750 #endif // WXWIN_COMPATIBILITY_2_2
752 // ----------------------------------------------------------------------------
754 // ----------------------------------------------------------------------------
756 wxWizardEvent::wxWizardEvent(wxEventType type
, int id
, bool direction
, wxWizardPage
* page
)
757 : wxNotifyEvent(type
, id
)
759 // Modified 10-20-2001 Robert Cavanaugh
760 // add the active page to the event data
761 m_direction
= direction
;
765 #endif // wxUSE_WIZARDDLG