]> git.saurik.com Git - wxWidgets.git/blame - src/osx/cocoa/button.mm
Applied Blit system options optimization to StretchBlit
[wxWidgets.git] / src / osx / cocoa / button.mm
CommitLineData
f033830e
SC
1/////////////////////////////////////////////////////////////////////////////
2// Name: src/osx/cocoa/button.mm
3// Purpose: wxButton
4// Author: Stefan Csomor
5// Modified by:
6// Created: 1998-01-01
a9a4f229 7// RCS-ID: $Id$
f033830e
SC
8// Copyright: (c) Stefan Csomor
9// Licence: wxWindows licence
10/////////////////////////////////////////////////////////////////////////////
11
12#include "wx/wxprec.h"
13
14#include "wx/button.h"
15
16#ifndef WX_PRECOMP
f033830e
SC
17#endif
18
f033830e
SC
19#include "wx/osx/private.h"
20
f672c969
VZ
21#if wxUSE_MARKUP
22 #include "wx/osx/cocoa/private/markuptoattr.h"
23#endif // wxUSE_MARKUP
24
25
f033830e
SC
26@implementation wxNSButton
27
4dd9fdf8 28+ (void)initialize
f033830e 29{
4dd9fdf8 30 static BOOL initialized = NO;
b727fcd3 31 if (!initialized)
f033830e 32 {
4dd9fdf8
SC
33 initialized = YES;
34 wxOSXCocoaClassAddWXMethods( self );
f033830e
SC
35 }
36}
37
f033830e
SC
38- (int) intValue
39{
40 switch ( [self state] )
41 {
42 case NSOnState:
43 return 1;
44 case NSMixedState:
45 return 2;
46 default:
47 return 0;
48 }
49}
50
51- (void) setIntValue: (int) v
52{
53 switch( v )
54 {
55 case 2:
56 [self setState:NSMixedState];
57 break;
58 case 1:
59 [self setState:NSOnState];
60 break;
61 default :
62 [self setState:NSOffState];
63 break;
64 }
65}
66
411a1c35
SC
67- (void) setTrackingTag: (NSTrackingRectTag)tag
68{
69 rectTag = tag;
70}
71
72- (NSTrackingRectTag) trackingTag
73{
74 return rectTag;
75}
76
f033830e
SC
77@end
78
58ce18f2
SC
79@interface NSView(PossibleSizeMethods)
80- (NSControlSize)controlSize;
81@end
82
e5d05b90
VZ
83namespace
84{
85
b38dc31f 86class wxButtonCocoaImpl : public wxWidgetCocoaImpl, public wxButtonImpl
e5d05b90
VZ
87{
88public:
89 wxButtonCocoaImpl(wxWindowMac *wxpeer, wxNSButton *v)
90 : wxWidgetCocoaImpl(wxpeer, v)
91 {
24e059c3 92 SetNeedsFrame(false);
e5d05b90
VZ
93 }
94
95 virtual void SetBitmap(const wxBitmap& bitmap)
96 {
b38dc31f 97 // switch bezel style for plain pushbuttons
b8702c67
SC
98 if ( bitmap.IsOk() )
99 {
100 if ([GetNSButton() bezelStyle] == NSRoundedBezelStyle)
101 [GetNSButton() setBezelStyle:NSRegularSquareBezelStyle];
102 }
103 else
104 {
105 [GetNSButton() setBezelStyle:NSRoundedBezelStyle];
106 }
107
e5d05b90
VZ
108 wxWidgetCocoaImpl::SetBitmap(bitmap);
109 }
110
6a219e34 111#if wxUSE_MARKUP
f672c969
VZ
112 virtual void SetLabelMarkup(const wxString& markup)
113 {
114 wxMarkupToAttrString toAttr(GetWXPeer(), markup);
115 NSMutableAttributedString *attrString = toAttr.GetNSAttributedString();
116
117 // Button text is always centered.
118 NSMutableParagraphStyle *
119 paragraphStyle = [[NSMutableParagraphStyle alloc] init];
120 [paragraphStyle setAlignment: NSCenterTextAlignment];
121 [attrString addAttribute:NSParagraphStyleAttributeName
122 value:paragraphStyle
123 range:NSMakeRange(0, [attrString length])];
124 [paragraphStyle release];
125
126 [GetNSButton() setAttributedTitle:attrString];
127 }
6a219e34 128#endif // wxUSE_MARKUP
f672c969 129
b38dc31f
SC
130 void SetPressedBitmap( const wxBitmap& bitmap )
131 {
132 NSButton* button = GetNSButton();
133 [button setAlternateImage: bitmap.GetNSImage()];
134 [button setButtonType:NSMomentaryChangeButton];
135 }
136
54ea2834
SC
137 void GetLayoutInset(int &left , int &top , int &right, int &bottom) const
138 {
139 left = top = right = bottom = 0;
140 NSControlSize size = NSRegularControlSize;
141 if ( [m_osxView respondsToSelector:@selector(controlSize)] )
58ce18f2 142 size = [m_osxView controlSize];
54ea2834
SC
143 else if ([m_osxView respondsToSelector:@selector(cell)])
144 {
145 id cell = [(id)m_osxView cell];
146 if ([cell respondsToSelector:@selector(controlSize)])
147 size = [cell controlSize];
148 }
149
150 if ( [GetNSButton() bezelStyle] == NSRoundedBezelStyle )
151 {
152 switch( size )
153 {
154 case NSRegularControlSize:
155 left = right = 6;
156 top = 4;
157 bottom = 8;
158 break;
159 case NSSmallControlSize:
160 left = right = 5;
161 top = 4;
162 bottom = 7;
163 break;
164 case NSMiniControlSize:
165 left = right = 1;
166 top = 0;
167 bottom = 2;
168 break;
169 }
170 }
171 }
172
173
e5d05b90
VZ
174private:
175 NSButton *GetNSButton() const
176 {
177 wxASSERT( [m_osxView isKindOfClass:[NSButton class]] );
178
179 return static_cast<NSButton *>(m_osxView);
180 }
181};
182
df04f800
SC
183} // anonymous namespace
184
7f08aa6c
SC
185extern "C" void SetBezelStyleFromBorderFlags(NSButton *v, long style);
186
73b1b996
VZ
187// set bezel style depending on the wxBORDER_XXX flags specified by the style
188void SetBezelStyleFromBorderFlags(NSButton *v, long style)
189{
190 if ( style & wxBORDER_NONE )
191 {
192 [v setBezelStyle:NSShadowlessSquareBezelStyle];
193 [v setBordered:NO];
194 }
195 else // we do have a border
196 {
197 // see trac #11128 for a thorough discussion
198 if ( (style & wxBORDER_MASK) == wxBORDER_RAISED )
199 [v setBezelStyle:NSRegularSquareBezelStyle];
200 else if ( (style & wxBORDER_MASK) == wxBORDER_SUNKEN )
201 [v setBezelStyle:NSSmallSquareBezelStyle];
24e059c3 202 else if ( (style & wxBORDER_MASK) == wxBORDER_SIMPLE )
73b1b996 203 [v setBezelStyle:NSShadowlessSquareBezelStyle];
24e059c3
SC
204 else
205 [v setBezelStyle:NSRegularSquareBezelStyle];
73b1b996
VZ
206 }
207}
208
f033830e 209
b727fcd3
VZ
210wxWidgetImplType* wxWidgetImpl::CreateButton( wxWindowMac* wxpeer,
211 wxWindowMac* WXUNUSED(parent),
212 wxWindowID id,
01495abf 213 const wxString& label,
b727fcd3 214 const wxPoint& pos,
f033830e 215 const wxSize& size,
24e059c3 216 long style,
b727fcd3 217 long WXUNUSED(extraStyle))
f033830e 218{
dbeddfb9 219 NSRect r = wxOSXGetFrameForControl( wxpeer, pos , size ) ;
f033830e 220 wxNSButton* v = [[wxNSButton alloc] initWithFrame:r];
b727fcd3 221
01495abf
VZ
222 // We can't display a custom label inside a button with help bezel style so
223 // we only use it if we are using the default label. wxButton itself checks
224 // if the label is just "Help" in which case it discards it and passes us
225 // an empty string.
226 if ( id == wxID_HELP && label.empty() )
f033830e
SC
227 {
228 [v setBezelStyle:NSHelpButtonBezelStyle];
229 }
230 else
231 {
24e059c3
SC
232 if ( style & wxBORDER_NONE )
233 {
234 [v setBezelStyle:NSShadowlessSquareBezelStyle];
235 [v setBordered:NO];
236 }
237 else
238 {
239 // the following styles only exist for certain sizes, so avoid them for
240 // multi-line
241 if ( label.Find('\n' ) == wxNOT_FOUND && label.Find('\r' ) == wxNOT_FOUND)
242 {
243 if ( (style & wxBORDER_MASK) == wxBORDER_RAISED )
244 [v setBezelStyle:NSRoundedBezelStyle];
245 else if ( (style & wxBORDER_MASK) == wxBORDER_SUNKEN )
246 [v setBezelStyle:NSTexturedRoundedBezelStyle];
247 else if ( (style & wxBORDER_MASK) == wxBORDER_SIMPLE )
248 [v setBezelStyle:NSShadowlessSquareBezelStyle];
249 else
250 [v setBezelStyle:NSRoundedBezelStyle];
251 }
252 else
253 {
254 if ( (style & wxBORDER_MASK) == wxBORDER_RAISED )
255 [v setBezelStyle:NSRegularSquareBezelStyle];
256 else if ( (style & wxBORDER_MASK) == wxBORDER_SUNKEN )
257 [v setBezelStyle:NSSmallSquareBezelStyle];
258 else if ( (style & wxBORDER_MASK) == wxBORDER_SIMPLE )
259 [v setBezelStyle:NSShadowlessSquareBezelStyle];
260 else
261 [v setBezelStyle:NSRegularSquareBezelStyle];
262 }
b727fcd3 263
24e059c3
SC
264 }
265 }
266
f033830e 267 [v setButtonType:NSMomentaryPushInButton];
e5d05b90 268 return new wxButtonCocoaImpl( wxpeer, v );
f033830e
SC
269}
270
271void wxWidgetCocoaImpl::SetDefaultButton( bool isDefault )
b727fcd3 272{
3f30bd1a
SC
273 if ( [m_osxView isKindOfClass:[NSButton class]] )
274 {
275 if ( isDefault )
276 [(NSButton*)m_osxView setKeyEquivalent: @"\r" ];
277 else
278 [(NSButton*)m_osxView setKeyEquivalent: @"" ];
279 }
f033830e
SC
280}
281
b727fcd3 282void wxWidgetCocoaImpl::PerformClick()
f033830e 283{
73b1b996
VZ
284 if ([m_osxView isKindOfClass:[NSControl class]])
285 [(NSControl*)m_osxView performClick:nil];
f033830e
SC
286}
287
b38dc31f
SC
288#if wxUSE_BMPBUTTON
289
290wxWidgetImplType* wxWidgetImpl::CreateBitmapButton( wxWindowMac* wxpeer,
291 wxWindowMac* WXUNUSED(parent),
292 wxWindowID WXUNUSED(id),
293 const wxBitmap& bitmap,
294 const wxPoint& pos,
295 const wxSize& size,
296 long style,
297 long WXUNUSED(extraStyle))
298{
299 NSRect r = wxOSXGetFrameForControl( wxpeer, pos , size ) ;
300 wxNSButton* v = [[wxNSButton alloc] initWithFrame:r];
73b1b996
VZ
301
302 SetBezelStyleFromBorderFlags(v, style);
303
a1b806b9 304 if (bitmap.IsOk())
b38dc31f 305 [v setImage:bitmap.GetNSImage() ];
73b1b996 306
b38dc31f
SC
307 [v setButtonType:NSMomentaryPushInButton];
308 wxWidgetCocoaImpl* c = new wxButtonCocoaImpl( wxpeer, v );
309 return c;
310}
311
73b1b996 312#endif // wxUSE_BMPBUTTON
b38dc31f 313
17ad51ed
SC
314//
315// wxDisclosureButton implementation
316//
317
318@interface wxDisclosureNSButton : NSButton
319{
320
321 BOOL isOpen;
322}
323
324- (void) updateImage;
325
326- (void) toggle;
327
328+ (NSImage *)rotateImage: (NSImage *)image;
329
330@end
331
da52d42b
SC
332static const char * disc_triangle_xpm[] = {
333"10 9 4 1",
334" c None",
335". c #737373",
336"+ c #989898",
337"- c #c6c6c6",
338" .- ",
339" ..+- ",
340" ....+ ",
341" ......- ",
342" .......- ",
343" ......- ",
344" ....+ ",
345" ..+- ",
346" .- ",
347};
348
17ad51ed
SC
349@implementation wxDisclosureNSButton
350
351+ (void)initialize
352{
353 static BOOL initialized = NO;
b727fcd3 354 if (!initialized)
17ad51ed
SC
355 {
356 initialized = YES;
357 wxOSXCocoaClassAddWXMethods( self );
358 }
359}
360
361- (id) initWithFrame:(NSRect) frame
362{
363 self = [super initWithFrame:frame];
17ad51ed
SC
364 isOpen = NO;
365 [self setImagePosition:NSImageLeft];
366 [self updateImage];
367 return self;
368}
369
370- (int) intValue
371{
372 return isOpen ? 1 : 0;
373}
374
375- (void) setIntValue: (int) v
376{
377 isOpen = ( v != 0 );
378 [self updateImage];
379}
380
381- (void) toggle
382{
383 isOpen = !isOpen;
384 [self updateImage];
385}
386
387wxCFRef<NSImage*> downArray ;
388
389- (void) updateImage
390{
da52d42b 391 static wxBitmap trianglebm(disc_triangle_xpm);
17ad51ed
SC
392 if ( downArray.get() == NULL )
393 {
adf264f2 394 downArray.reset( [[wxDisclosureNSButton rotateImage:trianglebm.GetNSImage()] retain] );
17ad51ed 395 }
b727fcd3 396
17ad51ed
SC
397 if ( isOpen )
398 [self setImage:(NSImage*)downArray.get()];
399 else
da52d42b 400 [self setImage:trianglebm.GetNSImage()];
17ad51ed
SC
401}
402
403+ (NSImage *)rotateImage: (NSImage *)image
404{
405 NSSize imageSize = [image size];
406 NSSize newImageSize = NSMakeSize(imageSize.height, imageSize.width);
407 NSImage* newImage = [[NSImage alloc] initWithSize: newImageSize];
b727fcd3 408
17ad51ed 409 [newImage lockFocus];
b727fcd3 410
17ad51ed
SC
411 NSAffineTransform* tm = [NSAffineTransform transform];
412 [tm translateXBy:newImageSize.width/2 yBy:newImageSize.height/2];
413 [tm rotateByDegrees:-90];
414 [tm translateXBy:-newImageSize.width/2 yBy:-newImageSize.height/2];
415 [tm concat];
b727fcd3
VZ
416
417
17ad51ed
SC
418 [image drawInRect:NSMakeRect(0,0,newImageSize.width, newImageSize.height)
419 fromRect:NSZeroRect operation:NSCompositeCopy fraction:1.0];
b727fcd3 420
17ad51ed 421 [newImage unlockFocus];
adf264f2 422 return [newImage autorelease];
17ad51ed
SC
423}
424
425@end
426
427class wxDisclosureTriangleCocoaImpl : public wxWidgetCocoaImpl
428{
429public :
430 wxDisclosureTriangleCocoaImpl(wxWindowMac* peer , WXWidget w) :
431 wxWidgetCocoaImpl(peer, w)
432 {
433 }
b727fcd3 434
17ad51ed
SC
435 ~wxDisclosureTriangleCocoaImpl()
436 {
437 }
438
439 virtual void controlAction(WXWidget slf, void* _cmd, void *sender)
440 {
441 wxDisclosureNSButton* db = (wxDisclosureNSButton*)m_osxView;
442 [db toggle];
443 wxWidgetCocoaImpl::controlAction(slf, _cmd, sender );
444 }
445};
446
b727fcd3
VZ
447wxWidgetImplType* wxWidgetImpl::CreateDisclosureTriangle( wxWindowMac* wxpeer,
448 wxWindowMac* WXUNUSED(parent),
73b1b996 449 wxWindowID WXUNUSED(winid),
dbeddfb9 450 const wxString& label,
b727fcd3 451 const wxPoint& pos,
dbeddfb9 452 const wxSize& size,
73b1b996 453 long style,
b727fcd3 454 long WXUNUSED(extraStyle))
dbeddfb9 455{
dbeddfb9 456 NSRect r = wxOSXGetFrameForControl( wxpeer, pos , size ) ;
17ad51ed 457 wxDisclosureNSButton* v = [[wxDisclosureNSButton alloc] initWithFrame:r];
73b1b996
VZ
458 if ( !label.empty() )
459 [v setTitle:wxCFStringRef(label).AsNSString()];
460
461 SetBezelStyleFromBorderFlags(v, style);
462
463 return new wxDisclosureTriangleCocoaImpl( wxpeer, v );
dbeddfb9 464}