1 /////////////////////////////////////////////////////////////////////////////
3 // Purpose: wxComboBox class
4 // Author: Julian Smart
8 // Copyright: (c) Julian Smart and Markus Holzem
9 // Licence: wxWindows licence
10 /////////////////////////////////////////////////////////////////////////////
13 #pragma implementation "combobox.h"
16 // For compilers that support precompilation, includes "wx.h".
17 #include "wx/wxprec.h"
29 #include "wx/combobox.h"
30 #include "wx/clipbrd.h"
31 #include "wx/msw/private.h"
33 #if !USE_SHARED_LIBRARY
34 IMPLEMENT_DYNAMIC_CLASS(wxComboBox
, wxControl
)
37 bool wxComboBox::MSWCommand(WXUINT param
, WXWORD
WXUNUSED(id
))
39 if (param
== CBN_SELCHANGE
)
41 wxCommandEvent
event(wxEVT_COMMAND_COMBOBOX_SELECTED
, m_windowId
);
42 event
.SetInt(GetSelection());
43 event
.SetEventObject(this);
44 event
.SetString(copystring(GetStringSelection()));
45 ProcessCommand(event
);
46 delete[] event
.GetString();
49 else if (param
== CBN_EDITCHANGE
)
51 wxCommandEvent
event(wxEVT_COMMAND_TEXT_UPDATED
, m_windowId
);
52 event
.SetString(copystring(GetValue()));
53 ProcessCommand(event
);
54 delete[] event
.GetString();
60 bool wxComboBox::Create(wxWindow
*parent
, wxWindowID id
,
61 const wxString
& value
,
64 int n
, const wxString choices
[],
66 const wxValidator
& validator
,
70 SetValidator(validator
);
71 if (parent
) parent
->AddChild(this);
72 SetBackgroundColour(parent
->GetDefaultBackgroundColour()) ;
73 SetForegroundColour(parent
->GetDefaultForegroundColour()) ;
76 m_windowStyle
= style
;
79 m_windowId
= (int)NewControlId();
88 long msStyle
= WS_CHILD
| WS_HSCROLL
| WS_VSCROLL
89 | WS_TABSTOP
| WS_VISIBLE
| CBS_NOINTEGRALHEIGHT
;
90 if (m_windowStyle
& wxCB_READONLY
)
91 msStyle
|= CBS_DROPDOWNLIST
;
92 else if (m_windowStyle
& wxCB_SIMPLE
) // A list (shown always) and edit control
93 msStyle
|= CBS_SIMPLE
;
95 msStyle
|= CBS_DROPDOWN
;
97 if (m_windowStyle
& wxCB_SORT
)
101 WXDWORD exStyle
= Determine3DEffects(WS_EX_CLIENTEDGE
, &want3D
) ;
103 // Even with extended styles, need to combine with WS_BORDER
104 // for them to look right.
105 if (want3D
|| (m_windowStyle
& wxSIMPLE_BORDER
) || (m_windowStyle
& wxRAISED_BORDER
) ||
106 (m_windowStyle
& wxSUNKEN_BORDER
) || (m_windowStyle
& wxDOUBLE_BORDER
))
107 msStyle
|= WS_BORDER
;
109 HWND wx_combo
= CreateWindowEx(exStyle
, "COMBOBOX", NULL
,
111 0, 0, 0, 0, (HWND
) parent
->GetHWND(), (HMENU
)m_windowId
,
112 wxGetInstance(), NULL
);
117 Ctl3dSubclassCtl(wx_combo);
123 m_hWnd
= (WXHWND
)wx_combo
;
125 // Subclass again for purposes of dialog editing mode
126 SubclassWin((WXHWND
)wx_combo
);
128 SetFont(* parent
->GetFont());
130 for (i
= 0; i
< n
; i
++)
131 SendMessage(wx_combo
, CB_INSERTSTRING
, i
, (LONG
)(const char *)choices
[i
]);
132 SendMessage(wx_combo
, CB_SETCURSEL
, i
, 0);
134 SetSize(x
, y
, width
, height
);
136 SetWindowText(wx_combo
, (const char *)value
);
141 wxString
wxComboBox::GetValue(void) const
143 GetWindowText((HWND
) GetHWND(), wxBuffer
, 500);
144 return wxString(wxBuffer
);
147 void wxComboBox::SetValue(const wxString
& value
)
149 // If newlines are denoted by just 10, must stick 13 in front.
151 int len
= value
.Length();
153 for (i
= 0; i
< len
; i
++)
155 if ((i
> 0) && (value
[i
] == 10) && (value
[i
-1] != 13))
160 char *tmp
= new char[len
+ singletons
+ 1];
162 for (i
= 0; i
< len
; i
++)
164 if ((i
> 0) && (value
[i
] == 10) && (value
[i
-1] != 13))
173 SetWindowText((HWND
) GetHWND(), tmp
);
177 SetWindowText((HWND
) GetHWND(), (const char *)value
);
180 // Clipboard operations
181 void wxComboBox::Copy(void)
183 HWND hWnd
= (HWND
) GetHWND();
184 SendMessage(hWnd
, WM_COPY
, 0, 0L);
187 void wxComboBox::Cut(void)
189 HWND hWnd
= (HWND
) GetHWND();
190 SendMessage(hWnd
, WM_CUT
, 0, 0L);
193 void wxComboBox::Paste(void)
195 HWND hWnd
= (HWND
) GetHWND();
196 SendMessage(hWnd
, WM_PASTE
, 0, 0L);
199 void wxComboBox::SetEditable(bool editable
)
201 // Can't implement in MSW?
202 // HWND hWnd = (HWND) GetHWND();
203 // SendMessage(hWnd, EM_SETREADONLY, (WPARAM)!editable, (LPARAM)0L);
206 void wxComboBox::SetInsertionPoint(long pos
)
209 HWND hWnd = (HWND) GetHWND();
211 SendMessage(hWnd, EM_SETSEL, pos, pos);
212 SendMessage(hWnd, EM_SCROLLCARET, (WPARAM)0, (LPARAM)0);
214 SendMessage(hWnd, EM_SETSEL, 0, MAKELPARAM(pos, pos));
217 SendMessage(hWnd, EM_REPLACESEL, 0, (LPARAM)nothing);
221 void wxComboBox::SetInsertionPointEnd(void)
224 long pos = GetLastPosition();
225 SetInsertionPoint(pos);
229 long wxComboBox::GetInsertionPoint(void) const
232 DWORD Pos=(DWORD)SendMessage((HWND) GetHWND(), EM_GETSEL, 0, 0L);
238 long wxComboBox::GetLastPosition(void) const
241 HWND hWnd = (HWND) GetHWND();
243 // Will always return a number > 0 (according to docs)
244 int noLines = (int)SendMessage(hWnd, EM_GETLINECOUNT, (WPARAM)0, (LPARAM)0L);
246 // This gets the char index for the _beginning_ of the last line
247 int charIndex = (int)SendMessage(hWnd, EM_LINEINDEX, (WPARAM)(noLines-1), (LPARAM)0L);
249 // Get number of characters in the last line. We'll add this to the character
250 // index for the last line, 1st position.
251 int lineLength = (int)SendMessage(hWnd, EM_LINELENGTH, (WPARAM)charIndex, (LPARAM)0L);
253 return (long)(charIndex + lineLength);
258 void wxComboBox::Replace(long from
, long to
, const wxString
& value
)
261 HWND hWnd
= (HWND
) GetHWND();
262 long fromChar
= from
;
265 // Set selection and remove it
267 SendMessage(hWnd
, CB_SETEDITSEL
, fromChar
, toChar
);
269 SendMessage(hWnd
, CB_SETEDITSEL
, (WPARAM
)0, (LPARAM
)MAKELONG(fromChar
, toChar
));
271 SendMessage(hWnd
, WM_CUT
, (WPARAM
)0, (LPARAM
)0);
273 // Now replace with 'value', by pasting.
274 wxSetClipboardData(wxDF_TEXT
, (wxObject
*)(const char *)value
, 0, 0);
276 // Paste into edit control
277 SendMessage(hWnd
, WM_PASTE
, (WPARAM
)0, (LPARAM
)0L);
281 void wxComboBox::Remove(long from
, long to
)
283 HWND hWnd
= (HWND
) GetHWND();
284 long fromChar
= from
;
287 // Cut all selected text
289 SendMessage(hWnd
, CB_SETEDITSEL
, fromChar
, toChar
);
291 SendMessage(hWnd
, CB_SETEDITSEL
, (WPARAM
)0, (LPARAM
)MAKELONG(fromChar
, toChar
));
293 SendMessage(hWnd
, WM_CUT
, (WPARAM
)0, (LPARAM
)0);
296 void wxComboBox::SetSelection(long from
, long to
)
298 HWND hWnd
= (HWND
) GetHWND();
299 long fromChar
= from
;
301 // if from and to are both -1, it means
302 // (in wxWindows) that all text should be selected.
303 // This translates into Windows convention
304 if ((from
== -1) && (to
== -1))
311 SendMessage(hWnd
, CB_SETEDITSEL
, (WPARAM
)fromChar
, (LPARAM
)toChar
);
312 // SendMessage(hWnd, EM_SCROLLCARET, (WPARAM)0, (LPARAM)0);
314 // WPARAM is 0: selection is scrolled into view
315 SendMessage(hWnd
, CB_SETEDITSEL
, (WPARAM
)0, (LPARAM
)MAKELONG(fromChar
, toChar
));