Files
wxWidgets/src/osx/cocoa/button.mm
sbrowne 723e61a4b5 Adjust inset border values for OS X >= 10.6.
The bottom border was off by one pixel, use the values tested to work for OS X
10.[6789].

See #16808.

(this is a backport of cb0625b9d9 from master)
2015-08-26 15:59:38 +02:00

511 lines
14 KiB
Plaintext

/////////////////////////////////////////////////////////////////////////////
// Name: src/osx/cocoa/button.mm
// Purpose: wxButton
// Author: Stefan Csomor
// Modified by:
// Created: 1998-01-01
// Copyright: (c) Stefan Csomor
// Licence: wxWindows licence
/////////////////////////////////////////////////////////////////////////////
#include "wx/wxprec.h"
#ifndef WX_PRECOMP
#include "wx/object.h"
#endif
#include "wx/button.h"
#include "wx/toplevel.h"
#include "wx/tglbtn.h"
#include "wx/osx/private.h"
#if wxUSE_MARKUP
#include "wx/osx/cocoa/private/markuptoattr.h"
#endif // wxUSE_MARKUP
@implementation wxNSButton
+ (void)initialize
{
static BOOL initialized = NO;
if (!initialized)
{
initialized = YES;
wxOSXCocoaClassAddWXMethods( self );
}
}
- (int) intValue
{
switch ( [self state] )
{
case NSOnState:
return 1;
case NSMixedState:
return 2;
default:
return 0;
}
}
- (void) setIntValue: (int) v
{
switch( v )
{
case 2:
[self setState:NSMixedState];
break;
case 1:
[self setState:NSOnState];
break;
default :
[self setState:NSOffState];
break;
}
}
- (void) setTrackingTag: (NSTrackingRectTag)tag
{
rectTag = tag;
}
- (NSTrackingRectTag) trackingTag
{
return rectTag;
}
@end
@interface NSView(PossibleSizeMethods)
- (NSControlSize)controlSize;
@end
wxButtonCocoaImpl::wxButtonCocoaImpl(wxWindowMac *wxpeer, wxNSButton *v)
: wxWidgetCocoaImpl(wxpeer, v)
{
SetNeedsFrame(false);
}
void wxButtonCocoaImpl::SetBitmap(const wxBitmap& bitmap)
{
// switch bezel style for plain pushbuttons
if ( bitmap.IsOk() )
{
if ([GetNSButton() bezelStyle] == NSRoundedBezelStyle)
[GetNSButton() setBezelStyle:NSRegularSquareBezelStyle];
}
else
{
[GetNSButton() setBezelStyle:NSRoundedBezelStyle];
}
wxWidgetCocoaImpl::SetBitmap(bitmap);
}
#if wxUSE_MARKUP
void wxButtonCocoaImpl::SetLabelMarkup(const wxString& markup)
{
wxMarkupToAttrString toAttr(GetWXPeer(), markup);
NSMutableAttributedString *attrString = toAttr.GetNSAttributedString();
// Button text is always centered.
NSMutableParagraphStyle *
paragraphStyle = [[NSMutableParagraphStyle alloc] init];
[paragraphStyle setAlignment: NSCenterTextAlignment];
[attrString addAttribute:NSParagraphStyleAttributeName
value:paragraphStyle
range:NSMakeRange(0, [attrString length])];
[paragraphStyle release];
[GetNSButton() setAttributedTitle:attrString];
}
#endif // wxUSE_MARKUP
void wxButtonCocoaImpl::SetPressedBitmap( const wxBitmap& bitmap )
{
NSButton* button = GetNSButton();
[button setAlternateImage: bitmap.GetNSImage()];
#if wxUSE_TOGGLEBTN
if ( GetWXPeer()->IsKindOf(wxCLASSINFO(wxToggleButton)) )
{
[button setButtonType:NSToggleButton];
}
else
#endif
{
[button setButtonType:NSMomentaryChangeButton];
}
}
void wxButtonCocoaImpl::GetLayoutInset(int &left , int &top , int &right, int &bottom) const
{
left = top = right = bottom = 0;
NSControlSize size = NSRegularControlSize;
if ( [m_osxView respondsToSelector:@selector(controlSize)] )
size = [m_osxView controlSize];
else if ([m_osxView respondsToSelector:@selector(cell)])
{
id cell = [(id)m_osxView cell];
if ([cell respondsToSelector:@selector(controlSize)])
size = [cell controlSize];
}
if ( [GetNSButton() bezelStyle] == NSRoundedBezelStyle )
{
switch( size )
{
case NSRegularControlSize:
left = right = 6;
top = 4;
bottom = 7;
break;
case NSSmallControlSize:
left = right = 5;
top = 4;
bottom = 6;
break;
case NSMiniControlSize:
left = right = 1;
top = 0;
bottom = 1;
break;
}
}
}
void wxButtonCocoaImpl::SetAcceleratorFromLabel(const wxString& label)
{
const int accelPos = wxControl::FindAccelIndex(label);
if ( accelPos != wxNOT_FOUND )
{
wxString accelstring(label[accelPos + 1]); // Skip '&' itself
accelstring.MakeLower();
wxCFStringRef cfText(accelstring);
[GetNSButton() setKeyEquivalent:cfText.AsNSString()];
[GetNSButton() setKeyEquivalentModifierMask:NSCommandKeyMask];
}
else
{
[GetNSButton() setKeyEquivalent:@""];
}
}
NSButton *wxButtonCocoaImpl::GetNSButton() const
{
wxASSERT( [m_osxView isKindOfClass:[NSButton class]] );
return static_cast<NSButton *>(m_osxView);
}
// Set bezel style depending on the wxBORDER_XXX flags specified by the style
// and also accounting for the label (bezels are different for multiline
// buttons and normal ones) and the ID (special bezel is used for help button).
//
// This is extern because it's also used in src/osx/cocoa/tglbtn.mm.
extern "C"
void
SetBezelStyleFromBorderFlags(NSButton *v,
long style,
wxWindowID winid,
const wxString& label = wxString(),
const wxBitmap& bitmap = wxBitmap())
{
// We can't display a custom label inside a button with help bezel style so
// we only use it if we are using the default label. wxButton itself checks
// if the label is just "Help" in which case it discards it and passes us
// an empty string.
if ( winid == wxID_HELP && label.empty() )
{
[v setBezelStyle:NSHelpButtonBezelStyle];
}
else
{
// We can't use rounded bezel styles neither for multiline buttons nor
// for buttons containing (big) icons as they are only meant to be used
// at certain sizes, so the style used depends on whether the label is
// single or multi line.
const bool
isSimpleText = (label.find_first_of("\n\r") == wxString::npos)
&& (!bitmap.IsOk() || bitmap.GetHeight() < 20);
NSBezelStyle bezel;
switch ( style & wxBORDER_MASK )
{
case wxBORDER_NONE:
bezel = NSShadowlessSquareBezelStyle;
[v setBordered:NO];
break;
case wxBORDER_SIMPLE:
bezel = NSShadowlessSquareBezelStyle;
break;
case wxBORDER_SUNKEN:
bezel = isSimpleText ? NSTexturedRoundedBezelStyle
: NSSmallSquareBezelStyle;
break;
default:
wxFAIL_MSG( "Unknown border style" );
// fall through
case 0:
case wxBORDER_STATIC:
case wxBORDER_RAISED:
case wxBORDER_THEME:
bezel = isSimpleText ? NSRoundedBezelStyle
: NSRegularSquareBezelStyle;
break;
}
[v setBezelStyle:bezel];
}
}
// Set the keyboard accelerator key from the label (e.g. "Click &Me")
void wxButton::OSXUpdateAfterLabelChange(const wxString& label)
{
wxButtonCocoaImpl *impl = static_cast<wxButtonCocoaImpl*>(GetPeer());
// Update the bezel style as may be necessary if our new label is multi
// line while the old one wasn't (or vice versa).
SetBezelStyleFromBorderFlags(impl->GetNSButton(),
GetWindowStyle(),
GetId(),
label);
// Skip setting the accelerator for the default buttons as this would
// overwrite the default "Enter" which should be preserved.
wxTopLevelWindow * const
tlw = wxDynamicCast(wxGetTopLevelParent(this), wxTopLevelWindow);
if ( tlw )
{
if ( tlw->GetDefaultItem() == this )
return;
}
impl->SetAcceleratorFromLabel(label);
}
wxWidgetImplType* wxWidgetImpl::CreateButton( wxWindowMac* wxpeer,
wxWindowMac* WXUNUSED(parent),
wxWindowID winid,
const wxString& label,
const wxPoint& pos,
const wxSize& size,
long style,
long WXUNUSED(extraStyle))
{
NSRect r = wxOSXGetFrameForControl( wxpeer, pos , size ) ;
wxNSButton* v = [[wxNSButton alloc] initWithFrame:r];
SetBezelStyleFromBorderFlags(v, style, winid, label);
[v setButtonType:NSMomentaryPushInButton];
wxButtonCocoaImpl* const impl = new wxButtonCocoaImpl( wxpeer, v );
impl->SetAcceleratorFromLabel(label);
return impl;
}
void wxWidgetCocoaImpl::SetDefaultButton( bool isDefault )
{
if ( [m_osxView isKindOfClass:[NSButton class]] )
{
if ( isDefault )
{
[(NSButton*)m_osxView setKeyEquivalent: @"\r" ];
[(NSButton*)m_osxView setKeyEquivalentModifierMask: 0];
}
else
[(NSButton*)m_osxView setKeyEquivalent: @"" ];
}
}
void wxWidgetCocoaImpl::PerformClick()
{
if ([m_osxView isKindOfClass:[NSControl class]])
[(NSControl*)m_osxView performClick:nil];
}
#if wxUSE_BMPBUTTON
wxWidgetImplType* wxWidgetImpl::CreateBitmapButton( wxWindowMac* wxpeer,
wxWindowMac* WXUNUSED(parent),
wxWindowID winid,
const wxBitmap& bitmap,
const wxPoint& pos,
const wxSize& size,
long style,
long WXUNUSED(extraStyle))
{
NSRect r = wxOSXGetFrameForControl( wxpeer, pos , size ) ;
wxNSButton* v = [[wxNSButton alloc] initWithFrame:r];
SetBezelStyleFromBorderFlags(v, style, winid, wxString(), bitmap);
if (bitmap.IsOk())
[v setImage:bitmap.GetNSImage() ];
[v setButtonType:NSMomentaryPushInButton];
wxWidgetCocoaImpl* c = new wxButtonCocoaImpl( wxpeer, v );
return c;
}
#endif // wxUSE_BMPBUTTON
//
// wxDisclosureButton implementation
//
@interface wxDisclosureNSButton : NSButton
{
BOOL isOpen;
}
- (void) updateImage;
- (void) toggle;
+ (NSImage *)rotateImage: (NSImage *)image;
@end
static const char * disc_triangle_xpm[] = {
"10 9 4 1",
" c None",
". c #737373",
"+ c #989898",
"- c #c6c6c6",
" .- ",
" ..+- ",
" ....+ ",
" ......- ",
" .......- ",
" ......- ",
" ....+ ",
" ..+- ",
" .- ",
};
@implementation wxDisclosureNSButton
+ (void)initialize
{
static BOOL initialized = NO;
if (!initialized)
{
initialized = YES;
wxOSXCocoaClassAddWXMethods( self );
}
}
- (id) initWithFrame:(NSRect) frame
{
self = [super initWithFrame:frame];
isOpen = NO;
[self setImagePosition:NSImageLeft];
[self updateImage];
return self;
}
- (int) intValue
{
return isOpen ? 1 : 0;
}
- (void) setIntValue: (int) v
{
isOpen = ( v != 0 );
[self updateImage];
}
- (void) toggle
{
isOpen = !isOpen;
[self updateImage];
}
wxCFRef<NSImage*> downArray ;
- (void) updateImage
{
static wxBitmap trianglebm(disc_triangle_xpm);
if ( downArray.get() == NULL )
{
downArray.reset( [[wxDisclosureNSButton rotateImage:trianglebm.GetNSImage()] retain] );
}
if ( isOpen )
[self setImage:(NSImage*)downArray.get()];
else
[self setImage:trianglebm.GetNSImage()];
}
+ (NSImage *)rotateImage: (NSImage *)image
{
NSSize imageSize = [image size];
NSSize newImageSize = NSMakeSize(imageSize.height, imageSize.width);
NSImage* newImage = [[NSImage alloc] initWithSize: newImageSize];
[newImage lockFocus];
NSAffineTransform* tm = [NSAffineTransform transform];
[tm translateXBy:newImageSize.width/2 yBy:newImageSize.height/2];
[tm rotateByDegrees:-90];
[tm translateXBy:-newImageSize.width/2 yBy:-newImageSize.height/2];
[tm concat];
[image drawInRect:NSMakeRect(0,0,newImageSize.width, newImageSize.height)
fromRect:NSZeroRect operation:NSCompositeCopy fraction:1.0];
[newImage unlockFocus];
return [newImage autorelease];
}
@end
class wxDisclosureTriangleCocoaImpl : public wxWidgetCocoaImpl
{
public :
wxDisclosureTriangleCocoaImpl(wxWindowMac* peer , WXWidget w) :
wxWidgetCocoaImpl(peer, w)
{
}
~wxDisclosureTriangleCocoaImpl()
{
}
virtual void controlAction(WXWidget slf, void* _cmd, void *sender)
{
wxDisclosureNSButton* db = (wxDisclosureNSButton*)m_osxView;
[db toggle];
wxWidgetCocoaImpl::controlAction(slf, _cmd, sender );
}
};
wxWidgetImplType* wxWidgetImpl::CreateDisclosureTriangle( wxWindowMac* wxpeer,
wxWindowMac* WXUNUSED(parent),
wxWindowID winid,
const wxString& label,
const wxPoint& pos,
const wxSize& size,
long style,
long WXUNUSED(extraStyle))
{
NSRect r = wxOSXGetFrameForControl( wxpeer, pos , size ) ;
wxDisclosureNSButton* v = [[wxDisclosureNSButton alloc] initWithFrame:r];
if ( !label.empty() )
[v setTitle:wxCFStringRef(label).AsNSString()];
SetBezelStyleFromBorderFlags(v, style, winid, label);
return new wxDisclosureTriangleCocoaImpl( wxpeer, v );
}