1 ///////////////////////////////////////////////////////////////////////////////
2 // Name: src/common/bookctrl.cpp
3 // Purpose: wxBookCtrlBase implementation
4 // Author: Vadim Zeitlin
8 // Copyright: (c) 2003 Vadim Zeitlin <vadim@wxwindows.org>
9 // Licence: wxWindows licence
10 ///////////////////////////////////////////////////////////////////////////////
12 // ============================================================================
14 // ============================================================================
16 // ----------------------------------------------------------------------------
18 // ----------------------------------------------------------------------------
20 // For compilers that support precompilation, includes "wx.h".
21 #include "wx/wxprec.h"
29 #include "wx/imaglist.h"
31 #include "wx/bookctrl.h"
33 // ============================================================================
35 // ============================================================================
37 // ----------------------------------------------------------------------------
39 // ----------------------------------------------------------------------------
41 IMPLEMENT_ABSTRACT_CLASS(wxBookCtrlBase
, wxControl
)
43 BEGIN_EVENT_TABLE(wxBookCtrlBase
, wxControl
)
44 EVT_SIZE(wxBookCtrlBase::OnSize
)
46 EVT_HELP(wxID_ANY
, wxBookCtrlBase::OnHelp
)
50 // ----------------------------------------------------------------------------
51 // constructors and destructors
52 // ----------------------------------------------------------------------------
54 void wxBookCtrlBase::Init()
58 m_ownsImageList
= false;
59 m_fitToCurrentPage
= false;
61 #if defined(__WXWINCE__)
68 m_controlSizer
= NULL
;
72 wxBookCtrlBase::Create(wxWindow
*parent
,
79 return wxControl::Create
91 wxBookCtrlBase::~wxBookCtrlBase()
93 if ( m_ownsImageList
)
100 // ----------------------------------------------------------------------------
102 // ----------------------------------------------------------------------------
104 void wxBookCtrlBase::SetImageList(wxImageList
*imageList
)
106 if ( m_ownsImageList
)
111 m_ownsImageList
= false;
114 m_imageList
= imageList
;
117 void wxBookCtrlBase::AssignImageList(wxImageList
* imageList
)
119 SetImageList(imageList
);
121 m_ownsImageList
= true;
124 // ----------------------------------------------------------------------------
126 // ----------------------------------------------------------------------------
128 void wxBookCtrlBase::DoInvalidateBestSize()
130 // notice that it is not necessary to invalidate our own best size
131 // explicitly if we have m_bookctrl as it will already invalidate the best
132 // size of its parent when its own size is invalidated and its parent is
135 m_bookctrl
->InvalidateBestSize();
137 wxControl::InvalidateBestSize();
140 void wxBookCtrlBase::SetPageSize(const wxSize
& size
)
142 SetClientSize(CalcSizeFromPage(size
));
145 wxSize
wxBookCtrlBase::DoGetBestSize() const
149 // iterate over all pages, get the largest width and height
150 const size_t nCount
= m_pages
.size();
151 for ( size_t nPage
= 0; nPage
< nCount
; nPage
++ )
153 const wxWindow
* const pPage
= m_pages
[nPage
];
156 wxSize
childBestSize(pPage
->GetBestSize());
158 if ( childBestSize
.x
> bestSize
.x
)
159 bestSize
.x
= childBestSize
.x
;
161 if ( childBestSize
.y
> bestSize
.y
)
162 bestSize
.y
= childBestSize
.y
;
166 if (m_fitToCurrentPage
&& GetCurrentPage())
167 bestSize
= GetCurrentPage()->GetBestSize();
169 // convert display area to window area, adding the size necessary for the
171 wxSize best
= CalcSizeFromPage(bestSize
);
176 wxRect
wxBookCtrlBase::GetPageRect() const
178 const wxSize size
= GetControllerSize();
181 wxRect
rectPage(pt
, GetClientSize());
183 switch ( GetWindowStyle() & wxBK_ALIGN_MASK
)
186 wxFAIL_MSG( _T("unexpected alignment") );
190 rectPage
.y
= size
.y
+ GetInternalBorder();
194 rectPage
.height
-= size
.y
+ GetInternalBorder();
195 if (rectPage
.height
< 0)
200 rectPage
.x
= size
.x
+ GetInternalBorder();
204 rectPage
.width
-= size
.x
+ GetInternalBorder();
205 if (rectPage
.width
< 0)
214 void wxBookCtrlBase::DoSize()
218 // we're not fully created yet or OnSize() should be hidden by derived class
226 // resize controller and the page area to fit inside our new size
227 const wxSize
sizeClient( GetClientSize() ),
228 sizeBorder( m_bookctrl
->GetSize() - m_bookctrl
->GetClientSize() ),
229 sizeCtrl( GetControllerSize() );
231 m_bookctrl
->SetClientSize( sizeCtrl
.x
- sizeBorder
.x
, sizeCtrl
.y
- sizeBorder
.y
);
233 const wxSize sizeNew
= m_bookctrl
->GetSize();
235 switch ( GetWindowStyle() & wxBK_ALIGN_MASK
)
238 wxFAIL_MSG( _T("unexpected alignment") );
243 // posCtrl is already ok
247 posCtrl
.y
= sizeClient
.y
- sizeNew
.y
;
251 posCtrl
.x
= sizeClient
.x
- sizeNew
.x
;
255 if ( m_bookctrl
->GetPosition() != posCtrl
)
256 m_bookctrl
->Move(posCtrl
);
259 // resize all pages to fit the new control size
260 const wxRect pageRect
= GetPageRect();
261 const unsigned pagesCount
= m_pages
.Count();
262 for ( unsigned int i
= 0; i
< pagesCount
; ++i
)
264 wxWindow
* const page
= m_pages
[i
];
267 wxASSERT_MSG( AllowNullPage(),
268 _T("Null page in a control that does not allow null pages?") );
272 page
->SetSize(pageRect
);
276 void wxBookCtrlBase::OnSize(wxSizeEvent
& event
)
283 wxSize
wxBookCtrlBase::GetControllerSize() const
288 const wxSize sizeClient
= GetClientSize(),
289 sizeBorder
= m_bookctrl
->GetSize() - m_bookctrl
->GetClientSize(),
290 sizeCtrl
= m_bookctrl
->GetBestSize() + sizeBorder
;
296 size
.x
= sizeClient
.x
;
299 else // left/right aligned
302 size
.y
= sizeClient
.y
;
308 // ----------------------------------------------------------------------------
309 // miscellaneous stuff
310 // ----------------------------------------------------------------------------
314 void wxBookCtrlBase::OnHelp(wxHelpEvent
& event
)
316 // determine where does this even originate from to avoid redirecting it
317 // back to the page which generated it (resulting in an infinite loop)
319 // notice that we have to check in the hard(er) way instead of just testing
320 // if the event object == this because the book control can have other
321 // subcontrols inside it (e.g. wxSpinButton in case of a notebook in wxUniv)
322 wxWindow
*source
= wxStaticCast(event
.GetEventObject(), wxWindow
);
323 while ( source
&& source
!= this && source
->GetParent() != this )
325 source
= source
->GetParent();
328 if ( source
&& m_pages
.Index(source
) == wxNOT_FOUND
)
330 // this event is for the book control itself, redirect it to the
331 // corresponding page
332 wxWindow
*page
= NULL
;
334 if ( event
.GetOrigin() == wxHelpEvent::Origin_HelpButton
)
336 // show help for the page under the mouse
337 const int pagePos
= HitTest(ScreenToClient(event
.GetPosition()));
339 if ( pagePos
!= wxNOT_FOUND
)
341 page
= GetPage((size_t)pagePos
);
344 else // event from keyboard or unknown source
346 // otherwise show the current page help
347 page
= GetCurrentPage();
352 // change event object to the page to avoid infinite recursion if
353 // we get this event ourselves if the page doesn't handle it
354 event
.SetEventObject(page
);
356 if ( page
->GetEventHandler()->ProcessEvent(event
) )
358 // don't call event.Skip()
363 //else: event coming from one of our pages already
370 // ----------------------------------------------------------------------------
372 // ----------------------------------------------------------------------------
375 wxBookCtrlBase::InsertPage(size_t nPage
,
377 const wxString
& WXUNUSED(text
),
378 bool WXUNUSED(bSelect
),
379 int WXUNUSED(imageId
))
381 wxCHECK_MSG( page
|| AllowNullPage(), false,
382 _T("NULL page in wxBookCtrlBase::InsertPage()") );
383 wxCHECK_MSG( nPage
<= m_pages
.size(), false,
384 _T("invalid page index in wxBookCtrlBase::InsertPage()") );
386 m_pages
.Insert(page
, nPage
);
388 page
->SetSize(GetPageRect());
390 DoInvalidateBestSize();
395 bool wxBookCtrlBase::DeletePage(size_t nPage
)
397 wxWindow
*page
= DoRemovePage(nPage
);
398 if ( !(page
|| AllowNullPage()) )
401 // delete NULL is harmless
407 wxWindow
*wxBookCtrlBase::DoRemovePage(size_t nPage
)
409 wxCHECK_MSG( nPage
< m_pages
.size(), NULL
,
410 _T("invalid page index in wxBookCtrlBase::DoRemovePage()") );
412 wxWindow
*pageRemoved
= m_pages
[nPage
];
413 m_pages
.RemoveAt(nPage
);
414 DoInvalidateBestSize();
419 int wxBookCtrlBase::GetNextPage(bool forward
) const
423 int nMax
= GetPageCount();
424 if ( nMax
-- ) // decrement it to get the last valid index
426 int nSel
= GetSelection();
428 // change selection wrapping if it becomes invalid
429 nPage
= forward
? nSel
== nMax
? 0
434 else // notebook is empty, no next page
442 int wxBookCtrlBase::DoSetSelection(size_t n
, int flags
)
444 wxCHECK_MSG( n
< GetPageCount(), wxNOT_FOUND
,
445 wxT("invalid page index in wxBookCtrlBase::DoSetSelection()") );
447 const int oldSel
= GetSelection();
449 if ( n
!= (size_t)oldSel
)
451 wxBookCtrlBaseEvent
*event
= CreatePageChangingEvent();
452 bool allowed
= false;
454 if ( flags
& SetSelection_SendEvent
)
456 event
->SetSelection(n
);
457 event
->SetOldSelection(oldSel
);
458 event
->SetEventObject(this);
460 allowed
= !GetEventHandler()->ProcessEvent(*event
) || event
->IsAllowed();
463 if ( !(flags
& SetSelection_SendEvent
) || allowed
)
465 if ( oldSel
!= wxNOT_FOUND
)
466 m_pages
[oldSel
]->Hide();
468 wxWindow
*page
= m_pages
[n
];
469 page
->SetSize(GetPageRect());
472 // change selection now to ignore the selection change event
473 UpdateSelectedPage(n
);
475 if ( flags
& SetSelection_SendEvent
)
477 // program allows the page change
478 MakeChangedEvent(*event
);
479 (void)GetEventHandler()->ProcessEvent(*event
);
490 #endif // wxUSE_BOOKCTRL