]>
git.saurik.com Git - wxWidgets.git/blob - src/msw/wince/choicece.cpp
   1 /////////////////////////////////////////////////////////////////////////////// 
   2 // Name:        src/msw/wince/choicece.cpp 
   3 // Purpose:     wxChoice implementation for smart phones driven by WinCE 
   4 // Author:      Wlodzimierz ABX Skiba 
   8 // Copyright:   (c) Wlodzimierz Skiba 
   9 // License:     wxWindows licence 
  10 /////////////////////////////////////////////////////////////////////////////// 
  13 // ============================================================================ 
  15 // ============================================================================ 
  17 // ---------------------------------------------------------------------------- 
  19 // ---------------------------------------------------------------------------- 
  21 #if defined(__GNUG__) && !defined(NO_GCC_PRAGMA) 
  22     #pragma implementation "choicece.h" 
  25 // For compilers that support precompilation, includes "wx.h". 
  26 #include "wx/wxprec.h" 
  33     #include "wx/choice.h" 
  36 #include "wx/spinbutt.h" // for wxSpinnerBestSize 
  39 #include "wx/msw/missing.h" 
  40 #include "wx/msw/winundef.h" 
  42 #if wxUSE_CHOICE && defined(__SMARTPHONE__) && defined(__WXWINCE__) 
  44 #if wxUSE_EXTENDED_RTTI 
  47 IMPLEMENT_DYNAMIC_CLASS(wxChoice
, wxControl
) 
  50 #define GetBuddyHwnd()      (HWND)(m_hwndBuddy) 
  52 #define IsVertical(wxStyle) ( (wxStyle & wxSP_HORIZONTAL) != wxSP_HORIZONTAL ) 
  54 // ---------------------------------------------------------------------------- 
  56 // ---------------------------------------------------------------------------- 
  58 // the margin between the up-down control and its buddy (can be arbitrary, 
  59 // choose what you like - or may be decide during run-time depending on the 
  61 static const int MARGIN_BETWEEN 
= 0; 
  63 // ============================================================================ 
  65 // ============================================================================ 
  67 wxArrayChoiceSpins 
wxChoice::ms_allChoiceSpins
; 
  69 // ---------------------------------------------------------------------------- 
  70 // wnd proc for the buddy text ctrl 
  71 // ---------------------------------------------------------------------------- 
  73 LRESULT APIENTRY _EXPORT 
wxBuddyChoiceWndProc(HWND hwnd
, 
  78     wxChoice 
*spin 
= (wxChoice 
*)wxGetWindowUserData(hwnd
); 
  80     // forward some messages (the key and focus ones only so far) to 
  85             // if the focus comes from the spin control itself, don't set it 
  86             // back to it -- we don't want to go into an infinite loop 
  87             if ( (WXHWND
)wParam 
== spin
->GetHWND() ) 
  96             spin
->MSWWindowProc(message
, wParam
, lParam
); 
  98             // The control may have been deleted at this point, so check. 
  99             if ( !::IsWindow(hwnd
) || wxGetWindowUserData(hwnd
) != spin 
) 
 104             // we want to get WXK_RETURN in order to generate the event for it 
 105             return DLGC_WANTCHARS
; 
 108     return ::CallWindowProc(CASTWNDPROC spin
->GetBuddyWndProc(), 
 109                             hwnd
, message
, wParam
, lParam
); 
 112 // ---------------------------------------------------------------------------- 
 114 // ---------------------------------------------------------------------------- 
 116 bool wxChoice::Create(wxWindow 
*parent
, 
 120                       int n
, const wxString choices
[], 
 122                       const wxValidator
& validator
, 
 123                       const wxString
& name
) 
 125     return CreateAndInit(parent
, id
, pos
, size
, n
, choices
, style
, 
 129 bool wxChoice::CreateAndInit(wxWindow 
*parent
, 
 133                              int n
, const wxString choices
[], 
 135                              const wxValidator
& validator
, 
 136                              const wxString
& name
) 
 138     if ( !(style 
& wxSP_VERTICAL
) ) 
 139         style 
|= wxSP_HORIZONTAL
; 
 141     if ( (style 
& wxBORDER_MASK
) == wxBORDER_DEFAULT 
) 
 142         style 
|= wxBORDER_SIMPLE
; 
 144     style 
|= wxSP_ARROW_KEYS
; 
 146     SetWindowStyle(style
); 
 149     WXDWORD msStyle 
= MSWGetStyle(GetWindowStyle(), & exStyle
) ; 
 151     wxSize 
sizeText(size
), sizeBtn(size
); 
 152     sizeBtn
.x 
= GetBestSpinerSize(IsVertical(style
)).x
; 
 154     if ( sizeText
.x 
== wxDefaultCoord 
) 
 156         // DEFAULT_ITEM_WIDTH is the default width for the text control 
 157         sizeText
.x 
= DEFAULT_ITEM_WIDTH 
+ MARGIN_BETWEEN 
+ sizeBtn
.x
; 
 160     sizeText
.x 
-= sizeBtn
.x 
+ MARGIN_BETWEEN
; 
 161     if ( sizeText
.x 
<= 0 ) 
 163         wxLogDebug(_T("not enough space for wxSpinCtrl!")); 
 167     posBtn
.x 
+= sizeText
.x 
+ MARGIN_BETWEEN
; 
 169     // we must create the list control before the spin button for the purpose 
 170     // of the dialog navigation: if there is a static text just before the spin 
 171     // control, activating it by Alt-letter should give focus to the text 
 172     // control, not the spin and the dialog navigation code will give focus to 
 173     // the next control (at Windows level), not the one after it 
 175     // create the text window 
 177     m_hwndBuddy 
= (WXHWND
)::CreateWindowEx
 
 179                      exStyle
,                // sunken border 
 180                      _T("LISTBOX"),          // window class 
 181                      NULL
,                   // no window title 
 182                      msStyle
,                // style (will be shown later) 
 183                      pos
.x
, pos
.y
,           // position 
 184                      0, 0,                   // size (will be set later) 
 185                      GetHwndOf(parent
),      // parent 
 186                      (HMENU
)-1,              // control id 
 187                      wxGetInstance(),        // app instance 
 188                      NULL                    
// unused client data 
 193         wxLogLastError(wxT("CreateWindow(buddy text window)")); 
 198     // initialize wxControl 
 199     if ( !CreateControl(parent
, id
, posBtn
, sizeBtn
, style
, validator
, name
) ) 
 202     // now create the real HWND 
 203     WXDWORD spiner_style 
= WS_VISIBLE 
| 
 209     if ( !IsVertical(style
) ) 
 210         spiner_style 
|= UDS_HORZ
; 
 212     if ( style 
& wxSP_WRAP 
) 
 213         spiner_style 
|= UDS_WRAP
; 
 215     if ( !MSWCreateControl(UPDOWN_CLASS
, spiner_style
, posBtn
, sizeBtn
, _T(""), 0) ) 
 218     // subclass the text ctrl to be able to intercept some events 
 219     wxSetWindowUserData(GetBuddyHwnd(), this); 
 220     m_wndProcBuddy 
= (WXFARPROC
)wxSetWindowProc(GetBuddyHwnd(), 
 221                                                 wxBuddyChoiceWndProc
); 
 223     // set up fonts and colours  (This is nomally done in MSWCreateControl) 
 226         SetFont(GetDefaultAttributes().font
); 
 228     // set the size of the text window - can do it only now, because we 
 229     // couldn't call DoGetBestSize() before as font wasn't set 
 230     if ( sizeText
.y 
<= 0 ) 
 233         wxGetCharSize(GetHWND(), &cx
, &cy
, GetFont()); 
 235         sizeText
.y 
= EDIT_HEIGHT_FROM_CHAR_HEIGHT(cy
); 
 240     (void)::ShowWindow(GetBuddyHwnd(), SW_SHOW
); 
 242     // associate the list window with the spin button 
 243     (void)::SendMessage(GetHwnd(), UDM_SETBUDDY
, (WPARAM
)GetBuddyHwnd(), 0); 
 245     // do it after finishing with m_hwndBuddy creation to avoid generating 
 246     // initial wxEVT_COMMAND_TEXT_UPDATED message 
 247     ms_allChoiceSpins
.Add(this); 
 249     // initialize the controls contents 
 250     for ( int i 
= 0; i 
< n
; i
++ ) 
 258 bool wxChoice::Create(wxWindow 
*parent
, 
 262                       const wxArrayString
& choices
, 
 264                       const wxValidator
& validator
, 
 265                       const wxString
& name
) 
 267     wxCArrayString 
chs(choices
); 
 268     return Create(parent
, id
, pos
, size
, chs
.GetCount(), chs
.GetStrings(), 
 269                   style
, validator
, name
); 
 272 WXDWORD 
wxChoice::MSWGetStyle(long style
, WXDWORD 
*exstyle
) const 
 274     // we never have an external border 
 275     WXDWORD msStyle 
= wxControl::MSWGetStyle
 
 277                         (style 
& ~wxBORDER_MASK
) | wxBORDER_NONE
, exstyle
 
 280     msStyle 
|= WS_VISIBLE
; 
 282     // wxChoice-specific styles 
 283     msStyle 
|= LBS_NOINTEGRALHEIGHT
; 
 284     if ( style 
& wxCB_SORT 
) 
 290 wxChoice::~wxChoice() 
 295 // ---------------------------------------------------------------------------- 
 296 // adding/deleting items to/from the list 
 297 // ---------------------------------------------------------------------------- 
 299 int wxChoice::DoAppend(const wxString
& item
) 
 301     int n 
= (int)::SendMessage(GetBuddyHwnd(), LB_ADDSTRING
, 0, (LPARAM
)item
.c_str()); 
 305         wxLogLastError(wxT("SendMessage(LB_ADDSTRING)")); 
 311 int wxChoice::DoInsert(const wxString
& item
, int pos
) 
 313     wxCHECK_MSG(!(GetWindowStyle() & wxCB_SORT
), -1, wxT("can't insert into choice")); 
 314     wxCHECK_MSG((pos
>=0) && (pos
<=GetCount()), -1, wxT("invalid index")); 
 316     int n 
= (int)::SendMessage(GetBuddyHwnd(), LB_INSERTSTRING
, pos
, (LPARAM
)item
.c_str()); 
 319         wxLogLastError(wxT("SendMessage(LB_INSERTSTRING)")); 
 325 void wxChoice::Delete(int n
) 
 327     wxCHECK_RET( n 
< GetCount(), wxT("invalid item index in wxChoice::Delete") ); 
 329     if ( HasClientObjectData() ) 
 331         delete GetClientObject(n
); 
 334     ::SendMessage(GetBuddyHwnd(), LB_DELETESTRING
, n
, 0); 
 337 void wxChoice::Clear() 
 341     ::SendMessage(GetBuddyHwnd(), LB_RESETCONTENT
, 0, 0); 
 344 void wxChoice::Free() 
 346     if ( HasClientObjectData() ) 
 348         size_t count 
= GetCount(); 
 349         for ( size_t n 
= 0; n 
< count
; n
++ ) 
 351             delete GetClientObject(n
); 
 356 // ---------------------------------------------------------------------------- 
 358 // ---------------------------------------------------------------------------- 
 360 int wxChoice::GetSelection() const 
 362     return (int)::SendMessage(GetBuddyHwnd(), LB_GETCURSEL
, 0, 0); 
 365 void wxChoice::SetSelection(int n
) 
 367     ::SendMessage(GetBuddyHwnd(), LB_SETCURSEL
, n
, 0); 
 370 // ---------------------------------------------------------------------------- 
 371 // string list functions 
 372 // ---------------------------------------------------------------------------- 
 374 int wxChoice::GetCount() const 
 376     return (int)::SendMessage(GetBuddyHwnd(), LB_GETCOUNT
, 0, 0); 
 379 int wxChoice::FindString(const wxString
& s
) const 
 381     int pos 
= (int)::SendMessage(GetBuddyHwnd(), LB_FINDSTRINGEXACT
, 
 382                                (WPARAM
)-1, (LPARAM
)s
.c_str()); 
 384     return pos 
== LB_ERR 
? wxNOT_FOUND 
: pos
; 
 387 void wxChoice::SetString(int n
, const wxString
& s
) 
 389     wxCHECK_RET( n 
>= 0 && n 
< GetCount(), 
 390                  wxT("invalid item index in wxChoice::SetString") ); 
 392     // we have to delete and add back the string as there is no way to change a 
 395     // we need to preserve the client data 
 397     if ( m_clientDataItemsType 
!= wxClientData_None 
) 
 399         data 
= DoGetItemClientData(n
); 
 401     else // no client data 
 406     ::SendMessage(GetBuddyHwnd(), LB_DELETESTRING
, n
, 0); 
 407     ::SendMessage(GetBuddyHwnd(), LB_INSERTSTRING
, n
, (LPARAM
)s
.c_str() ); 
 411         DoSetItemClientData(n
, data
); 
 413     //else: it's already NULL by default 
 416 wxString 
wxChoice::GetString(int n
) const 
 418     int len 
= (int)::SendMessage(GetBuddyHwnd(), LB_GETTEXTLEN
, n
, 0); 
 421     if ( len 
!= LB_ERR 
&& len 
> 0 ) 
 428                 (LPARAM
)(wxChar 
*)wxStringBuffer(str
, len
) 
 431             wxLogLastError(wxT("SendMessage(LB_GETLBTEXT)")); 
 438 // ---------------------------------------------------------------------------- 
 440 // ---------------------------------------------------------------------------- 
 442 void wxChoice::DoSetItemClientData( int n
, void* clientData 
) 
 444     if ( ::SendMessage(GetHwnd(), LB_SETITEMDATA
, 
 445                        n
, (LPARAM
)clientData
) == LB_ERR 
) 
 447         wxLogLastError(wxT("LB_SETITEMDATA")); 
 451 void* wxChoice::DoGetItemClientData( int n 
) const 
 453     LPARAM rc 
= ::SendMessage(GetHwnd(), LB_GETITEMDATA
, n
, 0); 
 456         wxLogLastError(wxT("LB_GETITEMDATA")); 
 458         // unfortunately, there is no way to return an error code to the user 
 465 void wxChoice::DoSetItemClientObject( int n
, wxClientData
* clientData 
) 
 467     DoSetItemClientData(n
, clientData
); 
 470 wxClientData
* wxChoice::DoGetItemClientObject( int n 
) const 
 472     return (wxClientData 
*)DoGetItemClientData(n
); 
 475 // ---------------------------------------------------------------------------- 
 477 // ---------------------------------------------------------------------------- 
 479 wxSize 
wxChoice::DoGetBestSize() const 
 481     wxSize sizeBtn 
= GetBestSpinerSize(IsVertical(GetWindowStyle())); 
 482     sizeBtn
.x 
+= DEFAULT_ITEM_WIDTH 
+ MARGIN_BETWEEN
; 
 485     wxGetCharSize(GetHWND(), NULL
, &y
, GetFont()); 
 486     y 
= EDIT_HEIGHT_FROM_CHAR_HEIGHT(y
); 
 488     // JACS: we should always use the height calculated 
 489     // from above, because otherwise we'll get a spin control 
 490     // that's too big. So never use the height calculated 
 491     // from wxSpinButton::DoGetBestSize(). 
 493     // if ( sizeBtn.y < y ) 
 495         // make the text tall enough 
 502 void wxChoice::DoMoveWindow(int x
, int y
, int width
, int height
) 
 504     int widthBtn 
= GetBestSpinerSize(IsVertical(GetWindowStyle())).x
; 
 505     int widthText 
= width 
- widthBtn 
- MARGIN_BETWEEN
; 
 506     if ( widthText 
<= 0 ) 
 508         wxLogDebug(_T("not enough space for wxSpinCtrl!")); 
 511     if ( !::MoveWindow(GetBuddyHwnd(), x
, y
, widthText
, height
, TRUE
) ) 
 513         wxLogLastError(wxT("MoveWindow(buddy)")); 
 516     x 
+= widthText 
+ MARGIN_BETWEEN
; 
 517     if ( !::MoveWindow(GetHwnd(), x
, y
, widthBtn
, height
, TRUE
) ) 
 519         wxLogLastError(wxT("MoveWindow")); 
 523 // get total size of the control 
 524 void wxChoice::DoGetSize(int *x
, int *y
) const 
 526     RECT spinrect
, textrect
, ctrlrect
; 
 527     GetWindowRect(GetHwnd(), &spinrect
); 
 528     GetWindowRect(GetBuddyHwnd(), &textrect
); 
 529     UnionRect(&ctrlrect
, &textrect
, &spinrect
); 
 532         *x 
= ctrlrect
.right 
- ctrlrect
.left
; 
 534         *y 
= ctrlrect
.bottom 
- ctrlrect
.top
; 
 537 void wxChoice::DoGetPosition(int *x
, int *y
) const 
 539     // hack: pretend that our HWND is the text control just for a moment 
 540     WXHWND hWnd 
= GetHWND(); 
 541     wxConstCast(this, wxChoice
)->m_hWnd 
= m_hwndBuddy
; 
 543     wxChoiceBase::DoGetPosition(x
, y
); 
 545     wxConstCast(this, wxChoice
)->m_hWnd 
= hWnd
; 
 548 #endif // wxUSE_CHOICE && __SMARTPHONE__ && __WXWINCE__