1 ///////////////////////////////////////////////////////////////////////////// 
   2 // Name:        src/msw/control.cpp 
   3 // Purpose:     wxControl class 
   4 // Author:      Julian Smart 
   8 // Copyright:   (c) Julian Smart 
   9 // Licence:     wxWindows licence 
  10 ///////////////////////////////////////////////////////////////////////////// 
  12 // ============================================================================ 
  14 // ============================================================================ 
  16 // ---------------------------------------------------------------------------- 
  18 // ---------------------------------------------------------------------------- 
  20 // For compilers that support precompilation, includes "wx.h". 
  21 #include "wx/wxprec.h" 
  29 #include "wx/control.h" 
  32     #include "wx/msw/wrapcctl.h" // include <commctrl.h> "properly" 
  35     #include "wx/dcclient.h" 
  37     #include "wx/settings.h" 
  38     #include "wx/ctrlsub.h" 
  42     #include "wx/listctrl.h" 
  43 #endif // wxUSE_LISTCTRL 
  46     #include "wx/treectrl.h" 
  47 #endif // wxUSE_TREECTRL 
  49 #include "wx/msw/private.h" 
  50 #include "wx/msw/uxtheme.h" 
  52 // ---------------------------------------------------------------------------- 
  54 // ---------------------------------------------------------------------------- 
  56 IMPLEMENT_ABSTRACT_CLASS(wxControl
, wxWindow
) 
  58 // ============================================================================ 
  59 // wxControl implementation 
  60 // ============================================================================ 
  62 // ---------------------------------------------------------------------------- 
  63 // control window creation 
  64 // ---------------------------------------------------------------------------- 
  66 bool wxControl::Create(wxWindow 
*parent
, 
  71                        const wxValidator
& wxVALIDATOR_PARAM(validator
), 
  74     if ( !wxWindow::Create(parent
, id
, pos
, size
, style
, name
) ) 
  78     SetValidator(validator
); 
  84 bool wxControl::MSWCreateControl(const wxChar 
*classname
, 
  85                                  const wxString
& label
, 
  90     WXDWORD msStyle 
= MSWGetStyle(GetWindowStyle(), &exstyle
); 
  92     return MSWCreateControl(classname
, msStyle
, pos
, size
, label
, exstyle
); 
  95 bool wxControl::MSWCreateControl(const wxChar 
*classname
, 
  99                                  const wxString
& label
, 
 102     // if no extended style given, determine it ourselves 
 103     if ( exstyle 
== (WXDWORD
)-1 ) 
 106         (void) MSWGetStyle(GetWindowStyle(), &exstyle
); 
 109     // all controls should have this style 
 112     // create the control visible if it's currently shown for wxWidgets 
 118     // choose the position for the control: we have a problem with default size 
 119     // here as we can't calculate the best size before the control exists 
 120     // (DoGetBestSize() may need to use m_hWnd), so just choose the minimal 
 121     // possible but non 0 size because 0 window width/height result in problems 
 123     int x 
= pos
.x 
== wxDefaultCoord 
? 0 : pos
.x
, 
 124         y 
= pos
.y 
== wxDefaultCoord 
? 0 : pos
.y
, 
 125         w 
= size
.x 
== wxDefaultCoord 
? 1 : size
.x
, 
 126         h 
= size
.y 
== wxDefaultCoord 
? 1 : size
.y
; 
 128     // ... and adjust it to account for a possible parent frames toolbar 
 129     AdjustForParentClientOrigin(x
, y
); 
 131     m_hWnd 
= (WXHWND
)::CreateWindowEx
 
 133                         exstyle
,            // extended style 
 134                         classname
,          // the kind of control to create 
 135                         label
.wx_str(),     // the window name 
 136                         style
,              // the window style 
 137                         x
, y
, w
, h
,         // the window position and size 
 138                         GetHwndOf(GetParent()),         // parent 
 139                         (HMENU
)wxUIntToPtr(GetId()),    // child id 
 140                         wxGetInstance(),    // app instance 
 141                         NULL                
// creation parameters 
 146         wxLogLastError(wxString::Format
 
 148                         _T("CreateWindowEx(\"%s\", flags=%08lx, ex=%08lx)"), 
 149                         classname
, style
, exstyle
 
 156     // Text labels starting with the character 0xff (which is a valid character 
 157     // in many code pages) don't appear correctly as CreateWindowEx() has some 
 158     // special treatment for this case, apparently the strings starting with -1 
 159     // are not really strings but something called "ordinals". There is no 
 160     // documentation about it but the fact is that the label gets mangled or 
 161     // not displayed at all if we don't do this, see #9572. 
 163     // Notice that 0xffff is not a valid Unicode character so the problem 
 164     // doesn't arise in Unicode build. 
 165     if ( !label
.empty() && label
[0] == -1 ) 
 166         ::SetWindowText(GetHwnd(), label
.wx_str()); 
 167 #endif // !wxUSE_UNICODE 
 169     // saving the label in m_labelOrig to return it verbatim 
 170     // later in GetLabel() 
 173     // install wxWidgets window proc for this window 
 176     // set up fonts and colours 
 182         wxFont font 
= GetDefaultAttributes().font
; 
 184         // if we set a font for {list,tree}ctrls and the font size is changed in 
 185         // the display properties then the font size for these controls doesn't 
 186         // automatically adjust when they receive WM_SETTINGCHANGE 
 188         // FIXME: replace the dynamic casts with virtual function calls!! 
 189 #if wxUSE_LISTCTRL || wxUSE_TREECTRL 
 190         bool testFont 
= false; 
 192         if ( wxDynamicCastThis(wxListCtrl
) ) 
 194 #endif // wxUSE_LISTCTRL 
 196         if ( wxDynamicCastThis(wxTreeCtrl
) ) 
 198 #endif // wxUSE_TREECTRL 
 202             // not sure if we need to explicitly set the font here for Win95/NT4 
 203             // but we definitely can't do it for any newer version 
 204             // see wxGetCCDefaultFont() in src/msw/settings.cpp for explanation 
 205             // of why this test works 
 207             // TODO: test Win95/NT4 to see if this is needed or breaks the 
 208             // font resizing as it does on newer versions 
 209             if ( font 
!= wxSystemSettings::GetFont(wxSYS_DEFAULT_GUI_FONT
) ) 
 214 #endif // wxUSE_LISTCTRL || wxUSE_TREECTRL 
 218             SetFont(GetDefaultAttributes().font
); 
 222     // set the size now if no initial size specified 
 223     SetInitialSize(size
); 
 228 // ---------------------------------------------------------------------------- 
 230 // ---------------------------------------------------------------------------- 
 232 WXDWORD 
wxControl::MSWGetStyle(long style
, WXDWORD 
*exstyle
) const 
 234     long msStyle 
= wxWindow::MSWGetStyle(style
, exstyle
); 
 236     if ( AcceptsFocusFromKeyboard() ) 
 238         msStyle 
|= WS_TABSTOP
; 
 244 wxSize 
wxControl::DoGetBestSize() const 
 247        return wxControlBase::DoGetBestSize(); 
 249     return wxSize(DEFAULT_ITEM_WIDTH
, DEFAULT_ITEM_HEIGHT
); 
 252 wxBorder 
wxControl::GetDefaultBorder() const 
 254     return wxControlBase::GetDefaultBorder(); 
 257 // This is a helper for all wxControls made with UPDOWN native control. 
 258 // In wxMSW it was only wxSpinCtrl derived from wxSpinButton but in 
 259 // WinCE of Smartphones this happens also for native wxTextCtrl, 
 260 // wxChoice and others. 
 261 wxSize 
wxControl::GetBestSpinnerSize(const bool is_vertical
) const 
 263     // take size according to layout 
 265 #if defined(__SMARTPHONE__) && defined(__WXWINCE__) 
 268                     ::GetSystemMetrics(is_vertical 
? SM_CXVSCROLL 
: SM_CXHSCROLL
), 
 269                     ::GetSystemMetrics(is_vertical 
? SM_CYVSCROLL 
: SM_CYHSCROLL
) 
 273     // correct size as for undocumented MSW variants cases (WinCE and perhaps others) 
 275         bestSize
.x 
= bestSize
.y
; 
 277         bestSize
.y 
= bestSize
.x
; 
 279     // double size according to layout 
 288 /* static */ wxVisualAttributes
 
 289 wxControl::GetClassDefaultAttributes(wxWindowVariant 
WXUNUSED(variant
)) 
 291     wxVisualAttributes attrs
; 
 293     // old school (i.e. not "common") controls use the standard dialog font 
 295     attrs
.font 
= wxSystemSettings::GetFont(wxSYS_DEFAULT_GUI_FONT
); 
 297     // most, or at least many, of the controls use the same colours as the 
 298     // buttons -- others will have to override this (and possibly simply call 
 299     // GetCompositeControlsDefaultAttributes() from their versions) 
 300     attrs
.colFg 
= wxSystemSettings::GetColour(wxSYS_COLOUR_BTNTEXT
); 
 301     attrs
.colBg 
= wxSystemSettings::GetColour(wxSYS_COLOUR_BTNFACE
); 
 306 // another version for the "composite", i.e. non simple controls 
 307 /* static */ wxVisualAttributes
 
 308 wxControl::GetCompositeControlsDefaultAttributes(wxWindowVariant 
WXUNUSED(variant
)) 
 310     wxVisualAttributes attrs
; 
 311     attrs
.font 
= wxSystemSettings::GetFont(wxSYS_DEFAULT_GUI_FONT
); 
 312     attrs
.colFg 
= wxSystemSettings::GetColour(wxSYS_COLOUR_WINDOWTEXT
); 
 313     attrs
.colBg 
= wxSystemSettings::GetColour(wxSYS_COLOUR_WINDOW
); 
 318 // ---------------------------------------------------------------------------- 
 320 // ---------------------------------------------------------------------------- 
 322 bool wxControl::ProcessCommand(wxCommandEvent
& event
) 
 324     return HandleWindowEvent(event
); 
 327 bool wxControl::MSWOnNotify(int idCtrl
, 
 331     wxEventType eventType 
wxDUMMY_INITIALIZE(wxEVT_NULL
); 
 333     NMHDR 
*hdr 
= (NMHDR
*) lParam
; 
 337             eventType 
= wxEVT_COMMAND_LEFT_CLICK
; 
 341             eventType 
= wxEVT_COMMAND_LEFT_DCLICK
; 
 345             eventType 
= wxEVT_COMMAND_RIGHT_CLICK
; 
 349             eventType 
= wxEVT_COMMAND_RIGHT_DCLICK
; 
 353             eventType 
= wxEVT_COMMAND_SET_FOCUS
; 
 357             eventType 
= wxEVT_COMMAND_KILL_FOCUS
; 
 361             eventType 
= wxEVT_COMMAND_ENTER
; 
 365             return wxWindow::MSWOnNotify(idCtrl
, lParam
, result
); 
 368     wxCommandEvent 
event(wxEVT_NULL
, m_windowId
); 
 369     event
.SetEventType(eventType
); 
 370     event
.SetEventObject(this); 
 372     return HandleWindowEvent(event
); 
 375 WXHBRUSH 
wxControl::DoMSWControlColor(WXHDC pDC
, wxColour colBg
, WXHWND hWnd
) 
 380         ::SetTextColor(hdc
, wxColourToRGB(GetForegroundColour())); 
 386         hbr 
= MSWGetBgBrush(pDC
, hWnd
); 
 388         // if the control doesn't have any bg colour, foreground colour will be 
 389         // ignored as the return value would be 0 -- so forcefully give it a 
 390         // non default background brush in this case 
 391         if ( !hbr 
&& m_hasFgCol 
) 
 392             colBg 
= GetBackgroundColour(); 
 395     // use the background colour override if a valid colour is given 
 398         ::SetBkColor(hdc
, wxColourToRGB(colBg
)); 
 400         // draw children with the same colour as the parent 
 401         wxBrush 
*brush 
= wxTheBrushList
->FindOrCreateBrush(colBg
, 
 403         hbr 
= (WXHBRUSH
)brush
->GetResourceHandle(); 
 406     // if we use custom background, we should set foreground ourselves too 
 407     if ( hbr 
&& !m_hasFgCol 
) 
 409         ::SetTextColor(hdc
, ::GetSysColor(COLOR_WINDOWTEXT
)); 
 411     //else: already set above 
 416 WXHBRUSH 
wxControl::MSWControlColor(WXHDC pDC
, WXHWND hWnd
) 
 420     if ( HasTransparentBackground() ) 
 421         ::SetBkMode((HDC
)pDC
, TRANSPARENT
); 
 422     else // if the control is opaque it shouldn't use the parents background 
 423         colBg 
= GetBackgroundColour(); 
 425     return DoMSWControlColor(pDC
, colBg
, hWnd
); 
 428 WXHBRUSH 
wxControl::MSWControlColorDisabled(WXHDC pDC
) 
 430     return DoMSWControlColor(pDC
, 
 431                              wxSystemSettings::GetColour(wxSYS_COLOUR_BTNFACE
), 
 435 // ---------------------------------------------------------------------------- 
 436 // wxControlWithItems 
 437 // ---------------------------------------------------------------------------- 
 439 void wxControlWithItems::MSWAllocStorage(const wxArrayStringsAdapter
& items
, 
 442     const unsigned numItems 
= items
.GetCount(); 
 443     unsigned long totalTextLength 
= numItems
; // for trailing '\0' characters 
 444     for ( unsigned i 
= 0; i 
< numItems
; ++i 
) 
 446         totalTextLength 
+= items
[i
].length(); 
 449     if ( SendMessage((HWND
)MSWGetItemsHWND(), wm
, numItems
, 
 450                      (LPARAM
)totalTextLength
*sizeof(wxChar
)) == LB_ERRSPACE 
) 
 452         wxLogLastError(wxT("SendMessage(XX_INITSTORAGE)")); 
 456 int wxControlWithItems::MSWInsertOrAppendItem(unsigned pos
, 
 457                                               const wxString
& item
, 
 460     LRESULT n 
= SendMessage((HWND
)MSWGetItemsHWND(), wm
, pos
, 
 461                             (LPARAM
)item
.wx_str()); 
 462     if ( n 
== CB_ERR 
|| n 
== CB_ERRSPACE 
) 
 464         wxLogLastError(wxT("SendMessage(XX_ADD/INSERTSTRING)")); 
 471 // --------------------------------------------------------------------------- 
 473 // --------------------------------------------------------------------------- 
 475 // this is used in radiobox.cpp and slider95.cpp and should be removed as soon 
 476 // as it is not needed there any more! 
 478 // Call this repeatedly for several wnds to find the overall size 
 480 // Call it initially with wxDefaultCoord for all values in rect. 
 481 // Keep calling for other widgets, and rect will be modified 
 482 // to calculate largest bounding rectangle. 
 483 void wxFindMaxSize(WXHWND wnd
, RECT 
*rect
) 
 485     int left 
= rect
->left
; 
 486     int right 
= rect
->right
; 
 488     int bottom 
= rect
->bottom
; 
 490     GetWindowRect((HWND
) wnd
, rect
); 
 495     if (left 
< rect
->left
) 
 498     if (right 
> rect
->right
) 
 504     if (bottom 
> rect
->bottom
) 
 505         rect
->bottom 
= bottom
; 
 508 #endif // wxUSE_CONTROLS