1 /////////////////////////////////////////////////////////////////////////////
2 // Name: univ/combobox.cpp
3 // Purpose: wxComboControl and wxComboBox implementation
4 // Author: Vadim Zeitlin
8 // Copyright: (c) 2000 SciTech Software, Inc. (www.scitechsoft.com)
9 // Licence: wxWindows licence
10 /////////////////////////////////////////////////////////////////////////////
15 +1. typing in the text should select the string in listbox
16 +2. scrollbars in listbox are unusable
17 +3. the initially selected item is not selected
18 ?4. kbd interface (what does GTK do?)
19 5. there is still autoscrolling without scrollbars - but is it bad?
22 // ============================================================================
24 // ============================================================================
26 // ----------------------------------------------------------------------------
28 // ----------------------------------------------------------------------------
31 #pragma implementation "univcombobox.h"
34 #include "wx/wxprec.h"
45 #include "wx/button.h"
46 #include "wx/combobox.h"
47 #include "wx/listbox.h"
48 #include "wx/textctrl.h"
49 #include "wx/bmpbuttn.h"
51 #include "wx/validate.h"
54 #include "wx/popupwin.h"
56 #include "wx/univ/renderer.h"
57 #include "wx/univ/inphand.h"
58 #include "wx/univ/theme.h"
61 The keyboard event flow:
63 1. they always come to the text ctrl
64 2. it forwards the ones it doesn't process to the wxComboControl
65 3. which passes them to the popup window if it is popped up
68 // ----------------------------------------------------------------------------
69 // wxComboButton is just a normal button except that it sends commands to the
70 // combobox and not its parent
71 // ----------------------------------------------------------------------------
73 class wxComboButton
: public wxBitmapButton
76 wxComboButton(wxComboControl
*combo
)
77 : wxBitmapButton(combo
->GetParent(), -1, wxNullBitmap
,
78 wxDefaultPosition
, wxDefaultSize
,
83 wxBitmap bmpNormal
, bmpPressed
, bmpDisabled
;
85 GetRenderer()->GetComboBitmaps(&bmpNormal
, &bmpPressed
, &bmpDisabled
);
86 SetBitmapLabel(bmpNormal
);
87 SetBitmapFocus(bmpNormal
);
88 SetBitmapSelected(bmpPressed
);
89 SetBitmapDisabled(bmpDisabled
);
91 SetSize(bmpNormal
.GetWidth(), bmpNormal
.GetHeight());
95 void OnButton(wxCommandEvent
& event
) { m_combo
->ShowPopup(); }
97 virtual wxSize
DoGetBestSize() const { return GetSize(); }
100 wxComboControl
*m_combo
;
102 DECLARE_EVENT_TABLE()
105 // ----------------------------------------------------------------------------
106 // wxComboListBox is a listbox modified to be used as a popup window in a
108 // ----------------------------------------------------------------------------
110 class wxComboListBox
: public wxListBox
, public wxComboPopup
114 wxComboListBox(wxComboControl
*combo
, int style
= 0);
115 virtual ~wxComboListBox();
117 // implement wxComboPopup methods
118 virtual bool SetSelection(const wxString
& value
);
119 virtual wxControl
*GetControl() { return this; }
120 virtual void OnShow();
123 // we shouldn't return height too big from here
124 virtual wxSize
DoGetBestClientSize() const;
126 // filter mouse move events happening outside the list box
127 void OnMouseMove(wxMouseEvent
& event
);
129 // called whenever the user selects or activates a listbox item
130 void OnSelect(wxCommandEvent
& event
);
132 // used to process wxUniv actions
133 bool PerformAction(const wxControlAction
& action
,
135 const wxString
& strArg
);
138 DECLARE_EVENT_TABLE()
141 // ----------------------------------------------------------------------------
142 // wxComboTextCtrl is a simple text ctrl which forwards
143 // wxEVT_COMMAND_TEXT_UPDATED events and all key events to the combobox
144 // ----------------------------------------------------------------------------
146 class wxComboTextCtrl
: public wxTextCtrl
149 wxComboTextCtrl(wxComboControl
*combo
,
150 const wxString
& value
,
152 const wxValidator
& validator
);
155 void OnKey(wxKeyEvent
& event
);
156 void OnText(wxCommandEvent
& event
);
159 wxComboControl
*m_combo
;
161 DECLARE_EVENT_TABLE()
164 // ----------------------------------------------------------------------------
165 // event tables and such
166 // ----------------------------------------------------------------------------
168 BEGIN_EVENT_TABLE(wxComboButton
, wxButton
)
169 EVT_BUTTON(-1, wxComboButton::OnButton
)
172 BEGIN_EVENT_TABLE(wxComboListBox
, wxListBox
)
173 EVT_LISTBOX(-1, wxComboListBox::OnSelect
)
174 EVT_LISTBOX_DCLICK(-1, wxComboListBox::OnSelect
)
175 EVT_MOTION(wxComboListBox::OnMouseMove
)
178 BEGIN_EVENT_TABLE(wxComboControl
, wxControl
)
179 EVT_KEY_DOWN(wxComboControl::OnKey
)
180 EVT_KEY_UP(wxComboControl::OnKey
)
183 BEGIN_EVENT_TABLE(wxComboTextCtrl
, wxTextCtrl
)
184 EVT_KEY_DOWN(wxComboTextCtrl::OnKey
)
185 EVT_KEY_UP(wxComboTextCtrl::OnKey
)
186 EVT_TEXT(-1, wxComboTextCtrl::OnText
)
189 IMPLEMENT_DYNAMIC_CLASS(wxComboBox
, wxControl
);
191 // ============================================================================
193 // ============================================================================
195 // ----------------------------------------------------------------------------
196 // wxComboControl creation
197 // ----------------------------------------------------------------------------
199 void wxComboControl::Init()
201 m_popup
= (wxComboPopup
*)NULL
;
202 m_winPopup
= (wxPopupComboWindow
*)NULL
;
203 m_isPopupShown
= FALSE
;
208 bool wxComboControl::Create(wxWindow
*parent
,
210 const wxString
& value
,
214 const wxValidator
& validator
,
215 const wxString
& name
)
217 // first create our own window, i.e. the one which will contain all
219 style
&= ~wxBORDER_NONE
;
220 style
|= wxBORDER_SUNKEN
;
221 if ( !wxControl::Create(parent
, id
, pos
, size
, style
, validator
, name
) )
224 // create the text control and the button as our siblings (*not* children),
225 // don't care about size/position here - they will be set in DoMoveWindow()
226 m_btn
= new wxComboButton(this);
227 m_text
= new wxComboTextCtrl(this,
229 style
& wxCB_READONLY
? wxTE_READONLY
: 0,
232 // for compatibility with the other ports, the height specified is the
233 // combined height of the combobox itself and the popup
236 // ok, use default height for popup too
241 m_heightPopup
= size
.y
- DoGetBestSize().y
;
244 DoSetSize(pos
.x
, pos
.y
, size
.x
, size
.y
);
246 // create the popup window immediately here to allow creating the controls
247 // with parent == GetPopupWindow() from the derived class ctor
248 m_winPopup
= new wxPopupComboWindow(this);
250 // have to disable this window to avoid interfering it with message
251 // processing to the text and the button... but pretend it is enabled to
252 // make IsEnabled() return TRUE
253 wxControl::Enable(FALSE
); // don't use non virtual Disable() here!
256 CreateInputHandler(wxINP_HANDLER_COMBOBOX
);
261 wxComboControl::~wxComboControl()
263 // as the button and the text control are the parent's children and not
264 // ours, we have to delete them manually - they are not deleted
265 // automatically by wxWindows when we're deleted
272 // ----------------------------------------------------------------------------
274 // ----------------------------------------------------------------------------
276 void wxComboControl::DoSetSize(int x
, int y
,
277 int width
, int height
,
280 // combo height is always fixed
281 wxControl::DoSetSize(x
, y
, width
, DoGetBestSize().y
, sizeFlags
);
284 wxSize
wxComboControl::DoGetBestClientSize() const
286 wxSize sizeBtn
= m_btn
->GetBestSize(),
287 sizeText
= m_text
->GetBestSize();
289 return wxSize(sizeBtn
.x
+ sizeText
.x
, wxMax(sizeBtn
.y
, sizeText
.y
));
292 void wxComboControl::DoMoveWindow(int x
, int y
, int width
, int height
)
294 wxControl::DoMoveWindow(x
, y
, width
, height
);
296 // position the subcontrols inside the client area
297 wxRect rectBorders
= GetRenderer()->GetBorderDimensions(GetBorder());
300 width
-= rectBorders
.x
+ rectBorders
.width
;
301 height
-= rectBorders
.y
+ rectBorders
.height
;
303 wxSize sizeBtn
= m_btn
->GetSize(),
304 sizeText
= m_text
->GetSize();
306 wxCoord wText
= width
- sizeBtn
.x
;
307 m_text
->SetSize(x
, y
, wText
, height
);
308 m_btn
->SetSize(x
+ wText
, y
, -1, height
);
311 // ----------------------------------------------------------------------------
313 // ----------------------------------------------------------------------------
315 bool wxComboControl::Enable(bool enable
)
317 if ( !wxControl::Enable(enable
) )
320 m_btn
->Enable(enable
);
321 m_text
->Enable(enable
);
326 bool wxComboControl::Show(bool show
)
328 if ( !wxControl::Show(show
) )
340 // ----------------------------------------------------------------------------
341 // popup window handling
342 // ----------------------------------------------------------------------------
344 void wxComboControl::SetPopupControl(wxComboPopup
*popup
)
349 void wxComboControl::ShowPopup()
351 wxCHECK_RET( m_popup
, _T("no popup to show in wxComboControl") );
352 wxCHECK_RET( !IsPopupShown(), _T("popup window already shown") );
354 wxControl
*control
= m_popup
->GetControl();
356 // size and position the popup window correctly
357 m_winPopup
->SetSize(GetSize().x
,
358 m_heightPopup
== -1 ? control
->GetBestSize().y
360 wxSize sizePopup
= m_winPopup
->GetClientSize();
361 control
->SetSize(0, 0, sizePopup
.x
, sizePopup
.y
);
363 // some controls don't accept the size we give then: e.g. a listbox may
364 // require more space to show its last row
365 wxSize sizeReal
= control
->GetSize();
366 if ( sizeReal
!= sizePopup
)
368 m_winPopup
->SetClientSize(sizeReal
);
371 m_winPopup
->PositionNearCombo();
374 m_winPopup
->Popup(m_text
);
376 m_popup
->SetSelection(m_text
->GetValue());
378 m_isPopupShown
= TRUE
;
381 void wxComboControl::HidePopup()
383 wxCHECK_RET( m_popup
, _T("no popup to hide in wxComboControl") );
384 wxCHECK_RET( IsPopupShown(), _T("popup window not shown") );
386 m_winPopup
->Dismiss();
388 m_isPopupShown
= FALSE
;
391 void wxComboControl::OnSelect(const wxString
& value
)
393 m_text
->SetValue(value
);
399 void wxComboControl::OnDismiss()
405 // ----------------------------------------------------------------------------
407 // ----------------------------------------------------------------------------
409 wxComboTextCtrl::wxComboTextCtrl(wxComboControl
*combo
,
410 const wxString
& value
,
412 const wxValidator
& validator
)
413 : wxTextCtrl(combo
->GetParent(), -1, value
,
414 wxDefaultPosition
, wxDefaultSize
,
415 wxBORDER_NONE
| style
,
421 void wxComboTextCtrl::OnText(wxCommandEvent
& event
)
423 if ( m_combo
->IsPopupShown() )
425 m_combo
->GetPopupControl()->SetSelection(GetValue());
428 // we need to make a copy of the event to have the correct originating
430 wxCommandEvent event2
= event
;
431 event2
.SetEventObject(m_combo
);
432 event2
.SetId(m_combo
->GetId());
434 // there is a small incompatibility with wxMSW here: the combobox gets the
435 // event before the text control in our case which corresponds to SMW
436 // CBN_EDITUPDATE notification and not CBN_EDITCHANGE one wxMSW currently
439 // if this is really a problem, we can play games with the event handlers
440 // to circumvent this
441 (void)m_combo
->ProcessEvent(event2
);
446 // pass the keys we don't process to the combo first
447 void wxComboTextCtrl::OnKey(wxKeyEvent
& event
)
449 switch ( event
.GetKeyCode() )
452 // the popup control gets it first but only if it is shown
453 if ( !m_combo
->IsPopupShown() )
464 (void)m_combo
->ProcessEvent(event
);
471 // ----------------------------------------------------------------------------
473 // ----------------------------------------------------------------------------
475 wxComboListBox::wxComboListBox(wxComboControl
*combo
, int style
)
476 : wxListBox(combo
->GetPopupWindow(), -1,
477 wxDefaultPosition
, wxDefaultSize
,
479 wxBORDER_SIMPLE
| wxLB_INT_HEIGHT
| style
),
482 // we don't react to the mouse events outside the window at all
486 wxComboListBox::~wxComboListBox()
490 bool wxComboListBox::SetSelection(const wxString
& value
)
492 // FindItem() would just find the current item for an empty string (it
493 // always matches), but we want to show the first one in such case
498 wxListBox::SetSelection(0);
500 //else: empty listbox - nothing to do
502 else if ( !FindItem(value
) )
511 void wxComboListBox::OnSelect(wxCommandEvent
& event
)
513 // first let the user code have the event
515 // all fields are already filled by the listbox, just change the event
516 // type and send it to the combo
517 wxCommandEvent event2
= event
;
518 event2
.SetEventType(wxEVT_COMMAND_COMBOBOX_SELECTED
);
519 event2
.SetEventObject(m_combo
);
520 event2
.SetId(m_combo
->GetId());
521 m_combo
->ProcessEvent(event2
);
523 // next update the combo and close the listbox
524 m_combo
->OnSelect(event
.GetString());
527 void wxComboListBox::OnShow()
531 bool wxComboListBox::PerformAction(const wxControlAction
& action
,
533 const wxString
& strArg
)
536 if ( action
== wxACTION_LISTBOX_FIND
)
538 // we don't let the listbox handle this as instead of just using the
539 // single key presses, as usual, we use the text ctrl value as prefix
540 // and this is done by wxComboControl itself
544 return wxListBox::PerformAction(action
, numArg
, strArg
);
547 void wxComboListBox::OnMouseMove(wxMouseEvent
& event
)
549 // while a wxComboListBox is shown, it always has capture, so if it doesn't
550 // we're about to go away anyhow (normally this shouldn't happen at all,
551 // but I don't put assert here as it just might do on other platforms and
552 // it doesn't break anythign anyhow)
553 if ( this == wxWindow::GetCapture() )
555 if ( HitTest(event
.GetPosition()) == wxHT_WINDOW_INSIDE
)
559 //else: popup shouldn't react to the mouse motions outside it, it only
560 // captures the mouse to be able to detect when it must be
561 // dismissed, so don't call Skip()
565 wxSize
wxComboListBox::DoGetBestClientSize() const
567 // don't return size too big or we risk to not fit on the screen
568 wxSize size
= wxListBox::DoGetBestClientSize();
569 wxCoord hChar
= GetCharHeight();
571 int nLines
= size
.y
/ hChar
;
573 // 10 is the same limit as used by wxMSW
582 // ----------------------------------------------------------------------------
584 // ----------------------------------------------------------------------------
586 void wxComboBox::Init()
588 m_lbox
= (wxListBox
*)NULL
;
591 bool wxComboBox::Create(wxWindow
*parent
,
593 const wxString
& value
,
597 const wxString
*choices
,
599 const wxValidator
& validator
,
600 const wxString
& name
)
602 if ( !wxComboControl::Create(parent
, id
, value
, pos
, size
, style
,
608 wxComboListBox
*combolbox
=
609 new wxComboListBox(this, style
& wxCB_SORT
? wxLB_SORT
: 0);
611 m_lbox
->Set(n
, choices
);
613 SetPopupControl(combolbox
);
618 wxComboBox::~wxComboBox()
622 // ----------------------------------------------------------------------------
623 // wxComboBox methods forwarded to wxTextCtrl
624 // ----------------------------------------------------------------------------
626 wxString
wxComboBox::GetValue() const
628 return GetText()->GetValue();
631 void wxComboBox::SetValue(const wxString
& value
)
633 GetText()->SetValue(value
);
636 void wxComboBox::Copy()
641 void wxComboBox::Cut()
646 void wxComboBox::Paste()
651 void wxComboBox::SetInsertionPoint(long pos
)
653 GetText()->SetInsertionPoint(pos
);
656 void wxComboBox::SetInsertionPointEnd()
658 GetText()->SetInsertionPointEnd();
661 long wxComboBox::GetInsertionPoint() const
663 return GetText()->GetInsertionPoint();
666 long wxComboBox::GetLastPosition() const
668 return GetText()->GetLastPosition();
671 void wxComboBox::Replace(long from
, long to
, const wxString
& value
)
673 GetText()->Replace(from
, to
, value
);
676 void wxComboBox::Remove(long from
, long to
)
678 GetText()->Remove(from
, to
);
681 void wxComboBox::SetSelection(long from
, long to
)
683 GetText()->SetSelection(from
, to
);
686 void wxComboBox::SetEditable(bool editable
)
688 GetText()->SetEditable(editable
);
691 // ----------------------------------------------------------------------------
692 // wxComboBox methods forwarded to wxListBox
693 // ----------------------------------------------------------------------------
695 void wxComboBox::Clear()
700 void wxComboBox::Delete(int n
)
702 GetLBox()->Delete(n
);
705 int wxComboBox::GetCount() const
707 return GetLBox()->GetCount();
710 wxString
wxComboBox::GetString(int n
) const
712 return GetLBox()->GetString(n
);
715 void wxComboBox::SetString(int n
, const wxString
& s
)
717 GetLBox()->SetString(n
, s
);
720 int wxComboBox::FindString(const wxString
& s
) const
722 return GetLBox()->FindString(s
);
725 void wxComboBox::Select(int n
)
727 wxCHECK_RET( (n
>= 0) && (n
< GetCount()), _T("invalid combobox index") );
729 GetLBox()->SetSelection(n
);
730 GetText()->SetValue(GetLBox()->GetString(n
));
733 int wxComboBox::GetSelection() const
735 // if the current value isn't one of the listbox strings, return -1
736 return FindString(GetText()->GetValue());
739 int wxComboBox::DoAppend(const wxString
& item
)
741 return GetLBox()->Append(item
);
744 void wxComboBox::DoSetItemClientData(int n
, void* clientData
)
746 GetLBox()->SetClientData(n
, clientData
);
749 void *wxComboBox::DoGetItemClientData(int n
) const
751 return GetLBox()->GetClientData(n
);
754 void wxComboBox::DoSetItemClientObject(int n
, wxClientData
* clientData
)
756 GetLBox()->SetClientObject(n
, clientData
);
759 wxClientData
* wxComboBox::DoGetItemClientObject(int n
) const
761 return GetLBox()->GetClientObject(n
);
764 // ----------------------------------------------------------------------------
766 // ----------------------------------------------------------------------------
768 void wxComboControl::OnKey(wxCommandEvent
& event
)
770 if ( m_isPopupShown
)
772 // pass it to the popped up control
773 (void)m_popup
->GetControl()->ProcessEvent(event
);
781 bool wxComboControl::PerformAction(const wxControlAction
& action
,
783 const wxString
& strArg
)
785 bool processed
= FALSE
;
786 if ( action
== wxACTION_COMBOBOX_POPUP
)
788 if ( !m_isPopupShown
)
795 else if ( action
== wxACTION_COMBOBOX_DISMISS
)
797 if ( m_isPopupShown
)
808 return wxControl::PerformAction(action
, numArg
, strArg
);
814 // ----------------------------------------------------------------------------
815 // wxStdComboBoxInputHandler
816 // ----------------------------------------------------------------------------
818 wxStdComboBoxInputHandler::wxStdComboBoxInputHandler(wxInputHandler
*inphand
)
819 : wxStdInputHandler(inphand
)
823 bool wxStdComboBoxInputHandler::HandleKey(wxControl
*control
,
824 const wxKeyEvent
& event
,
829 wxControlAction action
;
830 switch ( event
.GetKeyCode() )
833 action
= wxACTION_COMBOBOX_POPUP
;
837 action
= wxACTION_COMBOBOX_DISMISS
;
843 control
->PerformAction(action
);
849 return wxStdInputHandler::HandleKey(control
, event
, pressed
);
852 #endif // wxUSE_COMBOBOX