]> git.saurik.com Git - wxWidgets.git/blob - src/common/textentrycmn.cpp
Fix out of bounds string access in wxMSW wxDirDialog.
[wxWidgets.git] / src / common / textentrycmn.cpp
1 ///////////////////////////////////////////////////////////////////////////////
2 // Name: src/common/textentrycmn.cpp
3 // Purpose: wxTextEntryBase implementation
4 // Author: Vadim Zeitlin
5 // Created: 2007-09-26
6 // RCS-ID: $Id$
7 // Copyright: (c) 2007 Vadim Zeitlin <vadim@wxwindows.org>
8 // Licence: wxWindows licence
9 ///////////////////////////////////////////////////////////////////////////////
10
11 // ============================================================================
12 // declarations
13 // ============================================================================
14
15 // ----------------------------------------------------------------------------
16 // headers
17 // ----------------------------------------------------------------------------
18
19 // for compilers that support precompilation, includes "wx.h".
20 #include "wx/wxprec.h"
21
22 #ifdef __BORLANDC__
23 #pragma hdrstop
24 #endif
25
26 #if wxUSE_TEXTCTRL || wxUSE_COMBOBOX
27
28 #ifndef WX_PRECOMP
29 #include "wx/window.h"
30 #include "wx/dataobj.h"
31 #endif //WX_PRECOMP
32
33 #include "wx/textentry.h"
34 #include "wx/clipbrd.h"
35
36 // ----------------------------------------------------------------------------
37 // wxTextEntryHintData
38 // ----------------------------------------------------------------------------
39
40 class WXDLLIMPEXP_CORE wxTextEntryHintData wxBIND_OR_CONNECT_HACK_ONLY_BASE_CLASS
41 {
42 public:
43 wxTextEntryHintData(wxTextEntryBase *entry, wxWindow *win)
44 : m_entry(entry),
45 m_win(win),
46 m_text(m_entry->GetValue())
47 {
48 wxBIND_OR_CONNECT_HACK(win, wxEVT_SET_FOCUS, wxFocusEventHandler,
49 wxTextEntryHintData::OnSetFocus, this);
50 wxBIND_OR_CONNECT_HACK(win, wxEVT_KILL_FOCUS, wxFocusEventHandler,
51 wxTextEntryHintData::OnKillFocus, this);
52 wxBIND_OR_CONNECT_HACK(win, wxEVT_COMMAND_TEXT_UPDATED,
53 wxCommandEventHandler,
54 wxTextEntryHintData::OnTextChanged, this);
55 }
56
57 // default dtor is ok
58
59 // Get the real text of the control such as it was before we replaced it
60 // with the hint.
61 const wxString& GetText() const { return m_text; }
62
63 // Set the hint to show, shouldn't be empty normally.
64 //
65 // This should be called after creating a new wxTextEntryHintData object
66 // and may be called more times in the future.
67 void SetHintString(const wxString& hint)
68 {
69 m_hint = hint;
70
71 if ( !m_win->HasFocus() )
72 ShowHintIfAppropriate();
73 //else: The new hint will be shown later when we lose focus.
74 }
75
76 const wxString& GetHintString() const { return m_hint; }
77
78 // This is called whenever the text control contents changes.
79 //
80 // We call it ourselves when this change generates an event but it's also
81 // necessary to call it explicitly from wxTextEntry::ChangeValue() as it,
82 // by design, does not generate any events.
83 void HandleTextUpdate(const wxString& text)
84 {
85 m_text = text;
86
87 // If we're called because of a call to Set or ChangeValue(), the
88 // control may still have the hint text colour, reset it in this case.
89 RestoreTextColourIfNecessary();
90 }
91
92 private:
93 // Show the hint in the window if we should do it, i.e. if the window
94 // doesn't have any text of its own.
95 void ShowHintIfAppropriate()
96 {
97 // Never overwrite existing window text.
98 if ( !m_text.empty() )
99 return;
100
101 // Save the old text colour and set a more inconspicuous one for the
102 // hint.
103 m_colFg = m_win->GetForegroundColour();
104 m_win->SetForegroundColour(*wxLIGHT_GREY);
105
106 m_entry->DoSetValue(m_hint, wxTextEntryBase::SetValue_NoEvent);
107 }
108
109 // Restore the original text colour if we had changed it to show the hint
110 // and not restored it yet.
111 void RestoreTextColourIfNecessary()
112 {
113 if ( m_colFg.IsOk() )
114 {
115 m_win->SetForegroundColour(m_colFg);
116 m_colFg = wxColour();
117 }
118 }
119
120 void OnSetFocus(wxFocusEvent& event)
121 {
122 // If we had been showing the hint before, remove it now and restore
123 // the normal colour.
124 if ( m_text.empty() )
125 {
126 RestoreTextColourIfNecessary();
127
128 m_entry->DoSetValue(wxString(), wxTextEntryBase::SetValue_NoEvent);
129 }
130
131 event.Skip();
132 }
133
134 void OnKillFocus(wxFocusEvent& event)
135 {
136 // Restore the hint if the user didn't enter anything.
137 ShowHintIfAppropriate();
138
139 event.Skip();
140 }
141
142 void OnTextChanged(wxCommandEvent& event)
143 {
144 // Update the stored window text.
145 //
146 // Notice that we can't use GetValue() nor wxCommandEvent::GetString()
147 // which uses it internally because this would just forward back to us
148 // so go directly to the private method which returns the real control
149 // contents.
150 HandleTextUpdate(m_entry->DoGetValue());
151
152 event.Skip();
153 }
154
155
156 // the text control we're associated with (as its interface and its window)
157 wxTextEntryBase * const m_entry;
158 wxWindow * const m_win;
159
160 // the original foreground colour of m_win before we changed it
161 wxColour m_colFg;
162
163 // The hint passed to wxTextEntry::SetHint(), never empty.
164 wxString m_hint;
165
166 // The real text of the window.
167 wxString m_text;
168
169
170 wxDECLARE_NO_COPY_CLASS(wxTextEntryHintData);
171 };
172
173 // ============================================================================
174 // wxTextEntryBase implementation
175 // ============================================================================
176
177 wxTextEntryBase::~wxTextEntryBase()
178 {
179 delete m_hintData;
180 }
181
182 // ----------------------------------------------------------------------------
183 // text accessors
184 // ----------------------------------------------------------------------------
185
186 wxString wxTextEntryBase::GetValue() const
187 {
188 return m_hintData ? m_hintData->GetText() : DoGetValue();
189 }
190
191 wxString wxTextEntryBase::GetRange(long from, long to) const
192 {
193 wxString sel;
194 wxString value = GetValue();
195
196 if ( from < to && (long)value.length() >= to )
197 {
198 sel = value.substr(from, to - from);
199 }
200
201 return sel;
202 }
203
204 // ----------------------------------------------------------------------------
205 // text operations
206 // ----------------------------------------------------------------------------
207
208 void wxTextEntryBase::ChangeValue(const wxString& value)
209 {
210 DoSetValue(value, SetValue_NoEvent);
211
212 // As we didn't generate any events for wxTextEntryHintData to catch,
213 // notify it explicitly about our changed contents.
214 if ( m_hintData )
215 m_hintData->HandleTextUpdate(value);
216 }
217
218 void wxTextEntryBase::AppendText(const wxString& text)
219 {
220 SetInsertionPointEnd();
221 WriteText(text);
222 }
223
224 void wxTextEntryBase::DoSetValue(const wxString& value, int flags)
225 {
226 EventsSuppressor noeventsIf(this, !(flags & SetValue_SendEvent));
227
228 SelectAll();
229 WriteText(value);
230
231 SetInsertionPoint(0);
232 }
233
234 void wxTextEntryBase::Replace(long from, long to, const wxString& value)
235 {
236 {
237 EventsSuppressor noevents(this);
238 Remove(from, to);
239 }
240
241 SetInsertionPoint(from);
242 WriteText(value);
243 }
244
245 // ----------------------------------------------------------------------------
246 // selection
247 // ----------------------------------------------------------------------------
248
249 bool wxTextEntryBase::HasSelection() const
250 {
251 long from, to;
252 GetSelection(&from, &to);
253
254 return from < to;
255 }
256
257 void wxTextEntryBase::RemoveSelection()
258 {
259 long from, to;
260 GetSelection(& from, & to);
261 if (from != -1 && to != -1)
262 Remove(from, to);
263 }
264
265 wxString wxTextEntryBase::GetStringSelection() const
266 {
267 long from, to;
268 GetSelection(&from, &to);
269
270 return GetRange(from, to);
271 }
272
273 // ----------------------------------------------------------------------------
274 // clipboard
275 // ----------------------------------------------------------------------------
276
277 bool wxTextEntryBase::CanCopy() const
278 {
279 return HasSelection();
280 }
281
282 bool wxTextEntryBase::CanCut() const
283 {
284 return CanCopy() && IsEditable();
285 }
286
287 bool wxTextEntryBase::CanPaste() const
288 {
289 if ( IsEditable() )
290 {
291 #if wxUSE_CLIPBOARD
292 // check if there is any text on the clipboard
293 if ( wxTheClipboard->IsSupported(wxDF_TEXT)
294 #if wxUSE_UNICODE
295 || wxTheClipboard->IsSupported(wxDF_UNICODETEXT)
296 #endif // wxUSE_UNICODE
297 )
298 {
299 return true;
300 }
301 #endif // wxUSE_CLIPBOARD
302 }
303
304 return false;
305 }
306
307 // ----------------------------------------------------------------------------
308 // hints support
309 // ----------------------------------------------------------------------------
310
311 bool wxTextEntryBase::SetHint(const wxString& hint)
312 {
313 if ( !hint.empty() )
314 {
315 if ( !m_hintData )
316 m_hintData = new wxTextEntryHintData(this, GetEditableWindow());
317
318 m_hintData->SetHintString(hint);
319 }
320 else if ( m_hintData )
321 {
322 // Setting empty hint removes any currently set one.
323 delete m_hintData;
324 m_hintData = NULL;
325 }
326 //else: Setting empty hint when we don't have any doesn't do anything.
327
328 return true;
329 }
330
331 wxString wxTextEntryBase::GetHint() const
332 {
333 return m_hintData ? m_hintData->GetHintString() : wxString();
334 }
335
336 // ----------------------------------------------------------------------------
337 // margins support
338 // ----------------------------------------------------------------------------
339
340 bool wxTextEntryBase::DoSetMargins(const wxPoint& WXUNUSED(pt))
341 {
342 return false;
343 }
344
345 wxPoint wxTextEntryBase::DoGetMargins() const
346 {
347 return wxPoint(-1, -1);
348 }
349
350 // ----------------------------------------------------------------------------
351 // events
352 // ----------------------------------------------------------------------------
353
354 /* static */
355 bool wxTextEntryBase::SendTextUpdatedEvent(wxWindow *win)
356 {
357 wxCHECK_MSG( win, false, "can't send an event without a window" );
358
359 wxCommandEvent event(wxEVT_COMMAND_TEXT_UPDATED, win->GetId());
360
361 // do not do this as it could be very inefficient if the text control
362 // contains a lot of text and we're not using ref-counted wxString
363 // implementation -- instead, event.GetString() will query the control for
364 // its current text if needed
365 //event.SetString(win->GetValue());
366
367 event.SetEventObject(win);
368 return win->HandleWindowEvent(event);
369 }
370
371 #endif // wxUSE_TEXTCTRL || wxUSE_COMBOBOX