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 event
.SetEventObject(this);
54 ProcessCommand(event
);
55 delete[] event
.GetString();
61 bool wxComboBox::Create(wxWindow
*parent
, wxWindowID id
,
62 const wxString
& value
,
65 int n
, const wxString choices
[],
67 const wxValidator
& validator
,
71 SetValidator(validator
);
72 if (parent
) parent
->AddChild(this);
73 SetBackgroundColour(parent
->GetDefaultBackgroundColour()) ;
74 SetForegroundColour(parent
->GetDefaultForegroundColour()) ;
77 m_windowStyle
= style
;
80 m_windowId
= (int)NewControlId();
89 long msStyle
= WS_CHILD
| WS_HSCROLL
| WS_VSCROLL
90 | WS_TABSTOP
| WS_VISIBLE
| CBS_NOINTEGRALHEIGHT
;
91 if (m_windowStyle
& wxCB_READONLY
)
92 msStyle
|= CBS_DROPDOWNLIST
;
93 else if (m_windowStyle
& wxCB_SIMPLE
) // A list (shown always) and edit control
94 msStyle
|= CBS_SIMPLE
;
96 msStyle
|= CBS_DROPDOWN
;
98 if (m_windowStyle
& wxCB_SORT
)
102 WXDWORD exStyle
= Determine3DEffects(WS_EX_CLIENTEDGE
, &want3D
) ;
104 // Even with extended styles, need to combine with WS_BORDER
105 // for them to look right.
106 if (want3D
|| (m_windowStyle
& wxSIMPLE_BORDER
) || (m_windowStyle
& wxRAISED_BORDER
) ||
107 (m_windowStyle
& wxSUNKEN_BORDER
) || (m_windowStyle
& wxDOUBLE_BORDER
))
108 msStyle
|= WS_BORDER
;
110 HWND wx_combo
= CreateWindowEx(exStyle
, "COMBOBOX", NULL
,
112 0, 0, 0, 0, (HWND
) parent
->GetHWND(), (HMENU
)m_windowId
,
113 wxGetInstance(), NULL
);
118 Ctl3dSubclassCtl(wx_combo);
124 m_hWnd
= (WXHWND
)wx_combo
;
126 // Subclass again for purposes of dialog editing mode
127 SubclassWin((WXHWND
)wx_combo
);
129 SetFont(* parent
->GetFont());
131 for (i
= 0; i
< n
; i
++)
132 SendMessage(wx_combo
, CB_INSERTSTRING
, i
, (LONG
)(const char *)choices
[i
]);
133 SendMessage(wx_combo
, CB_SETCURSEL
, i
, 0);
135 SetSize(x
, y
, width
, height
);
137 SetWindowText(wx_combo
, (const char *)value
);
142 wxString
wxComboBox::GetValue(void) const
144 GetWindowText((HWND
) GetHWND(), wxBuffer
, 500);
145 return wxString(wxBuffer
);
148 void wxComboBox::SetValue(const wxString
& value
)
150 // If newlines are denoted by just 10, must stick 13 in front.
152 int len
= value
.Length();
154 for (i
= 0; i
< len
; i
++)
156 if ((i
> 0) && (value
[i
] == 10) && (value
[i
-1] != 13))
161 char *tmp
= new char[len
+ singletons
+ 1];
163 for (i
= 0; i
< len
; i
++)
165 if ((i
> 0) && (value
[i
] == 10) && (value
[i
-1] != 13))
174 SetWindowText((HWND
) GetHWND(), tmp
);
178 SetWindowText((HWND
) GetHWND(), (const char *)value
);
181 // Clipboard operations
182 void wxComboBox::Copy(void)
184 HWND hWnd
= (HWND
) GetHWND();
185 SendMessage(hWnd
, WM_COPY
, 0, 0L);
188 void wxComboBox::Cut(void)
190 HWND hWnd
= (HWND
) GetHWND();
191 SendMessage(hWnd
, WM_CUT
, 0, 0L);
194 void wxComboBox::Paste(void)
196 HWND hWnd
= (HWND
) GetHWND();
197 SendMessage(hWnd
, WM_PASTE
, 0, 0L);
200 void wxComboBox::SetEditable(bool editable
)
202 // Can't implement in MSW?
203 // HWND hWnd = (HWND) GetHWND();
204 // SendMessage(hWnd, EM_SETREADONLY, (WPARAM)!editable, (LPARAM)0L);
207 void wxComboBox::SetInsertionPoint(long pos
)
210 HWND hWnd = (HWND) GetHWND();
212 SendMessage(hWnd, EM_SETSEL, pos, pos);
213 SendMessage(hWnd, EM_SCROLLCARET, (WPARAM)0, (LPARAM)0);
215 SendMessage(hWnd, EM_SETSEL, 0, MAKELPARAM(pos, pos));
218 SendMessage(hWnd, EM_REPLACESEL, 0, (LPARAM)nothing);
222 void wxComboBox::SetInsertionPointEnd(void)
225 long pos = GetLastPosition();
226 SetInsertionPoint(pos);
230 long wxComboBox::GetInsertionPoint(void) const
233 DWORD Pos=(DWORD)SendMessage((HWND) GetHWND(), EM_GETSEL, 0, 0L);
239 long wxComboBox::GetLastPosition(void) const
242 HWND hWnd = (HWND) GetHWND();
244 // Will always return a number > 0 (according to docs)
245 int noLines = (int)SendMessage(hWnd, EM_GETLINECOUNT, (WPARAM)0, (LPARAM)0L);
247 // This gets the char index for the _beginning_ of the last line
248 int charIndex = (int)SendMessage(hWnd, EM_LINEINDEX, (WPARAM)(noLines-1), (LPARAM)0L);
250 // Get number of characters in the last line. We'll add this to the character
251 // index for the last line, 1st position.
252 int lineLength = (int)SendMessage(hWnd, EM_LINELENGTH, (WPARAM)charIndex, (LPARAM)0L);
254 return (long)(charIndex + lineLength);
259 void wxComboBox::Replace(long from
, long to
, const wxString
& value
)
262 HWND hWnd
= (HWND
) GetHWND();
263 long fromChar
= from
;
266 // Set selection and remove it
268 SendMessage(hWnd
, CB_SETEDITSEL
, fromChar
, toChar
);
270 SendMessage(hWnd
, CB_SETEDITSEL
, (WPARAM
)0, (LPARAM
)MAKELONG(fromChar
, toChar
));
272 SendMessage(hWnd
, WM_CUT
, (WPARAM
)0, (LPARAM
)0);
274 // Now replace with 'value', by pasting.
275 wxSetClipboardData(wxDF_TEXT
, (wxObject
*)(const char *)value
, 0, 0);
277 // Paste into edit control
278 SendMessage(hWnd
, WM_PASTE
, (WPARAM
)0, (LPARAM
)0L);
282 void wxComboBox::Remove(long from
, long to
)
284 HWND hWnd
= (HWND
) GetHWND();
285 long fromChar
= from
;
288 // Cut all selected text
290 SendMessage(hWnd
, CB_SETEDITSEL
, fromChar
, toChar
);
292 SendMessage(hWnd
, CB_SETEDITSEL
, (WPARAM
)0, (LPARAM
)MAKELONG(fromChar
, toChar
));
294 SendMessage(hWnd
, WM_CUT
, (WPARAM
)0, (LPARAM
)0);
297 void wxComboBox::SetSelection(long from
, long to
)
299 HWND hWnd
= (HWND
) GetHWND();
300 long fromChar
= from
;
302 // if from and to are both -1, it means
303 // (in wxWindows) that all text should be selected.
304 // This translates into Windows convention
305 if ((from
== -1) && (to
== -1))
312 SendMessage(hWnd
, CB_SETEDITSEL
, (WPARAM
)fromChar
, (LPARAM
)toChar
);
313 // SendMessage(hWnd, EM_SCROLLCARET, (WPARAM)0, (LPARAM)0);
315 // WPARAM is 0: selection is scrolled into view
316 SendMessage(hWnd
, CB_SETEDITSEL
, (WPARAM
)0, (LPARAM
)MAKELONG(fromChar
, toChar
));