don't try to resize the bitmap buttons with wxBORDER_NONE style to one of the standard sizes, this results in more problems than it solves and still doesn't look good; document wxBORDER_NONE semantics under Mac

git-svn-id: https://svn.wxwidgets.org/svn/wx/wxWidgets/trunk@61145 c3d73ce0-8a6f-49c7-b76d-6d57e0e08775
This commit is contained in:
Vadim Zeitlin 2009-06-20 20:05:52 +00:00
parent 8f2a8de6e7
commit 6398a32d3b
2 changed files with 96 additions and 110 deletions

View File

@ -33,7 +33,11 @@
without using this style a button which is only supposed to show a without using this style a button which is only supposed to show a
bitmap but uses a standard id would display a label too. bitmap but uses a standard id would display a label too.
@style{wxBORDER_NONE} @style{wxBORDER_NONE}
Creates a flat button. Windows and GTK+ only. Creates a button without border. This is currently implemented in MSW,
GTK2 and OSX ports but in the latter only applies to buttons with
bitmaps and using bitmap of one of the standard sizes only, namely
128*128, 48*48, 24*24 or 16*16. In all the other cases wxBORDER_NONE is
ignored under OSX.
@endStyleTable @endStyleTable
By default, i.e. if none of the alignment styles are specified, the label By default, i.e. if none of the alignment styles are specified, the label

View File

@ -22,138 +22,119 @@
#include "wx/osx/private.h" #include "wx/osx/private.h"
//--------------------------------------------------------------------------- namespace
// Helper functions
static wxBitmap wxMakeStdSizeBitmap(const wxBitmap& bitmap)
{ {
// in Mac OS X the icon controls (which are used for borderless bitmap
// buttons) can have only one of the few standard sizes and if they
// don't, the OS rescales them automatically resulting in really ugly
// images, so centre the image in a square of standard size instead
// the supported sizes, sorted in decreasng order
static const int stdSizes[] = { 128, 48, 32, 16, 0 };
const int width = bitmap.GetWidth();
const int height = bitmap.GetHeight();
wxBitmap newBmp(bitmap);
int n;
for ( n = 0; n < (int)WXSIZEOF(stdSizes); n++ )
{
const int sizeStd = stdSizes[n];
if ( width > sizeStd || height > sizeStd )
{
// it will become -1 if the bitmap is larger than the biggest
// supported size, this is intentional
n--;
break;
}
}
if ( n != -1 )
{
const int sizeStd = stdSizes[n];
if ( width != sizeStd || height != sizeStd )
{
wxASSERT_MSG( width <= sizeStd && height <= sizeStd,
_T("bitmap shouldn't be cropped") );
wxImage square_image = bitmap.ConvertToImage();
newBmp = square_image.Size
(
wxSize(sizeStd, sizeStd),
wxPoint((sizeStd - width)/2, (sizeStd-height)/2)
);
}
}
//else: let the system rescale the bitmap
return newBmp;
}
//---------------------------------------------------------------------------
// define a derived class to override SetBitmap() and also to provide
// InitButtonContentInfo() helper used by CreateBitmapButton()
class wxMacBitmapButton : public wxMacControl class wxMacBitmapButton : public wxMacControl
{ {
public: public:
wxMacBitmapButton( wxWindowMac* peer ) : wxMacControl(peer) wxMacBitmapButton(wxWindowMac* peer, const wxBitmap& bitmap, int style)
: wxMacControl(peer)
{ {
// decide what kind of contents the button will have: we want to use an
// icon for buttons with wxBORDER_NONE style as bevel buttons always do
// have a border but icons are limited to a few standard sizes only and
// are resized by the system with extremely ugly results if they don't
// fit (in the past we also tried being smart and pasting a bitmap
// instead of a larger square icon to avoid resizing but this resulted
// in buttons having different size than specified by wx API and
// breaking the layouts and still didn't look good so we don't even try
// to do this any more)
m_isIcon = (style & wxBORDER_NONE) &&
bitmap.IsOk() && IsOfStandardSize(bitmap);
} }
void SetBitmap(const wxBitmap& bitmap)
{
wxBitmap bmp;
if ( GetWXPeer()->HasFlag( wxBORDER_NONE ) )
{
bmp = wxMakeStdSizeBitmap(bitmap);
// TODO set bitmap in peer as well
}
else
bmp = bitmap;
virtual void SetBitmap(const wxBitmap& bitmap)
{
// unfortunately we can't avoid the ugly resizing problem mentioned
// above if a bitmap of supported size was used initially but was
// replaced with another one later as the control was already created
// as an icon control (although maybe we ought to recreate it?)
ControlButtonContentInfo info; ControlButtonContentInfo info;
InitButtonContentInfo(info, bitmap);
if ( GetWXPeer()->HasFlag( wxBORDER_NONE ) ) if ( info.contentType == kControlContentIconRef )
{ SetData(kControlIconPart, kControlIconContentTag, info);
wxMacCreateBitmapButton( &info, bmp, kControlContentIconRef ); else if ( info.contentType != kControlNoContent )
if ( info.contentType != kControlNoContent ) SetData(kControlButtonPart, kControlBevelButtonContentTag, info);
SetData( kControlIconPart, kControlIconContentTag, info );
}
else
{
wxMacCreateBitmapButton( &info, bmp );
if ( info.contentType != kControlNoContent )
SetData( kControlButtonPart, kControlBevelButtonContentTag, info );
}
wxMacReleaseBitmapButton( &info ); wxMacReleaseBitmapButton(&info);
} }
void InitButtonContentInfo(ControlButtonContentInfo& info,
const wxBitmap& bitmap)
{
wxMacCreateBitmapButton(&info, bitmap,
m_isIcon ? kControlContentIconRef : 0);
}
private:
// helper function: returns true if the given bitmap is of one of standard
// sizes supported by OS X icons
static bool IsOfStandardSize(const wxBitmap& bmp)
{
const int w = bmp.GetWidth();
return bmp.GetHeight() == w &&
(w == 128 || w == 48 || w == 32 || w == 16);
}
// true if this is an icon control, false if it's a bevel button
bool m_isIcon;
wxDECLARE_NO_COPY_CLASS(wxMacBitmapButton);
}; };
wxWidgetImplType* wxWidgetImpl::CreateBitmapButton( wxWindowMac* wxpeer, } // anonymous namespace
wxWindowMac* parent,
wxWindowID WXUNUSED(id), wxWidgetImplType* wxWidgetImpl::CreateBitmapButton( wxWindowMac* wxpeer,
wxWindowMac* parent,
wxWindowID WXUNUSED(id),
const wxBitmap& bitmap, const wxBitmap& bitmap,
const wxPoint& pos, const wxPoint& pos,
const wxSize& size, const wxSize& size,
long style, long style,
long WXUNUSED(extraStyle)) long WXUNUSED(extraStyle))
{ {
OSStatus err = noErr; wxMacBitmapButton* peer = new wxMacBitmapButton(wxpeer, bitmap, style);
ControlButtonContentInfo info;
OSStatus err;
WXWindow macParent = MAC_WXHWND(parent->MacGetTopLevelWindowRef());
Rect bounds = wxMacGetBoundsForControl( wxpeer, pos, size ); Rect bounds = wxMacGetBoundsForControl( wxpeer, pos, size );
wxMacControl* peer = new wxMacBitmapButton( wxpeer );
wxBitmap bmp;
if ( bitmap.Ok() && (style & wxBORDER_NONE) ) ControlButtonContentInfo info;
peer->InitButtonContentInfo(info, bitmap);
if ( info.contentType == kControlContentIconRef )
{ {
bmp = wxMakeStdSizeBitmap(bitmap); err = CreateIconControl
// TODO set bitmap in peer as well (
macParent,
&bounds,
&info,
false,
peer->GetControlRefAddr()
);
} }
else else // normal bevel button
bmp = bitmap;
if ( style & wxBORDER_NONE )
{ {
// contrary to the docs this control only works with iconrefs err = CreateBevelButtonControl
wxMacCreateBitmapButton( &info, bmp, kControlContentIconRef ); (
err = CreateIconControl( macParent,
MAC_WXHWND(parent->MacGetTopLevelWindowRef()), &bounds,
&bounds, &info, false, peer->GetControlRefAddr() ); CFSTR(""),
} style & wxBU_AUTODRAW ? kControlBevelButtonSmallBevel
else : kControlBevelButtonNormalBevel,
{ kControlBehaviorOffsetContents,
wxMacCreateBitmapButton( &info, bmp ); &info,
err = CreateBevelButtonControl( 0, // menu id (no associated menu)
MAC_WXHWND(parent->MacGetTopLevelWindowRef()), &bounds, CFSTR(""), 0, // menu behaviour (unused)
((style & wxBU_AUTODRAW) ? kControlBevelButtonSmallBevel : kControlBevelButtonNormalBevel ), 0, // menu placement (unused too)
kControlBehaviorOffsetContents, &info, 0, 0, 0, peer->GetControlRefAddr() ); peer->GetControlRefAddr()
);
} }
verify_noerr( err ); verify_noerr( err );
@ -161,4 +142,5 @@ wxWidgetImplType* wxWidgetImpl::CreateBitmapButton( wxWindowMac* wxpeer,
wxMacReleaseBitmapButton( &info ); wxMacReleaseBitmapButton( &info );
return peer; return peer;
} }
#endif
#endif // wxUSE_BMPBUTTON