1 ///////////////////////////////////////////////////////////////////////////// 
   2 // Name:        msw/combobox.cpp 
   3 // Purpose:     wxComboBox class 
   4 // Author:      Julian Smart 
   8 // Copyright:   (c) Julian Smart and Markus Holzem 
   9 // Licence:     wxWindows licence 
  10 ///////////////////////////////////////////////////////////////////////////// 
  12 // ============================================================================ 
  14 // ============================================================================ 
  16 // ---------------------------------------------------------------------------- 
  18 // ---------------------------------------------------------------------------- 
  21 #pragma implementation "combobox.h" 
  24 // For compilers that support precompilation, includes "wx.h". 
  25 #include "wx/wxprec.h" 
  34     #include "wx/settings.h" 
  36     // for wxEVT_COMMAND_TEXT_ENTER 
  37     #include "wx/textctrl.h" 
  40 #include "wx/combobox.h" 
  42 #include "wx/clipbrd.h" 
  43 #include "wx/msw/private.h" 
  46     #if !defined(__GNUWIN32_OLD__) || defined(__CYGWIN10__) 
  49     #include "wx/tooltip.h" 
  50 #endif // wxUSE_TOOLTIPS 
  52 // ---------------------------------------------------------------------------- 
  54 // ---------------------------------------------------------------------------- 
  56 IMPLEMENT_DYNAMIC_CLASS(wxComboBox
, wxControl
) 
  58 // ---------------------------------------------------------------------------- 
  59 // function prototypes 
  60 // ---------------------------------------------------------------------------- 
  62 LRESULT APIENTRY _EXPORT 
wxComboEditWndProc(HWND hWnd
, 
  67 // --------------------------------------------------------------------------- 
  69 // --------------------------------------------------------------------------- 
  71 // the pointer to standard radio button wnd proc 
  72 static WXFARPROC gs_wndprocEdit 
= (WXFARPROC
)NULL
; 
  74 // ============================================================================ 
  76 // ============================================================================ 
  78 // ---------------------------------------------------------------------------- 
  79 // wnd proc for subclassed edit control 
  80 // ---------------------------------------------------------------------------- 
  82 LRESULT APIENTRY _EXPORT 
wxComboEditWndProc(HWND hWnd
, 
  87     HWND hwndCombo 
= ::GetParent(hWnd
); 
  88     wxWindow 
*win 
= wxFindWinFromHandle((WXHWND
)hwndCombo
); 
  92         // forward some messages to the combobox to generate the appropriate 
 100                 wxComboBox 
*combo 
= wxDynamicCast(win
, wxComboBox
); 
 103                     // we can get WM_KILLFOCUS while our parent is already half 
 104                     // destroyed and hence doesn't look like a combobx any 
 105                     // longer, check for it to avoid bogus assert failures 
 106                     if ( !win
->IsBeingDeleted() ) 
 108                         wxFAIL_MSG( _T("should have combo as parent") ); 
 111                 else if ( combo
->MSWProcessEditMsg(message
, wParam
, lParam
) ) 
 121                 wxCHECK_MSG( win
, 0, _T("should have a parent") ); 
 123                 if ( win
->GetWindowStyle() & wxPROCESS_ENTER 
) 
 125                     // need to return a custom dlg code or we'll never get it 
 126                     return DLGC_WANTMESSAGE
; 
 131         // deal with tooltips here 
 132 #if wxUSE_TOOLTIPS && defined(TTN_NEEDTEXT) 
 135                 wxCHECK_MSG( win
, 0, _T("should have a parent") ); 
 137                 NMHDR
* hdr 
= (NMHDR 
*)lParam
; 
 138                 if ( hdr
->code 
== TTN_NEEDTEXT 
) 
 140                     wxToolTip 
*tooltip 
= win
->GetToolTip(); 
 143                         TOOLTIPTEXT 
*ttt 
= (TOOLTIPTEXT 
*)lParam
; 
 144                         ttt
->lpszText 
= (wxChar 
*)tooltip
->GetTip().c_str(); 
 152 #endif // wxUSE_TOOLTIPS 
 155     return ::CallWindowProc(CASTWNDPROC gs_wndprocEdit
, hWnd
, message
, wParam
, lParam
); 
 158 WXHBRUSH 
wxComboBox::OnCtlColor(WXHDC pDC
, WXHWND 
WXUNUSED(pWnd
), WXUINT 
WXUNUSED(nCtlColor
), 
 164                                WXUINT 
WXUNUSED(message
), 
 165                                WXWPARAM 
WXUNUSED(wParam
), 
 166                                WXLPARAM 
WXUNUSED(lParam
) 
 173         HBRUSH hbrush 
= Ctl3dCtlColorEx(message
, wParam
, lParam
); 
 174         return (WXHBRUSH
) hbrush
; 
 176 #endif // wxUSE_CTL3D 
 179     if (GetParent()->GetTransparentBackground()) 
 180         SetBkMode(hdc
, TRANSPARENT
); 
 182         SetBkMode(hdc
, OPAQUE
); 
 184     wxColour colBack 
= GetBackgroundColour(); 
 187         colBack 
= wxSystemSettings::GetColour(wxSYS_COLOUR_3DFACE
); 
 189     ::SetBkColor(hdc
, wxColourToRGB(colBack
)); 
 190     ::SetTextColor(hdc
, wxColourToRGB(GetForegroundColour())); 
 192     wxBrush 
*brush 
= wxTheBrushList
->FindOrCreateBrush(colBack
, wxSOLID
); 
 194     return (WXHBRUSH
)brush
->GetResourceHandle(); 
 197 // ---------------------------------------------------------------------------- 
 199 // ---------------------------------------------------------------------------- 
 201 bool wxComboBox::MSWProcessEditMsg(WXUINT msg
, WXWPARAM wParam
, WXLPARAM lParam
) 
 206             // for compatibility with wxTextCtrl, generate a special message 
 207             // when Enter is pressed 
 208             if ( wParam 
== VK_RETURN 
) 
 210                 wxCommandEvent 
event(wxEVT_COMMAND_TEXT_ENTER
, m_windowId
); 
 211                 InitCommandEvent(event
); 
 212                 event
.SetString(GetValue()); 
 213                 event
.SetInt(GetSelection()); 
 214                 ProcessCommand(event
); 
 217             return HandleChar(wParam
, lParam
, TRUE 
/* isASCII */); 
 220             return HandleKeyDown(wParam
, lParam
); 
 223             return HandleKeyUp(wParam
, lParam
); 
 226             return HandleSetFocus((WXHWND
)wParam
); 
 229             return HandleKillFocus((WXHWND
)wParam
); 
 235 bool wxComboBox::MSWCommand(WXUINT param
, WXWORD 
WXUNUSED(id
)) 
 242             sel 
= GetSelection(); 
 245                 value 
= GetString(sel
); 
 247                 wxCommandEvent 
event(wxEVT_COMMAND_COMBOBOX_SELECTED
, GetId()); 
 249                 event
.SetEventObject(this); 
 250                 event
.SetString(value
); 
 251                 ProcessCommand(event
); 
 258             // fall through: for compability with wxGTK, also send the text 
 259             // update event when the selection changes (this also seems more 
 260             // logical as the text does change) 
 264                 wxCommandEvent 
event(wxEVT_COMMAND_TEXT_UPDATED
, GetId()); 
 266                 // if sel != -1, value was initialized above (and we can't use 
 267                 // GetValue() here as it would return the old selection and we 
 273                 else // we're synthesizing text updated event from sel change 
 275                     // we need to do this because the user code expects 
 276                     // wxComboBox::GetValue() to return the new value from 
 277                     // "text updated" handler but it hadn't been updated yet 
 281                 event
.SetString(value
); 
 282                 event
.SetEventObject(this); 
 283                 ProcessCommand(event
); 
 288     // there is no return value for the CBN_ notifications, so always return 
 289     // FALSE from here to pass the message to DefWindowProc() 
 293 WXHWND 
wxComboBox::GetEditHWND() const 
 295     // this function should not be called for wxCB_READONLY controls, it is 
 296     // the callers responsability to check this 
 297     wxASSERT_MSG( !(GetWindowStyle() & wxCB_READONLY
), 
 298                   _T("read-only combobox doesn't have any edit control") ); 
 302     HWND hwndEdit 
= ::ChildWindowFromPoint(GetHwnd(), pt
); 
 303     if ( !hwndEdit 
|| hwndEdit 
== GetHwnd() ) 
 305         wxFAIL_MSG(_T("not read only combobox without edit control?")); 
 308     return (WXHWND
)hwndEdit
; 
 311 bool wxComboBox::Create(wxWindow 
*parent
, wxWindowID id
, 
 312                         const wxString
& value
, 
 315                         int n
, const wxString choices
[], 
 317                         const wxValidator
& validator
, 
 318                         const wxString
& name
) 
 320     // pretend that wxComboBox is hidden while it is positioned and resized and 
 321     // show it only right before leaving this method because otherwise there is 
 322     // some noticeable flicker while the control rearranges itself 
 325     // first create wxWin object 
 326     if ( !CreateControl(parent
, id
, pos
, size
, style
, validator
, name
) ) 
 329     // get the right style 
 330     long msStyle 
= WS_TABSTOP 
| WS_VSCROLL 
| WS_HSCROLL 
| 
 331                    CBS_AUTOHSCROLL 
| CBS_NOINTEGRALHEIGHT 
/* | WS_CLIPSIBLINGS */; 
 332     if ( style 
& wxCB_READONLY 
) 
 333         msStyle 
|= CBS_DROPDOWNLIST
; 
 334     else if ( style 
& wxCB_SIMPLE 
) 
 335         msStyle 
|= CBS_SIMPLE
; // A list (shown always) and edit control 
 337         msStyle 
|= CBS_DROPDOWN
; 
 339     if ( style 
& wxCB_SORT 
) 
 342     if ( style 
& wxCLIP_SIBLINGS 
) 
 343         msStyle 
|= WS_CLIPSIBLINGS
; 
 346     // and now create the MSW control 
 347     if ( !MSWCreateControl(_T("COMBOBOX"), msStyle
) ) 
 350     // A choice/combobox normally has a white background (or other, depending 
 351     // on global settings) rather than inheriting the parent's background colour. 
 352     SetBackgroundColour(wxSystemSettings::GetColour(wxSYS_COLOUR_WINDOW
)); 
 354     for ( int i 
= 0; i 
< n
; i
++ ) 
 359     if ( !value
.IsEmpty() ) 
 364     // do this after appending the values to the combobox so that autosizing 
 366     SetSize(pos
.x
, pos
.y
, size
.x
, size
.y
); 
 368     // a (not read only) combobox is, in fact, 2 controls: the combobox itself 
 369     // and an edit control inside it and if we want to catch events from this 
 370     // edit control, we must subclass it as well 
 371     if ( !(style 
& wxCB_READONLY
) ) 
 373         gs_wndprocEdit 
= (WXFARPROC
)::SetWindowLong
 
 377                                         (LPARAM
)wxComboEditWndProc
 
 381     // and finally, show the control 
 387 void wxComboBox::SetValue(const wxString
& value
) 
 389     if ( HasFlag(wxCB_READONLY
) ) 
 390         SetStringSelection(value
); 
 392         SetWindowText(GetHwnd(), value
.c_str()); 
 395 // Clipboard operations 
 396 void wxComboBox::Copy() 
 398   SendMessage(GetHwnd(), WM_COPY
, 0, 0L); 
 401 void wxComboBox::Cut() 
 403   SendMessage(GetHwnd(), WM_CUT
, 0, 0L); 
 406 void wxComboBox::Paste() 
 408   SendMessage(GetHwnd(), WM_PASTE
, 0, 0L); 
 411 void wxComboBox::SetEditable(bool WXUNUSED(editable
)) 
 413   // Can't implement in MSW? 
 414 //  HWND hWnd = GetHwnd(); 
 415 //  SendMessage(hWnd, EM_SETREADONLY, (WPARAM)!editable, (LPARAM)0L); 
 418 void wxComboBox::SetInsertionPoint(long pos
) 
 420     if ( GetWindowStyle() & wxCB_READONLY 
) 
 424     HWND hWnd 
= GetHwnd(); 
 425     ::SendMessage(hWnd
, CB_SETEDITSEL
, 0, MAKELPARAM(pos
, pos
)); 
 426     HWND hEditWnd 
= (HWND
) GetEditHWND() ; 
 429         // Scroll insertion point into view 
 430         SendMessage(hEditWnd
, EM_SCROLLCARET
, (WPARAM
)0, (LPARAM
)0); 
 431         // Why is this necessary? (Copied from wxTextCtrl::SetInsertionPoint) 
 432         SendMessage(hEditWnd
, EM_REPLACESEL
, 0, (LPARAM
)_T("")); 
 437 void wxComboBox::SetInsertionPointEnd() 
 439     // setting insertion point doesn't make sense for read only comboboxes 
 440     if ( !(GetWindowStyle() & wxCB_READONLY
) ) 
 442         long pos 
= GetLastPosition(); 
 443         SetInsertionPoint(pos
); 
 447 long wxComboBox::GetInsertionPoint() const 
 450     DWORD Pos
=(DWORD
)SendMessage(GetHwnd(), CB_GETEDITSEL
, 0, 0L); 
 457 long wxComboBox::GetLastPosition() const 
 459     HWND hEditWnd 
= (HWND
) GetEditHWND(); 
 461     // Get number of characters in the last (only) line. We'll add this to the character 
 462     // index for the last line, 1st position. 
 463     int lineLength 
= (int)SendMessage(hEditWnd
, EM_LINELENGTH
, (WPARAM
) 0, (LPARAM
)0L); 
 465     return (long)(lineLength
); 
 468 void wxComboBox::Replace(long from
, long to
, const wxString
& value
) 
 473     // Now replace with 'value', by pasting. 
 474     wxSetClipboardData(wxDF_TEXT
, (wxObject 
*)(const wxChar 
*)value
, 0, 0); 
 476     // Paste into edit control 
 477     SendMessage(GetHwnd(), WM_PASTE
, (WPARAM
)0, (LPARAM
)0L); 
 481 void wxComboBox::Remove(long from
, long to
) 
 483     // Set selection and remove it 
 484     SetSelection(from
, to
); 
 485     SendMessage(GetHwnd(), WM_CUT
, (WPARAM
)0, (LPARAM
)0); 
 488 void wxComboBox::SetSelection(long from
, long to
) 
 490     HWND hWnd 
= GetHwnd(); 
 491     long fromChar 
= from
; 
 493     // if from and to are both -1, it means 
 494     // (in wxWindows) that all text should be selected. 
 495     // This translates into Windows convention 
 496     if ((from 
== -1) && (to 
== -1)) 
 504     SendMessage(hWnd
, CB_SETEDITSEL
, (WPARAM
)0, (LPARAM
)MAKELONG(fromChar
, toChar
)) 
 506     SendMessage(hWnd
, CB_SETEDITSEL
, (WPARAM
)fromChar
, (LPARAM
)toChar
) 
 510         wxLogDebug(_T("CB_SETEDITSEL failed"));