Set svn properties on .cpp files.
[wxWidgets.git] / src / osx / notebook_osx.cpp
1 ///////////////////////////////////////////////////////////////////////////////
2 // Name: src/osx/notebook_osx.cpp
3 // Purpose: implementation of wxNotebook
4 // Author: Stefan Csomor
5 // Modified by:
6 // Created: 1998-01-01
7 // RCS-ID: $Id$
8 // Copyright: (c) Stefan Csomor
9 // Licence: wxWindows licence
10 ///////////////////////////////////////////////////////////////////////////////
11
12 #include "wx/wxprec.h"
13
14 #if wxUSE_NOTEBOOK
15
16 #include "wx/notebook.h"
17
18 #ifndef WX_PRECOMP
19 #include "wx/string.h"
20 #include "wx/log.h"
21 #include "wx/app.h"
22 #include "wx/image.h"
23 #endif
24
25 #include "wx/string.h"
26 #include "wx/imaglist.h"
27 #include "wx/osx/private.h"
28
29
30 // check that the page index is valid
31 #define IS_VALID_PAGE(nPage) ((nPage) < GetPageCount())
32
33 BEGIN_EVENT_TABLE(wxNotebook, wxBookCtrlBase)
34 EVT_NOTEBOOK_PAGE_CHANGED(wxID_ANY, wxNotebook::OnSelChange)
35
36 EVT_SIZE(wxNotebook::OnSize)
37 EVT_SET_FOCUS(wxNotebook::OnSetFocus)
38 EVT_NAVIGATION_KEY(wxNotebook::OnNavigationKey)
39 END_EVENT_TABLE()
40
41 IMPLEMENT_DYNAMIC_CLASS(wxNotebook, wxBookCtrlBase)
42
43
44 // common part of all ctors
45 void wxNotebook::Init()
46 {
47 m_nSelection = -1;
48 }
49
50 // default for dynamic class
51 wxNotebook::wxNotebook()
52 {
53 Init();
54 }
55
56 // the same arguments as for wxControl
57 wxNotebook::wxNotebook( wxWindow *parent,
58 wxWindowID id,
59 const wxPoint& pos,
60 const wxSize& size,
61 long style,
62 const wxString& name )
63 {
64 Init();
65
66 Create( parent, id, pos, size, style, name );
67 }
68
69 bool wxNotebook::Create( wxWindow *parent,
70 wxWindowID id,
71 const wxPoint& pos,
72 const wxSize& size,
73 long style,
74 const wxString& name )
75 {
76 m_macIsUserPane = false ;
77
78 if (! (style & wxBK_ALIGN_MASK))
79 style |= wxBK_TOP;
80
81 if ( !wxNotebookBase::Create( parent, id, pos, size, style, name ) )
82 return false;
83
84 m_peer = wxWidgetImpl::CreateTabView(this,parent, id, pos, size, style, GetExtraStyle() );
85
86 MacPostControlCreate( pos, size );
87
88 return true ;
89 }
90
91 // dtor
92 wxNotebook::~wxNotebook()
93 {
94 }
95
96 // ----------------------------------------------------------------------------
97 // wxNotebook accessors
98 // ----------------------------------------------------------------------------
99
100 void wxNotebook::SetPadding(const wxSize& WXUNUSED(padding))
101 {
102 // unsupported by OS
103 }
104
105 void wxNotebook::SetTabSize(const wxSize& WXUNUSED(sz))
106 {
107 // unsupported by OS
108 }
109
110 void wxNotebook::SetPageSize(const wxSize& size)
111 {
112 SetSize( CalcSizeFromPage( size ) );
113 }
114
115 wxSize wxNotebook::CalcSizeFromPage(const wxSize& sizePage) const
116 {
117 return DoGetSizeFromClientSize( sizePage );
118 }
119
120 int wxNotebook::DoSetSelection(size_t nPage, int flags)
121 {
122 wxCHECK_MSG( IS_VALID_PAGE(nPage), wxNOT_FOUND, wxT("DoSetSelection: invalid notebook page") );
123
124 if ( m_nSelection == wxNOT_FOUND || nPage != (size_t)m_nSelection )
125 {
126 if ( flags & SetSelection_SendEvent )
127 {
128 if ( !SendPageChangingEvent(nPage) )
129 {
130 // vetoed by program
131 return m_nSelection;
132 }
133 //else: program allows the page change
134
135 SendPageChangedEvent(m_nSelection, nPage);
136 }
137
138 ChangePage(m_nSelection, nPage);
139 }
140 //else: no change
141
142 return m_nSelection;
143 }
144
145 bool wxNotebook::SetPageText(size_t nPage, const wxString& strText)
146 {
147 wxCHECK_MSG( IS_VALID_PAGE(nPage), false, wxT("SetPageText: invalid notebook page") );
148
149 wxNotebookPage *page = m_pages[nPage];
150 page->SetLabel(wxStripMenuCodes(strText));
151 MacSetupTabs();
152
153 return true;
154 }
155
156 wxString wxNotebook::GetPageText(size_t nPage) const
157 {
158 wxCHECK_MSG( IS_VALID_PAGE(nPage), wxEmptyString, wxT("GetPageText: invalid notebook page") );
159
160 wxNotebookPage *page = m_pages[nPage];
161
162 return page->GetLabel();
163 }
164
165 int wxNotebook::GetPageImage(size_t nPage) const
166 {
167 wxCHECK_MSG( IS_VALID_PAGE(nPage), wxNOT_FOUND, wxT("GetPageImage: invalid notebook page") );
168
169 return m_images[nPage];
170 }
171
172 bool wxNotebook::SetPageImage(size_t nPage, int nImage)
173 {
174 wxCHECK_MSG( IS_VALID_PAGE(nPage), false,
175 wxT("SetPageImage: invalid notebook page") );
176 wxCHECK_MSG( m_imageList && nImage < m_imageList->GetImageCount(), false,
177 wxT("SetPageImage: invalid image index") );
178
179 if ( nImage != m_images[nPage] )
180 {
181 // if the item didn't have an icon before or, on the contrary, did have
182 // it but has lost it now, its size will change - but if the icon just
183 // changes, it won't
184 m_images[nPage] = nImage;
185
186 MacSetupTabs() ;
187 }
188
189 return true;
190 }
191
192 // ----------------------------------------------------------------------------
193 // wxNotebook operations
194 // ----------------------------------------------------------------------------
195
196 // remove one page from the notebook, without deleting the window
197 wxNotebookPage* wxNotebook::DoRemovePage(size_t nPage)
198 {
199 wxCHECK_MSG( IS_VALID_PAGE(nPage), NULL,
200 wxT("DoRemovePage: invalid notebook page") );
201
202 wxNotebookPage* page = m_pages[nPage] ;
203 m_pages.RemoveAt(nPage);
204
205 MacSetupTabs();
206
207 if (m_nSelection >= (int)GetPageCount())
208 m_nSelection = GetPageCount() - 1;
209
210 if (m_nSelection >= 0)
211 m_pages[m_nSelection]->Show(true);
212
213 InvalidateBestSize();
214
215 return page;
216 }
217
218 // remove all pages
219 bool wxNotebook::DeleteAllPages()
220 {
221 WX_CLEAR_ARRAY(m_pages) ;
222 MacSetupTabs();
223 m_nSelection = -1 ;
224 InvalidateBestSize();
225
226 return true;
227 }
228
229 // same as AddPage() but does it at given position
230 bool wxNotebook::InsertPage(size_t nPage,
231 wxNotebookPage *pPage,
232 const wxString& strText,
233 bool bSelect,
234 int imageId )
235 {
236 if ( !wxNotebookBase::InsertPage( nPage, pPage, strText, bSelect, imageId ) )
237 return false;
238
239 wxASSERT_MSG( pPage->GetParent() == this, wxT("notebook pages must have notebook as parent") );
240
241 // don't show pages by default (we'll need to adjust their size first)
242 pPage->Show( false ) ;
243
244 pPage->SetLabel( wxStripMenuCodes(strText) );
245
246 m_images.Insert( imageId, nPage );
247
248 MacSetupTabs();
249
250 wxRect rect = GetPageRect() ;
251 pPage->SetSize( rect );
252 if ( pPage->GetAutoLayout() )
253 pPage->Layout();
254
255 // now deal with the selection
256 // ---------------------------
257
258 // if the inserted page is before the selected one, we must update the
259 // index of the selected page
260
261 if ( int(nPage) <= m_nSelection )
262 {
263 m_nSelection++;
264
265 // while this still is the same page showing, we need to update the tabs
266 m_peer->SetValue( m_nSelection + 1 ) ;
267 }
268
269 // some page should be selected: either this one or the first one if there
270 // is still no selection
271 int selNew = -1;
272 if ( bSelect )
273 selNew = nPage;
274 else if ( m_nSelection == -1 )
275 selNew = 0;
276
277 if ( selNew != -1 )
278 SetSelection( selNew );
279
280 InvalidateBestSize();
281
282 return true;
283 }
284
285 int wxNotebook::HitTest(const wxPoint& WXUNUSED(pt), long * WXUNUSED(flags)) const
286 {
287 int resultV = wxNOT_FOUND;
288 #if 0
289 const int countPages = GetPageCount();
290
291 // we have to convert from Client to Window relative coordinates
292 wxPoint adjustedPt = pt + GetClientAreaOrigin();
293 // and now to HIView native ones
294 adjustedPt.x -= MacGetLeftBorderSize() ;
295 adjustedPt.y -= MacGetTopBorderSize() ;
296
297 HIPoint hipoint= { adjustedPt.x , adjustedPt.y } ;
298 HIViewPartCode outPart = 0 ;
299 OSStatus err = HIViewGetPartHit( m_peer->GetControlRef(), &hipoint, &outPart );
300
301 int max = m_peer->GetMaximum() ;
302 if ( outPart == 0 && max > 0 )
303 {
304 // this is a hack, as unfortunately a hit on an already selected tab returns 0,
305 // so we have to go some extra miles to make sure we select something different
306 // and try again ..
307 int val = m_peer->GetValue() ;
308 int maxval = max ;
309 if ( max == 1 )
310 {
311 m_peer->SetMaximum( 2 ) ;
312 maxval = 2 ;
313 }
314
315 if ( val == 1 )
316 m_peer->SetValue( maxval ) ;
317 else
318 m_peer->SetValue( 1 ) ;
319
320 err = HIViewGetPartHit( m_peer->GetControlRef(), &hipoint, &outPart );
321
322 m_peer->SetValue( val ) ;
323 if ( max == 1 )
324 m_peer->SetMaximum( 1 ) ;
325 }
326
327 if ( outPart >= 1 && outPart <= countPages )
328 resultV = outPart - 1 ;
329
330 if (flags != NULL)
331 {
332 *flags = 0;
333
334 // we cannot differentiate better
335 if (resultV >= 0)
336 *flags |= wxBK_HITTEST_ONLABEL;
337 else
338 *flags |= wxBK_HITTEST_NOWHERE;
339 }
340 #endif
341 return resultV;
342 }
343
344 // Added by Mark Newsam
345 // When a page is added or deleted to the notebook this function updates
346 // information held in the control so that it matches the order
347 // the user would expect.
348 //
349 void wxNotebook::MacSetupTabs()
350 {
351 m_peer->SetupTabs(*this);
352 Refresh();
353 }
354
355 wxRect wxNotebook::GetPageRect() const
356 {
357 wxSize size = GetClientSize() ;
358
359 return wxRect( 0 , 0 , size.x , size.y ) ;
360 }
361
362 // ----------------------------------------------------------------------------
363 // wxNotebook callbacks
364 // ----------------------------------------------------------------------------
365
366 // @@@ OnSize() is used for setting the font when it's called for the first
367 // time because doing it in ::Create() doesn't work (for unknown reasons)
368 void wxNotebook::OnSize(wxSizeEvent& event)
369 {
370 unsigned int nCount = m_pages.Count();
371 wxRect rect = GetPageRect() ;
372
373 for ( unsigned int nPage = 0; nPage < nCount; nPage++ )
374 {
375 wxNotebookPage *pPage = m_pages[nPage];
376 pPage->SetSize(rect);
377 if ( pPage->GetAutoLayout() )
378 pPage->Layout();
379 }
380
381 // If the selected page is hidden at this point, the notebook
382 // has become visible for the first time after creation, and
383 // we postponed showing the page in ChangePage().
384 // So show the selected page now.
385 if ( m_nSelection != -1 )
386 {
387 wxNotebookPage *pPage = m_pages[m_nSelection];
388 if ( !pPage->IsShown() )
389 {
390 pPage->Show( true );
391 pPage->SetFocus();
392 }
393 }
394
395 // Processing continues to next OnSize
396 event.Skip();
397 }
398
399 void wxNotebook::OnSelChange(wxBookCtrlEvent& event)
400 {
401 // is it our tab control?
402 if ( event.GetEventObject() == this )
403 ChangePage(event.GetOldSelection(), event.GetSelection());
404
405 // we want to give others a chance to process this message as well
406 event.Skip();
407 }
408
409 void wxNotebook::OnSetFocus(wxFocusEvent& event)
410 {
411 // set focus to the currently selected page if any
412 if ( m_nSelection != -1 )
413 m_pages[m_nSelection]->SetFocus();
414
415 event.Skip();
416 }
417
418 void wxNotebook::OnNavigationKey(wxNavigationKeyEvent& event)
419 {
420 if ( event.IsWindowChange() )
421 {
422 // change pages
423 AdvanceSelection( event.GetDirection() );
424 }
425 else
426 {
427 // we get this event in 2 cases
428 //
429 // a) one of our pages might have generated it because the user TABbed
430 // out from it in which case we should propagate the event upwards and
431 // our parent will take care of setting the focus to prev/next sibling
432 //
433 // or
434 //
435 // b) the parent panel wants to give the focus to us so that we
436 // forward it to our selected page. We can't deal with this in
437 // OnSetFocus() because we don't know which direction the focus came
438 // from in this case and so can't choose between setting the focus to
439 // first or last panel child
440 wxWindow *parent = GetParent();
441
442 // the cast is here to fix a GCC ICE
443 if ( ((wxWindow*)event.GetEventObject()) == parent )
444 {
445 // no, it doesn't come from child, case (b): forward to a page
446 if ( m_nSelection != -1 )
447 {
448 // so that the page knows that the event comes from it's parent
449 // and is being propagated downwards
450 event.SetEventObject( this );
451
452 wxWindow *page = m_pages[m_nSelection];
453 if ( !page->HandleWindowEvent( event ) )
454 {
455 page->SetFocus();
456 }
457 //else: page manages focus inside it itself
458 }
459 else
460 {
461 // we have no pages - still have to give focus to _something_
462 SetFocus();
463 }
464 }
465 else
466 {
467 // it comes from our child, case (a), pass to the parent
468 if ( parent )
469 {
470 event.SetCurrentFocus( this );
471 parent->HandleWindowEvent( event );
472 }
473 }
474 }
475 }
476
477 // ----------------------------------------------------------------------------
478 // wxNotebook base class virtuals
479 // ----------------------------------------------------------------------------
480
481 #if wxUSE_CONSTRAINTS
482
483 // override these 2 functions to do nothing: everything is done in OnSize
484
485 void wxNotebook::SetConstraintSizes(bool WXUNUSED(recurse))
486 {
487 // don't set the sizes of the pages - their correct size is not yet known
488 wxControl::SetConstraintSizes( false );
489 }
490
491 bool wxNotebook::DoPhase(int WXUNUSED(nPhase))
492 {
493 return true;
494 }
495
496 #endif // wxUSE_CONSTRAINTS
497
498 void wxNotebook::Command(wxCommandEvent& WXUNUSED(event))
499 {
500 wxFAIL_MSG(wxT("wxNotebook::Command not implemented"));
501 }
502
503 // ----------------------------------------------------------------------------
504 // wxNotebook helper functions
505 // ----------------------------------------------------------------------------
506
507 // hide the currently active panel and show the new one
508 void wxNotebook::ChangePage(int nOldSel, int nSel)
509 {
510 if (nOldSel == nSel)
511 return;
512
513 if ( nOldSel != -1 )
514 m_pages[nOldSel]->Show( false );
515
516 if ( nSel != -1 )
517 {
518 wxNotebookPage *pPage = m_pages[nSel];
519 if ( IsShownOnScreen() )
520 {
521 pPage->Show( true );
522 pPage->SetFocus();
523 }
524 else
525 {
526 // Postpone Show() until the control is actually shown.
527 // Otherwise this forces the containing toplevel window
528 // to show, even if it's just being created and called
529 // AddPage() without intent to show the window yet.
530 // We Show() the selected page in our OnSize handler,
531 // unless it already is shown.
532 }
533 }
534
535 m_nSelection = nSel;
536 m_peer->SetValue( m_nSelection + 1 ) ;
537 }
538
539 bool wxNotebook::OSXHandleClicked( double WXUNUSED(timestampsec) )
540 {
541 bool status = false ;
542
543 SInt32 newSel = m_peer->GetValue() - 1 ;
544 if ( newSel != m_nSelection )
545 {
546 wxBookCtrlEvent changing(
547 wxEVT_COMMAND_NOTEBOOK_PAGE_CHANGING, m_windowId,
548 newSel , m_nSelection );
549 changing.SetEventObject( this );
550 HandleWindowEvent( changing );
551
552 if ( changing.IsAllowed() )
553 {
554 wxBookCtrlEvent event(
555 wxEVT_COMMAND_NOTEBOOK_PAGE_CHANGED, m_windowId,
556 newSel, m_nSelection );
557 event.SetEventObject( this );
558 HandleWindowEvent( event );
559 }
560 else
561 {
562 m_peer->SetValue( m_nSelection + 1 ) ;
563 }
564
565 status = true ;
566 }
567
568 return status ;
569 }
570
571 #endif