]> git.saurik.com Git - wxWidgets.git/blame - src/mac/toolbar.cpp
applied workaround patch to get rid of crashes in wxrcedit on MSW
[wxWidgets.git] / src / mac / toolbar.cpp
CommitLineData
e9576ca5
SC
1/////////////////////////////////////////////////////////////////////////////
2// Name: toolbar.cpp
3// Purpose: wxToolBar
4// Author: AUTHOR
5// Modified by:
6// Created: 04/01/98
7// RCS-ID: $Id$
2f1ae414 8// Copyright: (c) AUTHORy
e9576ca5
SC
9/////////////////////////////////////////////////////////////////////////////
10
11#ifdef __GNUG__
12#pragma implementation "toolbar.h"
13#endif
14
15#include "wx/wx.h"
519cb848
SC
16
17#if wxUSE_TOOLBAR
18
e9576ca5 19#include "wx/toolbar.h"
2f1ae414
SC
20#include "wx/notebook.h"
21#include "wx/tabctrl.h"
e9576ca5 22
2f1ae414 23#if !USE_SHARED_LIBRARY
12ed316d 24IMPLEMENT_DYNAMIC_CLASS(wxToolBar, wxToolBarBase)
e9576ca5
SC
25
26BEGIN_EVENT_TABLE(wxToolBar, wxToolBarBase)
2f1ae414
SC
27 EVT_MOUSE_EVENTS( wxToolBar::OnMouse )
28 EVT_PAINT( wxToolBar::OnPaint )
e9576ca5 29END_EVENT_TABLE()
2f1ae414 30#endif
e9576ca5 31
519cb848
SC
32#include <wx/mac/uma.h>
33
37e2cb08
SC
34// ----------------------------------------------------------------------------
35// private classes
36// ----------------------------------------------------------------------------
37
38class wxToolBarTool : public wxToolBarToolBase
39{
40public:
41 wxToolBarTool(wxToolBar *tbar,
42 int id,
43 const wxBitmap& bitmap1,
44 const wxBitmap& bitmap2,
45 bool toggle,
46 wxObject *clientData,
47 const wxString& shortHelpString,
48 const wxString& longHelpString)
49 : wxToolBarToolBase(tbar, id, bitmap1, bitmap2, toggle,
50 clientData, shortHelpString, longHelpString)
51 {
52 m_nSepCount = 0;
2f1ae414 53 m_index = -1 ;
37e2cb08
SC
54 }
55
56 wxToolBarTool(wxToolBar *tbar, wxControl *control)
57 : wxToolBarToolBase(tbar, control)
58 {
59 m_nSepCount = 1;
2f1ae414 60 m_index = -1 ;
37e2cb08
SC
61 }
62
63 // set/get the number of separators which we use to cover the space used by
64 // a control in the toolbar
65 void SetSeparatorsCount(size_t count) { m_nSepCount = count; }
66 size_t GetSeparatorsCount() const { return m_nSepCount; }
67
68 int m_index ;
69private:
70 size_t m_nSepCount;
71};
72
73
74// ============================================================================
75// implementation
76// ============================================================================
77
78// ----------------------------------------------------------------------------
79// wxToolBarTool
80// ----------------------------------------------------------------------------
81
895f5af7
SC
82const short kwxMacToolBarToolDefaultWidth = 24 ;
83const short kwxMacToolBarToolDefaultHeight = 22 ;
84const short kwxMacToolBarTopMargin = 2 ;
85const short kwxMacToolBarLeftMargin = 2 ;
86
2f1ae414 87
37e2cb08
SC
88wxToolBarToolBase *wxToolBar::CreateTool(int id,
89 const wxBitmap& bitmap1,
90 const wxBitmap& bitmap2,
91 bool toggle,
92 wxObject *clientData,
93 const wxString& shortHelpString,
94 const wxString& longHelpString)
95{
96 return new wxToolBarTool(this, id, bitmap1, bitmap2, toggle,
97 clientData, shortHelpString, longHelpString);
98}
99
100wxToolBarToolBase *wxToolBar::CreateTool(wxControl *control)
101{
102 return new wxToolBarTool(this, control);
103}
104
37e2cb08 105void wxToolBar::Init()
e9576ca5
SC
106{
107 m_maxWidth = -1;
108 m_maxHeight = -1;
895f5af7
SC
109 m_defaultWidth = kwxMacToolBarToolDefaultWidth;
110 m_defaultHeight = kwxMacToolBarToolDefaultHeight;
e9576ca5
SC
111}
112
113bool wxToolBar::Create(wxWindow *parent, wxWindowID id, const wxPoint& pos, const wxSize& size,
114 long style, const wxString& name)
115{
e9576ca5 116
519cb848
SC
117 int x = pos.x;
118 int y = pos.y;
119 int width = size.x;
120 int height = size.y;
e9576ca5 121
519cb848
SC
122 if (width <= 0)
123 width = 100;
124 if (height <= 0)
125 height = 30;
126 if (x < 0)
127 x = 0;
128 if (y < 0)
129 y = 0;
e9576ca5 130
895f5af7
SC
131 SetName(name);
132
133 m_windowStyle = style;
134 parent->AddChild(this);
135
136 m_backgroundColour = parent->GetBackgroundColour() ;
137 m_foregroundColour = parent->GetForegroundColour() ;
138
139 if (id == -1)
140 m_windowId = NewControlId();
141 else
142 m_windowId = id;
143
144 {
145 m_width = size.x ;
146 m_height = size.y ;
147 int x = pos.x ;
148 int y = pos.y ;
149 AdjustForParentClientOrigin(x, y, wxSIZE_USE_EXISTING);
150 m_x = x ;
151 m_y = y ;
152 }
a90c95aa
RR
153
154 return TRUE;
e9576ca5
SC
155}
156
157wxToolBar::~wxToolBar()
158{
7810c95b
SC
159 // we must refresh the frame size when the toolbar is deleted but the frame
160 // is not - otherwise toolbar leaves a hole in the place it used to occupy
e9576ca5
SC
161}
162
2f1ae414
SC
163PicHandle MakePict(GWorldPtr wp, GWorldPtr mask ) ;
164PicHandle MakePict(GWorldPtr wp, GWorldPtr mask )
519cb848
SC
165{
166 CGrafPtr origPort ;
167 GDHandle origDev ;
168
169 PicHandle pict; // this is the Picture we give back
170
171 RGBColor gray = { 0xCCCC ,0xCCCC , 0xCCCC } ;
2f1ae414
SC
172 RGBColor white = { 0xffff ,0xffff , 0xffff } ;
173 RGBColor black = { 0x0000 ,0x0000 , 0x0000 } ;
519cb848 174
2f1ae414
SC
175 unsigned char *maskimage = NULL ;
176 Rect portRect ;
177 GetPortBounds( wp , &portRect ) ;
178 int width = portRect.right - portRect.left ;
179 int height = portRect.bottom - portRect.top ;
519cb848 180
2f1ae414
SC
181 LockPixels( GetGWorldPixMap( wp ) ) ;
182 GetGWorld( &origPort , &origDev ) ;
519cb848 183
2f1ae414
SC
184 if ( mask )
185 {
186
187 maskimage = (unsigned char*) malloc( width * height ) ;
188 SetGWorld( mask , NULL ) ;
189 LockPixels( GetGWorldPixMap( mask ) ) ;
190 for ( int y = 0 ; y < height ; ++y )
191 {
192 for( int x = 0 ; x < width ; ++x )
193 {
194 RGBColor col ;
195
196 GetCPixel( x + portRect.left , y + portRect.top , &col ) ;
197 maskimage[y*width + x] = ( col.red == 0 ) ; // for monochrome masks
198 }
199 }
200 UnlockPixels( GetGWorldPixMap( mask ) ) ;
201 }
519cb848 202
519cb848
SC
203 SetGWorld( wp , NULL ) ;
204
2f1ae414 205 pict = OpenPicture(&portRect); // open a picture, this disables drawing
519cb848
SC
206 if(!pict)
207 return NULL;
2f1ae414
SC
208
209 RGBBackColor( &gray ) ;
210 RGBForeColor( &black ) ;
211 EraseRect(&portRect) ;
519cb848 212 RGBBackColor( &white ) ;
2f1ae414
SC
213
214 if ( maskimage )
215 {
216 for ( int y = 0 ; y < height ; ++y )
217 {
218 for( int x = 0 ; x < width ; ++x )
219 {
220 if ( maskimage[y*width + x] )
221 {
222 RGBColor col ;
223
224 GetCPixel( x + portRect.left , y + portRect.top , &col ) ;
225 SetCPixel( x + portRect.left , y + portRect.top , &col ) ;
226 }
227 }
228 }
229 free( maskimage ) ;
230 maskimage = NULL ;
231 }
232 else
233 {
234 CopyBits(GetPortBitMapForCopyBits(wp), // src PixMap - we copy image over itself -
235 GetPortBitMapForCopyBits(wp), // dst PixMap - no drawing occurs -
236 &portRect, // srcRect - it will be recorded and compressed -
237 &portRect, // dstRect - into the picture that is open -
519cb848
SC
238 srcCopy,NULL); // copyMode and no clip region
239
2f1ae414 240 }
519cb848 241 ClosePicture(); // We are done recording the picture
2f1ae414 242 UnlockPixels( GetGWorldPixMap( wp ) ) ;
519cb848
SC
243 SetGWorld( origPort , origDev ) ;
244 return pict; // return our groovy pict handle
245}
246
37e2cb08 247bool wxToolBar::Realize()
e9576ca5 248{
519cb848
SC
249 if (m_tools.Number() == 0)
250 return FALSE;
e9576ca5 251
80645074
SC
252 Point localOrigin ;
253 Rect clipRect ;
254 WindowRef window ;
255 wxWindow *win ;
256
7810c95b 257 GetParent()->MacGetPortParams( &localOrigin , &clipRect , &window , &win ) ;
80645074
SC
258
259 Rect toolbarrect = { m_y + localOrigin.v , m_x + localOrigin.h ,
260 m_y + m_height + localOrigin.v , m_x + m_width + localOrigin.h} ;
519cb848 261 ControlFontStyleRec controlstyle ;
80645074 262
519cb848
SC
263 controlstyle.flags = kControlUseFontMask ;
264 controlstyle.font = kControlFontSmallSystemFont ;
265
266 wxNode *node = m_tools.First();
267 int noButtons = 0;
268 int x = 0 ;
7c551d95
SC
269 wxSize toolSize = GetToolSize() ;
270 int tw, th;
271 GetSize(& tw, & th);
895f5af7
SC
272
273 int maxWidth = 0 ;
274 int maxHeight = 0 ;
275
519cb848
SC
276 while (node)
277 {
278 wxToolBarTool *tool = (wxToolBarTool *)node->Data();
37e2cb08 279 wxBitmapRefData * bmap = (wxBitmapRefData*) ( tool->GetBitmap1().GetRefData()) ;
519cb848 280
2f1ae414 281 if( !tool->IsSeparator() )
519cb848 282 {
895f5af7 283 Rect toolrect = { toolbarrect.top + m_yMargin + kwxMacToolBarTopMargin, toolbarrect.left + x + m_xMargin + kwxMacToolBarLeftMargin , 0 , 0 } ;
7c551d95
SC
284 toolrect.right = toolrect.left + toolSize.x ;
285 toolrect.bottom = toolrect.top + toolSize.y ;
519cb848
SC
286
287 PicHandle icon = NULL ;
288 if ( bmap )
289 {
290 if ( bmap->m_bitmapType == kMacBitmapTypePict )
291 icon = bmap->m_hPict ;
292 else if ( bmap->m_bitmapType == kMacBitmapTypeGrafWorld )
293 {
2f1ae414
SC
294 if ( tool->GetBitmap1().GetMask() )
295 {
296 icon = MakePict( bmap->m_hBitmap , tool->GetBitmap1().GetMask()->GetMaskBitmap() ) ;
297 }
298 else
299 {
300 icon = MakePict( bmap->m_hBitmap , NULL ) ;
301 }
519cb848
SC
302 }
303 }
304
305 ControlHandle m_macToolHandle ;
2f1ae414 306
8208e181
SC
307 SInt16 behaviour = kControlBehaviorOffsetContents ;
308 if ( tool->CanBeToggled() )
309 behaviour += kControlBehaviorToggles ;
310
519cb848
SC
311 if ( icon )
312 {
313 m_macToolHandle = UMANewControl( window , &toolrect , "\p" , true , 0 ,
8208e181 314 behaviour + kControlContentPictHandle , 0 , kControlBevelButtonNormalBevelProc , (long) this ) ;
519cb848
SC
315 ControlButtonContentInfo info ;
316
317 info.contentType = kControlContentPictHandle ;
318 info.u.picture = icon ;
319
320 UMASetControlData( m_macToolHandle , kControlButtonPart , kControlBevelButtonContentTag , sizeof(info) , (char*) &info ) ;
321 }
322 else
323 {
324 m_macToolHandle = UMANewControl( window , &toolrect , "\p" , true , 0 ,
8208e181 325 behaviour , 0 , kControlBevelButtonNormalBevelProc , (long) this ) ;
519cb848
SC
326 }
327 m_macToolHandles.Add( m_macToolHandle ) ;
2f1ae414
SC
328 tool->m_index = m_macToolHandles.Count() -1 ;
329 if ( !tool->IsEnabled() )
330 {
331 UMADeactivateControl( m_macToolHandle ) ;
332 }
333 if ( tool->CanBeToggled() && tool->IsToggled() )
334 {
335 ::SetControlValue( m_macToolHandle , 1 ) ;
336 }
519cb848 337 UMASetControlFontStyle( m_macToolHandle , &controlstyle ) ;
2f1ae414
SC
338 ControlHandle container = GetParent()->MacGetContainerForEmbedding() ;
339 wxASSERT_MSG( container != NULL , "No valid mac container control" ) ;
340 UMAEmbedControl( m_macToolHandle , container ) ;
519cb848 341
7c551d95 342 x += (int)toolSize.x;
519cb848
SC
343 noButtons ++;
344 }
345 else
346 {
347 m_macToolHandles.Add( NULL ) ;
7c551d95 348 x += (int)toolSize.x / 4;
519cb848 349 }
895f5af7
SC
350 if ( toolbarrect.left + x + m_xMargin + kwxMacToolBarLeftMargin- m_x - localOrigin.h > maxWidth)
351 maxWidth = toolbarrect.left + x + kwxMacToolBarLeftMargin+ m_xMargin - m_x - localOrigin.h;
352 if (toolbarrect.top + m_yMargin + kwxMacToolBarTopMargin - m_y - localOrigin.v > maxHeight)
353 maxHeight = toolbarrect.top + kwxMacToolBarTopMargin + m_yMargin - m_y - localOrigin.v ;
7c551d95 354
519cb848
SC
355 node = node->Next();
356 }
357
7c551d95
SC
358 if ( GetWindowStyleFlag() & wxTB_HORIZONTAL )
359 {
7810c95b
SC
360 if ( m_maxRows == 0 )
361 {
362 // if not set yet, only one row
363 SetRows(1);
364 }
895f5af7
SC
365 maxWidth = tw ;
366 maxHeight += toolSize.y;
367 maxHeight += m_yMargin + kwxMacToolBarTopMargin;
368 m_maxHeight = maxHeight ;
7c551d95
SC
369 }
370 else
371 {
7810c95b
SC
372 if ( noButtons > 0 && m_maxRows == 0 )
373 {
374 // if not set yet, have one column
375 SetRows(noButtons);
376 }
895f5af7
SC
377 maxHeight = th ;
378 maxWidth += toolSize.x;
379 maxWidth += m_xMargin + kwxMacToolBarLeftMargin;
380 m_maxWidth = maxWidth ;
7c551d95
SC
381 }
382
895f5af7 383 SetSize(maxWidth, maxHeight);
7c551d95 384
519cb848 385 return TRUE;
e9576ca5
SC
386}
387
388void wxToolBar::SetToolBitmapSize(const wxSize& size)
389{
a90c95aa 390 m_defaultWidth = size.x+2; m_defaultHeight = size.y+2;
e9576ca5
SC
391}
392
e9576ca5
SC
393// The button size is bigger than the bitmap size
394wxSize wxToolBar::GetToolSize() const
395{
2f1ae414 396 return wxSize(m_defaultWidth + 4, m_defaultHeight + 4);
e9576ca5
SC
397}
398
519cb848
SC
399void wxToolBar::MacHandleControlClick( ControlHandle control , SInt16 controlpart )
400{
401 int index = 0 ;
402 for ( index = 0 ; index < m_macToolHandles.Count() ; ++index )
403 {
404 if ( m_macToolHandles[index] == (void*) control )
405 {
2f1ae414
SC
406 wxToolBarTool *tool = (wxToolBarTool *)m_tools.Nth( index )->Data();
407 if ( tool->CanBeToggled() )
408 {
409 tool->Toggle( GetControlValue( control ) ) ;
410 }
411 OnLeftClick( tool->GetId() , tool -> IsToggled() ) ;
412 break ;
519cb848
SC
413 }
414 }
415}
416
37e2cb08 417void wxToolBar::SetRows(int nRows)
e9576ca5 418{
37e2cb08 419 if ( nRows == m_maxRows )
e9576ca5 420 {
37e2cb08
SC
421 // avoid resizing the frame uselessly
422 return;
e9576ca5 423 }
37e2cb08
SC
424
425 m_maxRows = nRows;
e9576ca5
SC
426}
427
37e2cb08 428wxToolBarToolBase *wxToolBar::FindToolForPosition(wxCoord x, wxCoord y) const
e9576ca5 429{
37e2cb08 430 MacClientToRootWindow( &x , &y ) ;
2f1ae414 431 Point pt = { y ,x } ;
37e2cb08
SC
432
433 int index = 0 ;
434 for ( index = 0 ; index < m_macToolHandles.Count() ; ++index )
435 {
2f1ae414 436 if ( m_macToolHandles[index] )
37e2cb08 437 {
2f1ae414
SC
438 Rect bounds ;
439 GetControlBounds((ControlHandle) m_macToolHandles[index], &bounds ) ;
440 if ( PtInRect( pt , &bounds ) )
441 {
442 return (wxToolBarTool*) (m_tools.Nth( index )->Data() ) ;
443 }
37e2cb08
SC
444 }
445 }
446
447 return (wxToolBarToolBase *)NULL;
e9576ca5
SC
448}
449
2f1ae414
SC
450wxString wxToolBar::MacGetToolTipString( wxPoint &pt )
451{
452 wxToolBarToolBase* tool = FindToolForPosition( pt.x , pt.y ) ;
453 if ( tool )
454 {
455 return tool->GetShortHelp() ;
456 }
457 return "" ;
458}
459
37e2cb08 460void wxToolBar::DoEnableTool(wxToolBarToolBase *t, bool enable)
e9576ca5 461{
37e2cb08 462 wxToolBarTool *tool = (wxToolBarTool *)t;
2f1ae414
SC
463 if ( tool->m_index < 0 )
464 return ;
465
8208e181 466 ControlHandle control = (ControlHandle) m_macToolHandles[ tool->m_index ] ;
2f1ae414
SC
467
468 if ( enable )
469 UMAActivateControl( control ) ;
8208e181 470 else
2f1ae414 471 UMADeactivateControl( control ) ;
e9576ca5
SC
472}
473
37e2cb08 474void wxToolBar::DoToggleTool(wxToolBarToolBase *t, bool toggle)
e9576ca5 475{
37e2cb08 476 wxToolBarTool *tool = (wxToolBarTool *)t;
2f1ae414
SC
477 if ( tool->m_index < 0 )
478 return ;
8208e181
SC
479
480 ControlHandle control = (ControlHandle) m_macToolHandles[ tool->m_index ] ;
481 ::SetControlValue( control , toggle ) ;
37e2cb08 482}
7c551d95 483
37e2cb08
SC
484bool wxToolBar::DoInsertTool(size_t WXUNUSED(pos),
485 wxToolBarToolBase *tool)
486{
487 // nothing special to do here - we really create the toolbar buttons in
488 // Realize() later
489 tool->Attach(this);
7c551d95 490
37e2cb08
SC
491 return TRUE;
492}
e9576ca5 493
37e2cb08
SC
494void wxToolBar::DoSetToggle(wxToolBarToolBase *t, bool toggle)
495{
496 wxToolBarTool *tool = (wxToolBarTool *)t;
497 // TODO: set toggle state
e9576ca5
SC
498}
499
37e2cb08
SC
500bool wxToolBar::DoDeleteTool(size_t pos, wxToolBarToolBase *tool)
501{
502 return TRUE ;
503}
2f1ae414
SC
504
505void wxToolBar::OnPaint(wxPaintEvent& event)
506{
80645074
SC
507 Point localOrigin ;
508 Rect clipRect ;
509 WindowRef window ;
510 wxWindow *win ;
511
7810c95b 512 GetParent()->MacGetPortParams( &localOrigin , &clipRect , &window , &win ) ;
80645074
SC
513 if ( window && win )
514 {
515 wxMacDrawingHelper help( win ) ;
516 // the mac control manager always assumes to have the origin at 0,0
517 SetOrigin( 0 , 0 ) ;
518
519 bool hasTabBehind = false ;
520 wxWindow* parent = GetParent() ;
521 while ( parent )
522 {
523 if( parent->MacGetWindowData() )
524 {
525 UMASetThemeWindowBackground( win->MacGetWindowData()->m_macWindow , kThemeBrushDialogBackgroundActive , false ) ;
526 break ;
527 }
528
529 if( parent->IsKindOf( CLASSINFO( wxNotebook ) ) || parent->IsKindOf( CLASSINFO( wxTabCtrl ) ))
530 {
531 if ( ((wxControl*)parent)->GetMacControl() )
532 SetUpControlBackground( ((wxControl*)parent)->GetMacControl() , -1 , true ) ;
533 break ;
534 }
535
536 parent = parent->GetParent() ;
537 }
895f5af7 538
80645074 539 Rect toolbarrect = { m_y + localOrigin.v , m_x + localOrigin.h ,
895f5af7 540 m_y + localOrigin.v + m_height , m_x + localOrigin.h + m_width } ;
80645074
SC
541
542 UMADrawThemePlacard( &toolbarrect , IsEnabled() ? kThemeStateActive : kThemeStateInactive) ;
543 {
544 int index = 0 ;
545 for ( index = 0 ; index < m_macToolHandles.Count() ; ++index )
546 {
547 if ( m_macToolHandles[index] )
548 {
549 UMADrawControl( (ControlHandle) m_macToolHandles[index] ) ;
550 }
551 }
552 }
553 UMASetThemeWindowBackground( win->MacGetWindowData()->m_macWindow , win->MacGetWindowData()->m_macWindowBackgroundTheme , false ) ;
80645074 554 }
2f1ae414 555}
895f5af7 556
2f1ae414
SC
557void wxToolBar::OnMouse( wxMouseEvent &event )
558{
559
560 if (event.GetEventType() == wxEVT_LEFT_DOWN || event.GetEventType() == wxEVT_LEFT_DCLICK )
561 {
562
563 int x = event.m_x ;
564 int y = event.m_y ;
565
566 MacClientToRootWindow( &x , &y ) ;
567
568 ControlHandle control ;
569 Point localwhere ;
570 GrafPtr port ;
571 SInt16 controlpart ;
572 WindowRef window = GetMacRootWindow() ;
573
574 localwhere.h = x ;
575 localwhere.v = y ;
576
577 short modifiers = 0;
578
579 if ( !event.m_leftDown && !event.m_rightDown )
580 modifiers |= btnState ;
581
582 if ( event.m_shiftDown )
583 modifiers |= shiftKey ;
584
585 if ( event.m_controlDown )
586 modifiers |= controlKey ;
587
588 if ( event.m_altDown )
589 modifiers |= optionKey ;
590
591 if ( event.m_metaDown )
592 modifiers |= cmdKey ;
593
594 controlpart = FindControl( localwhere , window , &control ) ;
595 {
2f1ae414
SC
596 if ( control && UMAIsControlActive( control ) )
597 {
598 {
599 if ( controlpart == kControlIndicatorPart && !UMAHasAppearance() )
600 controlpart = UMAHandleControlClick( control , localwhere , modifiers , (ControlActionUPP) NULL ) ;
601 else
602 controlpart = UMAHandleControlClick( control , localwhere , modifiers , (ControlActionUPP) -1 ) ;
603 wxTheApp->s_lastMouseDown = 0 ;
604 if ( controlpart && ! ( ( UMAHasAppearance() || (controlpart != kControlIndicatorPart) )
605 && (IsKindOf( CLASSINFO( wxScrollBar ) ) ) ) ) // otherwise we will get the event twice
606 {
607 MacHandleControlClick( control , controlpart ) ;
608 }
609 }
610 }
611 }
612 }
613}
614
519cb848
SC
615#endif // wxUSE_TOOLBAR
616