1 /////////////////////////////////////////////////////////////////////////////
2 // Name: msw/button.cpp
4 // Author: Julian Smart
8 // Copyright: (c) Julian Smart and Markus Holzem
9 // Licence: wxWindows license
10 /////////////////////////////////////////////////////////////////////////////
12 // ============================================================================
14 // ============================================================================
16 // ----------------------------------------------------------------------------
18 // ----------------------------------------------------------------------------
21 #pragma implementation "button.h"
24 // For compilers that support precompilation, includes "wx.h".
25 #include "wx/wxprec.h"
34 #include "wx/button.h"
37 #include "wx/bmpbuttn.h"
38 #include "wx/settings.h"
39 #include "wx/dcscreen.h"
42 #include "wx/msw/private.h"
44 // ----------------------------------------------------------------------------
46 // ----------------------------------------------------------------------------
48 IMPLEMENT_DYNAMIC_CLASS(wxButton
, wxControl
)
50 // this macro tries to adjust the default button height to a reasonable value
51 // using the char height as the base
52 #define BUTTON_HEIGHT_FROM_CHAR_HEIGHT(cy) (11*EDIT_HEIGHT_FROM_CHAR_HEIGHT(cy)/10)
54 // ============================================================================
56 // ============================================================================
58 // ----------------------------------------------------------------------------
59 // creation/destruction
60 // ----------------------------------------------------------------------------
62 bool wxButton::Create(wxWindow
*parent
,
64 const wxString
& label
,
68 const wxValidator
& validator
,
71 if ( !CreateControl(parent
, id
, pos
, size
, style
, validator
, name
) )
75 WXDWORD msStyle
= MSWGetStyle(style
, &exstyle
);
78 // if the label contains several lines we must explicitly tell the button
79 // about it or it wouldn't draw it correctly ("\n"s would just appear as
82 // NB: we do it here and not in MSWGetStyle() because we need the label
83 // value and m_label is not set yet when MSWGetStyle() is called;
84 // besides changing BS_MULTILINE during run-time is pointless anyhow
85 if ( label
.find(_T('\n')) != wxString::npos
)
87 msStyle
|= BS_MULTILINE
;
91 return MSWCreateControl(_T("BUTTON"), msStyle
, pos
, size
, label
, exstyle
);
98 // ----------------------------------------------------------------------------
100 // ----------------------------------------------------------------------------
102 WXDWORD
wxButton::MSWGetStyle(long style
, WXDWORD
*exstyle
) const
104 // buttons never have an external border, they draw their own one
105 WXDWORD msStyle
= wxControl::MSWGetStyle
107 (style
& ~wxBORDER_MASK
) | wxBORDER_NONE
, exstyle
110 // we must use WS_CLIPSIBLINGS with the buttons or they would draw over
111 // each other in any resizeable dialog which has more than one button in
113 msStyle
|= WS_CLIPSIBLINGS
;
116 // don't use "else if" here: weird as it is, but you may combine wxBU_LEFT
117 // and wxBU_RIGHT to get BS_CENTER!
118 if ( style
& wxBU_LEFT
)
120 if ( style
& wxBU_RIGHT
)
122 if ( style
& wxBU_TOP
)
124 if ( style
& wxBU_BOTTOM
)
125 msStyle
|= BS_BOTTOM
;
131 // ----------------------------------------------------------------------------
132 // size management including autosizing
133 // ----------------------------------------------------------------------------
135 wxSize
wxButton::DoGetBestSize() const
137 wxString label
= wxGetWindowText(GetHWND());
139 GetTextExtent(label
, &wBtn
, NULL
);
142 wxGetCharSize(GetHWND(), &wChar
, &hChar
, &GetFont());
144 // add a margin - the button is wider than just its label
147 // the button height is proportional to the height of the font used
148 int hBtn
= BUTTON_HEIGHT_FROM_CHAR_HEIGHT(hChar
);
150 wxSize sz
= GetDefaultSize();
151 if (wBtn
> sz
.x
) sz
.x
= wBtn
;
152 if (hBtn
> sz
.y
) sz
.y
= hBtn
;
158 wxSize
wxButtonBase::GetDefaultSize()
160 static wxSize s_sizeBtn
;
162 if ( s_sizeBtn
.x
== 0 )
165 dc
.SetFont(wxSystemSettings::GetFont(wxSYS_DEFAULT_GUI_FONT
));
167 // the size of a standard button in the dialog units is 50x14,
168 // translate this to pixels
169 // NB1: the multipliers come from the Windows convention
170 // NB2: the extra +1/+2 were needed to get the size be the same as the
171 // size of the buttons in the standard dialog - I don't know how
172 // this happens, but on my system this size is 75x23 in pixels and
173 // 23*8 isn't even divisible by 14... Would be nice to understand
174 // why these constants are needed though!
175 s_sizeBtn
.x
= (50 * (dc
.GetCharWidth() + 1))/4;
176 s_sizeBtn
.y
= ((14 * dc
.GetCharHeight()) + 2)/8;
182 // ----------------------------------------------------------------------------
183 // default button handling
184 // ----------------------------------------------------------------------------
186 // set this button as the (permanently) default one in its panel
187 void wxButton::SetDefault()
189 wxWindow
*parent
= GetParent();
191 wxCHECK_RET( parent
, _T("button without parent?") );
193 // set this one as the default button both for wxWindows and Windows
194 wxWindow
*winOldDefault
= parent
->SetDefaultItem(this);
195 ::SendMessage(GetWinHwnd(parent
), DM_SETDEFID
, m_windowId
, 0L);
197 UpdateDefaultStyle(this, winOldDefault
);
200 void wxButton::SetTmpDefault()
202 wxWindow
*parent
= GetParent();
204 wxCHECK_RET( parent
, _T("button without parent?") );
206 wxWindow
*winOldDefault
= parent
->GetDefaultItem();
207 parent
->SetTmpDefaultItem(this);
208 if ( winOldDefault
!= this )
210 UpdateDefaultStyle(this, winOldDefault
);
212 //else: no styles to update
215 void wxButton::UnsetTmpDefault()
217 wxWindow
*parent
= GetParent();
219 wxCHECK_RET( parent
, _T("button without parent?") );
221 parent
->SetTmpDefaultItem(NULL
);
223 wxWindow
*winOldDefault
= parent
->GetDefaultItem();
224 if ( winOldDefault
!= this )
226 UpdateDefaultStyle(winOldDefault
, this);
228 //else: we had been default before anyhow
233 wxButton::UpdateDefaultStyle(wxWindow
*winDefault
, wxWindow
*winOldDefault
)
235 // clear the BS_DEFPUSHBUTTON for the old default button
236 wxButton
*btnOldDefault
= wxDynamicCast(winOldDefault
, wxButton
);
237 if ( btnOldDefault
&& btnOldDefault
!= winDefault
)
239 // remove the BS_DEFPUSHBUTTON style from the other button
240 long style
= ::GetWindowLong(GetHwndOf(btnOldDefault
), GWL_STYLE
);
242 // don't do it with the owner drawn buttons because it will reset
243 // BS_OWNERDRAW style bit too (BS_OWNERDRAW & BS_DEFPUSHBUTTON != 0)!
244 if ( (style
& BS_OWNERDRAW
) != BS_OWNERDRAW
)
246 style
&= ~BS_DEFPUSHBUTTON
;
247 ::SendMessage(GetHwndOf(btnOldDefault
), BM_SETSTYLE
, style
, 1L);
251 // redraw the button - it will notice itself that it's not the
252 // default one any longer
253 btnOldDefault
->Refresh();
257 // and set BS_DEFPUSHBUTTON for this button
258 wxButton
*btnDefault
= wxDynamicCast(winDefault
, wxButton
);
261 long style
= ::GetWindowLong(GetHwndOf(btnDefault
), GWL_STYLE
);
262 if ( (style
& BS_OWNERDRAW
) != BS_OWNERDRAW
)
264 style
|= BS_DEFPUSHBUTTON
;
265 ::SendMessage(GetHwndOf(btnDefault
), BM_SETSTYLE
, style
, 1L);
269 btnDefault
->Refresh();
274 // ----------------------------------------------------------------------------
276 // ----------------------------------------------------------------------------
278 bool wxButton::SendClickEvent()
280 wxCommandEvent
event(wxEVT_COMMAND_BUTTON_CLICKED
, GetId());
281 event
.SetEventObject(this);
283 return ProcessCommand(event
);
286 void wxButton::Command(wxCommandEvent
& event
)
288 ProcessCommand(event
);
291 // ----------------------------------------------------------------------------
292 // event/message handlers
293 // ----------------------------------------------------------------------------
295 bool wxButton::MSWCommand(WXUINT param
, WXWORD
WXUNUSED(id
))
297 bool processed
= FALSE
;
300 case 1: // message came from an accelerator
301 case BN_CLICKED
: // normal buttons send this
302 case BN_DOUBLECLICKED
: // owner-drawn ones also send this
303 processed
= SendClickEvent();
310 long wxButton::MSWWindowProc(WXUINT nMsg
, WXWPARAM wParam
, WXLPARAM lParam
)
312 // when we receive focus, we want to temporary become the default button in
313 // our parent panel so that pressing "Enter" would activate us -- and when
314 // losing it we should restore the previous default button as well
315 if ( nMsg
== WM_SETFOCUS
)
319 // let the default processing take place too
321 else if ( nMsg
== WM_KILLFOCUS
)
325 else if ( nMsg
== WM_LBUTTONDBLCLK
)
327 // emulate a click event to force an owner-drawn button to change its
328 // appearance - without this, it won't do it
329 (void)wxControl::MSWWindowProc(WM_LBUTTONDOWN
, wParam
, lParam
);
331 // and continue with processing the message normally as well
334 // let the base class do all real processing
335 return wxControl::MSWWindowProc(nMsg
, wParam
, lParam
);
338 // ----------------------------------------------------------------------------
339 // owner-drawn buttons support
340 // ----------------------------------------------------------------------------
346 static void DrawButtonText(HDC hdc
,
348 const wxString
& text
,
351 COLORREF colOld
= SetTextColor(hdc
, col
);
352 int modeOld
= SetBkMode(hdc
, TRANSPARENT
);
354 DrawText(hdc
, text
, text
.length(), pRect
,
355 DT_CENTER
| DT_VCENTER
| DT_SINGLELINE
);
357 SetBkMode(hdc
, modeOld
);
358 SetTextColor(hdc
, colOld
);
361 static void DrawRect(HDC hdc
, const RECT
& r
)
363 MoveToEx(hdc
, r
.left
, r
.top
, NULL
);
364 LineTo(hdc
, r
.right
, r
.top
);
365 LineTo(hdc
, r
.right
, r
.bottom
);
366 LineTo(hdc
, r
.left
, r
.bottom
);
367 LineTo(hdc
, r
.left
, r
.top
);
370 void wxButton::MakeOwnerDrawn()
372 long style
= GetWindowLong(GetHwnd(), GWL_STYLE
);
373 if ( (style
& BS_OWNERDRAW
) != BS_OWNERDRAW
)
376 style
|= BS_OWNERDRAW
;
377 SetWindowLong(GetHwnd(), GWL_STYLE
, style
);
381 bool wxButton::SetBackgroundColour(const wxColour
&colour
)
383 if ( !wxControl::SetBackgroundColour(colour
) )
396 bool wxButton::SetForegroundColour(const wxColour
&colour
)
398 if ( !wxControl::SetForegroundColour(colour
) )
412 The button frame looks like this normally:
415 WHHHHHHHHHHHHHHHHGB W = white (HILIGHT)
416 WH GB H = light grey (LIGHT)
417 WH GB G = dark grey (SHADOW)
418 WH GB B = black (DKSHADOW)
423 When the button is selected, the button becomes like this (the total button
424 size doesn't change):
435 When the button is pushed (while selected) it is like:
447 static void DrawButtonFrame(HDC hdc
, const RECT
& rectBtn
,
448 bool selected
, bool pushed
)
451 CopyRect(&r
, &rectBtn
);
453 HPEN hpenBlack
= CreatePen(PS_SOLID
, 1, GetSysColor(COLOR_3DDKSHADOW
)),
454 hpenGrey
= CreatePen(PS_SOLID
, 1, GetSysColor(COLOR_3DSHADOW
)),
455 hpenLightGr
= CreatePen(PS_SOLID
, 1, GetSysColor(COLOR_3DLIGHT
)),
456 hpenWhite
= CreatePen(PS_SOLID
, 1, GetSysColor(COLOR_3DHILIGHT
));
458 HPEN hpenOld
= (HPEN
)SelectObject(hdc
, hpenBlack
);
467 (void)SelectObject(hdc
, hpenGrey
);
468 InflateRect(&r
, -1, -1);
478 InflateRect(&r
, -1, -1);
481 MoveToEx(hdc
, r
.left
, r
.bottom
, NULL
);
482 LineTo(hdc
, r
.right
, r
.bottom
);
483 LineTo(hdc
, r
.right
, r
.top
- 1);
485 (void)SelectObject(hdc
, hpenWhite
);
486 MoveToEx(hdc
, r
.left
, r
.bottom
- 1, NULL
);
487 LineTo(hdc
, r
.left
, r
.top
);
488 LineTo(hdc
, r
.right
, r
.top
);
490 (void)SelectObject(hdc
, hpenLightGr
);
491 MoveToEx(hdc
, r
.left
+ 1, r
.bottom
- 2, NULL
);
492 LineTo(hdc
, r
.left
+ 1, r
.top
+ 1);
493 LineTo(hdc
, r
.right
- 1, r
.top
+ 1);
495 (void)SelectObject(hdc
, hpenGrey
);
496 MoveToEx(hdc
, r
.left
+ 1, r
.bottom
- 1, NULL
);
497 LineTo(hdc
, r
.right
- 1, r
.bottom
- 1);
498 LineTo(hdc
, r
.right
- 1, r
.top
);
501 (void)SelectObject(hdc
, hpenOld
);
502 DeleteObject(hpenWhite
);
503 DeleteObject(hpenLightGr
);
504 DeleteObject(hpenGrey
);
505 DeleteObject(hpenBlack
);
508 bool wxButton::MSWOnDraw(WXDRAWITEMSTRUCT
*wxdis
)
510 LPDRAWITEMSTRUCT lpDIS
= (LPDRAWITEMSTRUCT
)wxdis
;
513 CopyRect(&rectBtn
, &lpDIS
->rcItem
);
515 COLORREF colBg
= wxColourToRGB(GetBackgroundColour()),
516 colFg
= wxColourToRGB(GetForegroundColour());
518 HDC hdc
= lpDIS
->hDC
;
519 UINT state
= lpDIS
->itemState
;
521 // first, draw the background
522 HBRUSH hbrushBackground
= ::CreateSolidBrush(colBg
);
524 FillRect(hdc
, &rectBtn
, hbrushBackground
);
526 // draw the border for the current state
527 bool selected
= (state
& ODS_SELECTED
) != 0;
530 wxPanel
*panel
= wxDynamicCast(GetParent(), wxPanel
);
533 selected
= panel
->GetDefaultItem() == this;
536 bool pushed
= (SendMessage(GetHwnd(), BM_GETSTATE
, 0, 0) & BST_PUSHED
) != 0;
538 DrawButtonFrame(hdc
, rectBtn
, selected
, pushed
);
540 // draw the focus rect if needed
541 if ( state
& ODS_FOCUS
)
544 CopyRect(&rectFocus
, &rectBtn
);
546 // I don't know where does this constant come from, but this is how
547 // Windows draws them
548 InflateRect(&rectFocus
, -4, -4);
550 DrawFocusRect(hdc
, &rectFocus
);
555 // the label is shifted by 1 pixel to create "pushed" effect
556 OffsetRect(&rectBtn
, 1, 1);
559 DrawButtonText(hdc
, &rectBtn
, GetLabel(),
560 state
& ODS_DISABLED
? GetSysColor(COLOR_GRAYTEXT
)
563 ::DeleteObject(hbrushBackground
);
570 #endif // wxUSE_BUTTON