1 ///////////////////////////////////////////////////////////////////////////// 
   2 // Name:        msw/radiobox.cpp 
   3 // Purpose:     wxRadioBox implementation 
   4 // Author:      Julian Smart 
   8 // Copyright:   (c) Julian Smart and Markus Holzem 
   9 // Licence:     wxWindows license 
  10 ///////////////////////////////////////////////////////////////////////////// 
  12 // =========================================================================== 
  14 // =========================================================================== 
  16 // --------------------------------------------------------------------------- 
  18 // --------------------------------------------------------------------------- 
  21     #pragma implementation "radiobox.h" 
  24 // For compilers that support precompilation, includes "wx.h". 
  25 #include "wx/wxprec.h" 
  34     #include "wx/bitmap.h" 
  36     #include "wx/radiobox.h" 
  37     #include "wx/settings.h" 
  41 #include "wx/msw/private.h" 
  44     #if !defined(__GNUWIN32_OLD__) || defined(__CYGWIN10__) 
  47     #include "wx/tooltip.h" 
  48 #endif // wxUSE_TOOLTIPS 
  50 IMPLEMENT_DYNAMIC_CLASS(wxRadioBox
, wxControl
) 
  52 // there are two possible ways to create the radio buttons: either as children 
  53 // of the radiobox or as siblings of it - allow playing with both variants for 
  54 // now, eventually we will choose the best one for our purposes 
  56 // two main problems are the keyboard navigation inside the radiobox (arrows 
  57 // should switch between buttons, not pass focus to the next control) and the 
  58 // tooltips - a tooltip is associated with the radiobox itself, not the 
  61 // the problems with setting this to 1: 
  62 // a) Alt-<mnemonic of radiobox> isn't handled properly by IsDialogMessage() 
  63 //    because it sets focus to the next control accepting it which is not a 
  64 //    radio button but a radiobox sibling in this case - the only solution to 
  65 //    this would be to handle Alt-<mnemonic> ourselves 
  66 // b) the problems with setting radiobox colours under Win98/2K were reported 
  67 //    but I couldn't reproduce it so I have no idea about what causes it 
  69 // the problems with setting this to 0: 
  70 // a) the tooltips are not shown for the radiobox - possible solution: make 
  71 //    TTM_WINDOWFROMPOS handling code in msw/tooltip.cpp work (easier said than 
  72 //    done because I don't know why it doesn't work) 
  73 #define RADIOBTN_PARENT_IS_RADIOBOX 0 
  75 // --------------------------------------------------------------------------- 
  77 // --------------------------------------------------------------------------- 
  79 // wnd proc for radio buttons 
  81 LRESULT APIENTRY _EXPORT 
wxRadioBtnWndProc(HWND hWnd
, 
  86 // --------------------------------------------------------------------------- 
  88 // --------------------------------------------------------------------------- 
  90 // the pointer to standard radio button wnd proc 
  91 static WXFARPROC s_wndprocRadioBtn 
= (WXFARPROC
)NULL
; 
  95 // =========================================================================== 
  97 // =========================================================================== 
  99 // --------------------------------------------------------------------------- 
 101 // --------------------------------------------------------------------------- 
 103 int wxRadioBox::GetCount() const 
 108 int wxRadioBox::GetColumnCount() const 
 113 int wxRadioBox::GetRowCount() const 
 118 // returns the number of rows 
 119 int wxRadioBox::GetNumVer() const 
 121     if ( m_windowStyle 
& wxRA_SPECIFY_ROWS 
) 
 127         return (m_noItems 
+ m_majorDim 
- 1)/m_majorDim
; 
 131 // returns the number of columns 
 132 int wxRadioBox::GetNumHor() const 
 134     if ( m_windowStyle 
& wxRA_SPECIFY_ROWS 
) 
 136         return (m_noItems 
+ m_majorDim 
- 1)/m_majorDim
; 
 144 bool wxRadioBox::MSWCommand(WXUINT cmd
, WXWORD id
) 
 146     if ( cmd 
== BN_CLICKED 
) 
 151         int selectedButton 
= -1; 
 153         for ( int i 
= 0; i 
< m_noItems
; i
++ ) 
 155             if ( id 
== wxGetWindowId(m_radioButtons
[i
]) ) 
 163         if ( selectedButton 
== -1 ) 
 165             // just ignore it - due to a hack with WM_NCHITTEST handling in our 
 166             // wnd proc, we can receive dummy click messages when we click near 
 167             // the radiobox edge (this is ugly but Julian wouldn't let me get 
 172         if ( selectedButton 
!= m_selectedButton 
) 
 174             m_selectedButton 
= selectedButton
; 
 176             SendNotificationEvent(); 
 178         //else: don't generate events when the selection doesn't change 
 186 #if WXWIN_COMPATIBILITY 
 187 wxRadioBox::wxRadioBox(wxWindow 
*parent
, wxFunction func
, const char *title
, 
 188         int x
, int y
, int width
, int height
, 
 189         int n
, char **choices
, 
 190         int majorDim
, long style
, const char *name
) 
 192     wxString 
*choices2 
= new wxString
[n
]; 
 193     for ( int i 
= 0; i 
< n
; i 
++) choices2
[i
] = choices
[i
]; 
 194     Create(parent
, -1, title
, wxPoint(x
, y
), wxSize(width
, height
), n
, choices2
, majorDim
, style
, 
 195             wxDefaultValidator
, name
); 
 200 #endif // WXWIN_COMPATIBILITY 
 203 wxRadioBox::wxRadioBox() 
 205     m_selectedButton 
= -1; 
 208     m_radioButtons 
= NULL
; 
 211     m_radioHeight 
= NULL
; 
 214 bool wxRadioBox::Create(wxWindow 
*parent
, 
 216                         const wxString
& title
, 
 220                         const wxString choices
[], 
 223                         const wxValidator
& val
, 
 224                         const wxString
& name
) 
 226     // initialize members 
 227     m_selectedButton 
= -1; 
 230     m_majorDim 
= majorDim 
== 0 ? n 
: majorDim
; 
 231     m_noRowsOrCols 
= majorDim
; 
 233     // common initialization 
 234     if ( !CreateControl(parent
, id
, pos
, size
, style
, val
, name
) ) 
 237     // create the static box 
 238     if ( !MSWCreateControl(wxT("BUTTON"), BS_GROUPBOX 
| WS_GROUP
, 
 239                            pos
, size
, title
, 0) ) 
 242     // and now create the buttons 
 244 #if RADIOBTN_PARENT_IS_RADIOBOX 
 245     HWND hwndParent 
= GetHwnd(); 
 247     HWND hwndParent 
= GetHwndOf(parent
); 
 250     // Some radio boxes test consecutive id. 
 251     (void)NewControlId(); 
 252     m_radioButtons 
= new WXHWND
[n
]; 
 253     m_radioWidth 
= new int[n
]; 
 254     m_radioHeight 
= new int[n
]; 
 257     wxFont
& font 
= GetFont(); 
 260         hfont 
= font
.GetResourceHandle(); 
 263     for ( int i 
= 0; i 
< n
; i
++ ) 
 266         m_radioHeight
[i
] = -1; 
 267         long styleBtn 
= BS_AUTORADIOBUTTON 
| WS_TABSTOP 
| WS_CHILD 
| WS_VISIBLE
; 
 268         if ( i 
== 0 && style 
== 0 ) 
 269             styleBtn 
|= WS_GROUP
; 
 271         long newId 
= NewControlId(); 
 273         HWND hwndBtn 
= ::CreateWindow(_T("BUTTON"), 
 276                                       0, 0, 0, 0,   // will be set in SetSize() 
 284             wxLogLastError(wxT("CreateWindow(radio btn)")); 
 289         m_radioButtons
[i
] = (WXHWND
)hwndBtn
; 
 291         SubclassRadioButton((WXHWND
)hwndBtn
); 
 295             ::SendMessage(hwndBtn
, WM_SETFONT
, (WPARAM
)hfont
, 0L); 
 298         m_subControls
.Add(newId
); 
 301     // Create a dummy radio control to end the group. 
 302     (void)::CreateWindow(_T("BUTTON"), 
 304                          WS_GROUP 
| BS_AUTORADIOBUTTON 
| WS_CHILD
, 
 305                          0, 0, 0, 0, hwndParent
, 
 306                          (HMENU
)NewControlId(), wxGetInstance(), NULL
); 
 310     SetSize(pos
.x
, pos
.y
, size
.x
, size
.y
); 
 315 wxRadioBox::~wxRadioBox() 
 317     m_isBeingDeleted 
= TRUE
; 
 322         for (i 
= 0; i 
< m_noItems
; i
++) 
 323             ::DestroyWindow((HWND
)m_radioButtons
[i
]); 
 324         delete[] m_radioButtons
; 
 328         delete[] m_radioWidth
; 
 330         delete[] m_radioHeight
; 
 334 void wxRadioBox::SetString(int item
, const wxString
& label
) 
 336     wxCHECK_RET( item 
>= 0 && item 
< m_noItems
, wxT("invalid radiobox index") ); 
 338     m_radioWidth
[item
] = m_radioHeight
[item
] = -1; 
 339     SetWindowText((HWND
)m_radioButtons
[item
], label
.c_str()); 
 342 void wxRadioBox::SetSelection(int N
) 
 344     wxCHECK_RET( (N 
>= 0) && (N 
< m_noItems
), wxT("invalid radiobox index") ); 
 346     // Following necessary for Win32s, because Win32s translate BM_SETCHECK 
 347     if (m_selectedButton 
>= 0 && m_selectedButton 
< m_noItems
) 
 348         ::SendMessage((HWND
) m_radioButtons
[m_selectedButton
], BM_SETCHECK
, 0, 0L); 
 350     ::SendMessage((HWND
)m_radioButtons
[N
], BM_SETCHECK
, 1, 0L); 
 351     ::SetFocus((HWND
)m_radioButtons
[N
]); 
 353     m_selectedButton 
= N
; 
 356 // Get single selection, for single choice list items 
 357 int wxRadioBox::GetSelection() const 
 359     return m_selectedButton
; 
 362 // Find string for position 
 363 wxString 
wxRadioBox::GetString(int item
) const 
 365     wxCHECK_MSG( item 
>= 0 && item 
< m_noItems
, wxEmptyString
, 
 366                  wxT("invalid radiobox index") ); 
 368     return wxGetWindowText(m_radioButtons
[item
]); 
 371 // ---------------------------------------------------------------------------- 
 373 // ---------------------------------------------------------------------------- 
 375 wxSize 
wxRadioBox::GetMaxButtonSize() const 
 377     // calculate the max button size 
 380     for ( int i 
= 0 ; i 
< m_noItems
; i
++ ) 
 383         if ( m_radioWidth
[i
] < 0 ) 
 385             GetTextExtent(wxGetWindowText(m_radioButtons
[i
]), &width
, &height
); 
 387             // adjust the size to take into account the radio box itself 
 388             // FIXME this is totally bogus! 
 395             width 
= m_radioWidth
[i
]; 
 396             height 
= m_radioHeight
[i
]; 
 399         if ( widthMax 
< width 
) 
 401         if ( heightMax 
< height 
) 
 405     return wxSize(widthMax
, heightMax
); 
 408 wxSize 
wxRadioBox::GetTotalButtonSize(const wxSize
& sizeBtn
) const 
 410     // the radiobox should be big enough for its buttons 
 412     wxGetCharSize(m_hWnd
, &cx1
, &cy1
, &GetFont()); 
 414     int extraHeight 
= cy1
; 
 416     /* We'll assume the adjustments below are OK for Win 3.1 too 
 417 #if defined(CTL3D) && !CTL3D 
 418     // Requires a bigger group box in plain Windows 
 424     int height 
= GetNumVer() * sizeBtn
.y 
+ cy1
/2 + extraHeight
; 
 425     int width  
= GetNumHor() * (sizeBtn
.x 
+ cx1
) + cx1
; 
 427     // Add extra space under the label, if it exists. 
 428     if (!wxControl::GetLabel().IsEmpty()) 
 431     // and also wide enough for its label 
 433     GetTextExtent(GetTitle(), &widthLabel
, NULL
); 
 434     widthLabel 
+= RADIO_SIZE
; // FIXME this is bogus too 
 435     if ( widthLabel 
> width 
) 
 438     return wxSize(width
, height
); 
 441 wxSize 
wxRadioBox::DoGetBestSize() const 
 443     return GetTotalButtonSize(GetMaxButtonSize()); 
 446 // Restored old code. 
 447 void wxRadioBox::DoSetSize(int x
, int y
, int width
, int height
, int sizeFlags
) 
 449     int currentX
, currentY
; 
 450     GetPosition(¤tX
, ¤tY
); 
 451     int widthOld
, heightOld
; 
 452     GetSize(&widthOld
, &heightOld
); 
 457     if (x 
== -1 && !(sizeFlags 
& wxSIZE_ALLOW_MINUS_ONE
)) 
 459     if (y 
== -1 && !(sizeFlags 
& wxSIZE_ALLOW_MINUS_ONE
)) 
 462 #if RADIOBTN_PARENT_IS_RADIOBOX 
 471     wxGetCharSize(m_hWnd
, &cx1
, &cy1
, & GetFont()); 
 473     // Attempt to have a look coherent with other platforms: We compute the 
 474     // biggest toggle dim, then we align all items according this value. 
 475     wxSize maxSize 
= GetMaxButtonSize(); 
 476     int maxWidth 
= maxSize
.x
, 
 477         maxHeight 
= maxSize
.y
; 
 479     wxSize totSize 
= GetTotalButtonSize(maxSize
); 
 480     int totWidth 
= totSize
.x
, 
 481         totHeight 
= totSize
.y
; 
 483     // only change our width/height if asked for 
 486         if ( sizeFlags 
& wxSIZE_AUTO_WIDTH 
) 
 494         if ( sizeFlags 
& wxSIZE_AUTO_HEIGHT 
) 
 500     ::MoveWindow(GetHwnd(), xx
, yy
, width
, height
, TRUE
); 
 502     // Now position all the buttons: the current button will be put at 
 503     // wxPoint(x_offset, y_offset) and the new row/column will start at 
 504     // startX/startY. The size of all buttons will be the same wxSize(maxWidth, 
 505     // maxHeight) except for the buttons in the last column which should extend 
 506     // to the right border of radiobox and thus can be wider than this. 
 508     // Also, remember that wxRA_SPECIFY_COLS means that we arrange buttons in 
 509     // left to right order and m_majorDim is the number of columns while 
 510     // wxRA_SPECIFY_ROWS means that the buttons are arranged top to bottom and 
 511     // m_majorDim is the number of rows. 
 516     // Add extra space under the label, if it exists. 
 517     if (!wxControl::GetLabel().IsEmpty()) 
 520     int startX 
= x_offset
; 
 521     int startY 
= y_offset
; 
 523     for ( int i 
= 0; i 
< m_noItems
; i
++ ) 
 525         // the last button in the row may be wider than the other ones as the 
 526         // radiobox may be wider than the sum of the button widths (as it 
 527         // happens, for example, when the radiobox label is very long) 
 529         if ( m_windowStyle 
& wxRA_SPECIFY_COLS 
) 
 531             // item is the last in its row if it is a multiple of the number of 
 532             // columns or if it is just the last item 
 534             isLastInTheRow 
= ((n 
% m_majorDim
) == 0) || (n 
== m_noItems
); 
 536         else // wxRA_SPECIFY_ROWS 
 538             // item is the last in the row if it is in the last columns 
 539             isLastInTheRow 
= i 
>= (m_noItems
/m_majorDim
)*m_majorDim
; 
 542         // is this the start of new row/column? 
 543         if ( i 
&& (i 
% m_majorDim 
== 0) ) 
 545             if ( m_windowStyle 
& wxRA_SPECIFY_ROWS 
) 
 547                 // start of new column 
 549                 x_offset 
+= maxWidth 
+ cx1
; 
 551             else // start of new row 
 554                 y_offset 
+= maxHeight
; 
 555                 if (m_radioWidth
[0]>0) 
 561         if ( isLastInTheRow 
) 
 563             // make the button go to the end of radio box 
 564             widthBtn 
= startX 
+ width 
- x_offset 
- 2*cx1
; 
 565             if ( widthBtn 
< maxWidth 
) 
 570             // normal button, always of the same size 
 574         // VZ: make all buttons of the same, maximal size - like this they 
 575         //     cover the radiobox entirely and the radiobox tooltips are always 
 576         //     shown (otherwise they are not when the mouse pointer is in the 
 577         //     radiobox part not belonging to any radiobutton) 
 578         ::MoveWindow((HWND
)m_radioButtons
[i
], 
 579                      x_offset
, y_offset
, widthBtn
, maxHeight
, 
 582         // where do we put the next button? 
 583         if ( m_windowStyle 
& wxRA_SPECIFY_ROWS 
) 
 586             y_offset 
+= maxHeight
; 
 587             if (m_radioWidth
[0]>0) 
 592             // to the right of this one 
 593             x_offset 
+= widthBtn 
+ cx1
; 
 598 void wxRadioBox::GetSize(int *width
, int *height
) const 
 601     rect
.left 
= -1; rect
.right 
= -1; rect
.top 
= -1; rect
.bottom 
= -1; 
 604         wxFindMaxSize(m_hWnd
, &rect
); 
 607     for (i 
= 0; i 
< m_noItems
; i
++) 
 608         wxFindMaxSize(m_radioButtons
[i
], &rect
); 
 610     *width 
= rect
.right 
- rect
.left
; 
 611     *height 
= rect
.bottom 
- rect
.top
; 
 614 void wxRadioBox::GetPosition(int *x
, int *y
) const 
 616     wxWindow 
*parent 
= GetParent(); 
 617     RECT rect 
= { -1, -1, -1, -1 }; 
 620     for (i 
= 0; i 
< m_noItems
; i
++) 
 621         wxFindMaxSize(m_radioButtons
[i
], &rect
); 
 624         wxFindMaxSize(m_hWnd
, &rect
); 
 626     // Since we now have the absolute screen coords, if there's a parent we 
 627     // must subtract its top left corner 
 633         ::ScreenToClient((HWND
) parent
->GetHWND(), &point
); 
 636     // We may be faking the client origin. So a window that's really at (0, 30) 
 637     // may appear (to wxWin apps) to be at (0, 0). 
 640         wxPoint 
pt(GetParent()->GetClientAreaOrigin()); 
 649 void wxRadioBox::SetFocus() 
 653         if (m_selectedButton 
== -1) 
 654             ::SetFocus((HWND
) m_radioButtons
[0]); 
 656             ::SetFocus((HWND
) m_radioButtons
[m_selectedButton
]); 
 661 bool wxRadioBox::Show(bool show
) 
 663     if ( !wxControl::Show(show
) ) 
 666     int nCmdShow 
= show 
? SW_SHOW 
: SW_HIDE
; 
 667     for ( int i 
= 0; i 
< m_noItems
; i
++ ) 
 669         ::ShowWindow((HWND
)m_radioButtons
[i
], nCmdShow
); 
 675 // Enable a specific button 
 676 void wxRadioBox::Enable(int item
, bool enable
) 
 678     wxCHECK_RET( item 
>= 0 && item 
< m_noItems
, 
 679                  wxT("invalid item in wxRadioBox::Enable()") ); 
 681     ::EnableWindow((HWND
) m_radioButtons
[item
], enable
); 
 684 // Enable all controls 
 685 bool wxRadioBox::Enable(bool enable
) 
 687     if ( !wxControl::Enable(enable
) ) 
 690     for (int i 
= 0; i 
< m_noItems
; i
++) 
 691         ::EnableWindow((HWND
) m_radioButtons
[i
], enable
); 
 696 // Show a specific button 
 697 void wxRadioBox::Show(int item
, bool show
) 
 699     wxCHECK_RET( item 
>= 0 && item 
< m_noItems
, 
 700                  wxT("invalid item in wxRadioBox::Show()") ); 
 702     ::ShowWindow((HWND
)m_radioButtons
[item
], show 
? SW_SHOW 
: SW_HIDE
); 
 705 bool wxRadioBox::ContainsHWND(WXHWND hWnd
) const 
 707     size_t count 
= GetCount(); 
 708     for ( size_t i 
= 0; i 
< count
; i
++ ) 
 710         if ( GetRadioButtons()[i
] == hWnd 
) 
 717 void wxRadioBox::Command(wxCommandEvent 
& event
) 
 719     SetSelection (event
.m_commandInt
); 
 720     ProcessCommand (event
); 
 723 // NB: if this code is changed, wxGetWindowForHWND() which relies on having the 
 724 //     radiobox pointer in GWL_USERDATA for radio buttons must be updated too! 
 725 void wxRadioBox::SubclassRadioButton(WXHWND hWndBtn
) 
 727     // No GWL_USERDATA in Win16, so omit this subclassing. 
 729     HWND hwndBtn 
= (HWND
)hWndBtn
; 
 731     if ( !s_wndprocRadioBtn 
) 
 732         s_wndprocRadioBtn 
= (WXFARPROC
)::GetWindowLong(hwndBtn
, GWL_WNDPROC
); 
 734     ::SetWindowLong(hwndBtn
, GWL_WNDPROC
, (long)wxRadioBtnWndProc
); 
 735     ::SetWindowLong(hwndBtn
, GWL_USERDATA
, (long)this); 
 739 void wxRadioBox::SendNotificationEvent() 
 741     wxCommandEvent 
event(wxEVT_COMMAND_RADIOBOX_SELECTED
, m_windowId
); 
 742     event
.SetInt( m_selectedButton 
); 
 743     event
.SetString( GetString(m_selectedButton
) ); 
 744     event
.SetEventObject( this ); 
 745     ProcessCommand(event
); 
 748 bool wxRadioBox::SetFont(const wxFont
& font
) 
 750     if ( !wxControl::SetFont(font
) ) 
 756     // also set the font of our radio buttons 
 757     WXHFONT hfont 
= wxFont(font
).GetResourceHandle(); 
 758     for ( int n 
= 0; n 
< m_noItems
; n
++ ) 
 760         HWND hwndBtn 
= (HWND
)m_radioButtons
[n
]; 
 761         ::SendMessage(hwndBtn
, WM_SETFONT
, (WPARAM
)hfont
, 0L); 
 763         // otherwise the buttons are not redrawn correctly 
 764         ::InvalidateRect(hwndBtn
, NULL
, FALSE 
/* don't erase bg */); 
 770 // ---------------------------------------------------------------------------- 
 772 // ---------------------------------------------------------------------------- 
 774 long wxRadioBox::MSWWindowProc(WXUINT nMsg
, WXWPARAM wParam
, WXLPARAM lParam
) 
 779         case WM_CTLCOLORSTATIC
: 
 780             // set the colour of the radio buttons to be the same as ours 
 782                 HDC hdc 
= (HDC
)wParam
; 
 784                 const wxColour
& colBack 
= GetBackgroundColour(); 
 785                 ::SetBkColor(hdc
, wxColourToRGB(colBack
)); 
 786                 ::SetTextColor(hdc
, wxColourToRGB(GetForegroundColour())); 
 788                 wxBrush 
*brush 
= wxTheBrushList
->FindOrCreateBrush(colBack
, wxSOLID
); 
 790                 return (WXHBRUSH
)brush
->GetResourceHandle(); 
 794         // VZ: this code breaks radiobox redrawing under Windows XP, don't use 
 795         //     it. If you need to get messages from the static controls, 
 796         //     create them as a child of another (non static) window 
 798         // This is required for the radiobox to be sensitive to mouse input, 
 799         // e.g. for Dialog Editor. 
 802                 int xPos 
= LOWORD(lParam
);  // horizontal position of cursor 
 803                 int yPos 
= HIWORD(lParam
);  // vertical position of cursor 
 805                 ScreenToClient(&xPos
, &yPos
); 
 807                 // Make sure you can drag by the top of the groupbox, but let 
 808                 // other (enclosed) controls get mouse events also 
 810                     return (long)HTCLIENT
; 
 816     return wxControl::MSWWindowProc(nMsg
, wParam
, lParam
); 
 819 WXHBRUSH 
wxRadioBox::OnCtlColor(WXHDC pDC
, WXHWND 
WXUNUSED(pWnd
), WXUINT 
WXUNUSED(nCtlColor
), 
 825                                WXUINT 
WXUNUSED(message
), 
 826                                WXWPARAM 
WXUNUSED(wParam
), 
 827                                WXLPARAM 
WXUNUSED(lParam
) 
 834         HBRUSH hbrush 
= Ctl3dCtlColorEx(message
, wParam
, lParam
); 
 835         return (WXHBRUSH
) hbrush
; 
 837 #endif // wxUSE_CTL3D 
 840     if (GetParent()->GetTransparentBackground()) 
 841         SetBkMode(hdc
, TRANSPARENT
); 
 843         SetBkMode(hdc
, OPAQUE
); 
 845     wxColour colBack 
= GetBackgroundColour(); 
 848         colBack 
= wxSystemSettings::GetColour(wxSYS_COLOUR_3DFACE
); 
 850     ::SetBkColor(hdc
, wxColourToRGB(colBack
)); 
 851     ::SetTextColor(hdc
, wxColourToRGB(GetForegroundColour())); 
 853     wxBrush 
*brush 
= wxTheBrushList
->FindOrCreateBrush(colBack
, wxSOLID
); 
 855     return (WXHBRUSH
)brush
->GetResourceHandle(); 
 859 // --------------------------------------------------------------------------- 
 860 // window proc for radio buttons 
 861 // --------------------------------------------------------------------------- 
 865 LRESULT APIENTRY _EXPORT 
wxRadioBtnWndProc(HWND hwnd
, 
 873             // we must tell IsDialogMessage()/our kbd processing code that we 
 874             // want to process arrows ourselves because neither of them is 
 875             // smart enough to handle arrows properly for us 
 877                 long lDlgCode 
= ::CallWindowProc(CASTWNDPROC s_wndprocRadioBtn
, hwnd
, 
 878                                                  message
, wParam
, lParam
); 
 880                 return lDlgCode 
| DLGC_WANTARROWS
; 
 886                 NMHDR
* hdr 
= (NMHDR 
*)lParam
; 
 887                 if ( (int)hdr
->code 
== TTN_NEEDTEXT 
) 
 889                     wxRadioBox 
*radiobox 
= (wxRadioBox 
*) 
 890                         ::GetWindowLong(hwnd
, GWL_USERDATA
); 
 892                     wxCHECK_MSG( radiobox
, 0, 
 893                                  wxT("radio button without radio box?") ); 
 895                     wxToolTip 
*tooltip 
= radiobox
->GetToolTip(); 
 898                         TOOLTIPTEXT 
*ttt 
= (TOOLTIPTEXT 
*)lParam
; 
 899                         ttt
->lpszText 
= (wxChar 
*)tooltip
->GetTip().c_str(); 
 907 #endif // wxUSE_TOOLTIPS 
 911                 wxRadioBox 
*radiobox 
= (wxRadioBox 
*) 
 912                     ::GetWindowLong(hwnd
, GWL_USERDATA
); 
 914                 wxCHECK_MSG( radiobox
, 0, wxT("radio button without radio box?") ); 
 916                 bool processed 
= TRUE
; 
 940                         // just to suppress the compiler warning 
 946                     int selOld 
= radiobox
->GetSelection(); 
 947                     int selNew 
= radiobox
->GetNextItem
 
 951                                   radiobox
->GetWindowStyle() 
 954                     if ( selNew 
!= selOld 
) 
 956                         radiobox
->SetSelection(selNew
); 
 958                         // emulate the button click 
 959                         radiobox
->SendNotificationEvent(); 
 970                 wxRadioBox 
*radiobox 
= (wxRadioBox 
*) 
 971                         ::GetWindowLong(hwnd
, GWL_USERDATA
); 
 973                 wxCHECK_MSG( radiobox
, 0, wxT("radio button without radio box?") ); 
 975                 bool processed 
= TRUE
; 
 977                 HELPINFO
* info 
= (HELPINFO
*) lParam
; 
 978                 // Don't yet process menu help events, just windows 
 979                 if (info
->iContextType 
== HELPINFO_WINDOW
) 
 981                     wxWindow
* subjectOfHelp 
= radiobox
; 
 982                     bool eventProcessed 
= FALSE
; 
 983                     while (subjectOfHelp 
&& !eventProcessed
) 
 985                         wxHelpEvent 
helpEvent(wxEVT_HELP
, subjectOfHelp
->GetId(), wxPoint(info
->MousePos
.x
, info
->MousePos
.y
) ) ; // info->iCtrlId); 
 986                         helpEvent
.SetEventObject(radiobox
); 
 987                         eventProcessed 
= radiobox
->GetEventHandler()->ProcessEvent(helpEvent
); 
 989                         // Go up the window hierarchy until the event is handled (or not) 
 990                         subjectOfHelp 
= subjectOfHelp
->GetParent(); 
 992                     processed 
= eventProcessed
; 
 994                 else if (info
->iContextType 
== HELPINFO_MENUITEM
) 
 996                     wxHelpEvent 
helpEvent(wxEVT_HELP
, info
->iCtrlId
) ; 
 997                     helpEvent
.SetEventObject(radiobox
); 
 998                     processed 
= radiobox
->GetEventHandler()->ProcessEvent(helpEvent
); 
1000                 else processed 
= FALSE
; 
1010     return ::CallWindowProc(CASTWNDPROC s_wndprocRadioBtn
, hwnd
, message
, wParam
, lParam
); 
1015 #endif // wxUSE_RADIOBOX