]>
Commit | Line | Data |
---|---|---|
c801d85f KB |
1 | ///////////////////////////////////////////////////////////////////////////// |
2 | // Name: textctrl.cpp | |
3 | // Purpose: | |
4 | // Author: Robert Roebling | |
5 | // Created: 01/02/97 | |
6 | // Id: | |
7 | // Copyright: (c) 1998 Robert Roebling, Julian Smart and Markus Holzem | |
13289f04 | 8 | // Licence: wxWindows licence |
c801d85f KB |
9 | ///////////////////////////////////////////////////////////////////////////// |
10 | ||
11 | #ifdef __GNUG__ | |
12 | #pragma implementation "textctrl.h" | |
13 | #endif | |
14 | ||
15 | #include "wx/textctrl.h" | |
16 | #include "wx/utils.h" | |
1a5a8367 | 17 | #include <wx/intl.h> |
c801d85f KB |
18 | |
19 | //----------------------------------------------------------------------------- | |
2f2aa628 | 20 | // "changed" |
c801d85f KB |
21 | //----------------------------------------------------------------------------- |
22 | ||
6de97a3b | 23 | static void gtk_text_changed_callback( GtkWidget *WXUNUSED(widget), wxTextCtrl *win ) |
484e45bf | 24 | { |
9406d962 | 25 | win->SetModified(); |
903f689b RR |
26 | |
27 | wxCommandEvent event( wxEVT_COMMAND_TEXT_UPDATED, win->m_windowId ); | |
28 | wxString val( win->GetValue() ); | |
29 | if (!val.IsNull()) event.m_commandString = WXSTRINGCAST val; | |
30 | event.SetEventObject( win ); | |
31 | win->GetEventHandler()->ProcessEvent( event ); | |
6de97a3b | 32 | } |
112892b9 | 33 | |
2f2aa628 RR |
34 | //----------------------------------------------------------------------------- |
35 | // wxTextCtrl | |
36 | //----------------------------------------------------------------------------- | |
37 | ||
38 | IMPLEMENT_DYNAMIC_CLASS(wxTextCtrl,wxControl) | |
39 | ||
c801d85f | 40 | BEGIN_EVENT_TABLE(wxTextCtrl, wxControl) |
903f689b | 41 | EVT_CHAR(wxTextCtrl::OnChar) |
c801d85f KB |
42 | END_EVENT_TABLE() |
43 | ||
44 | wxTextCtrl::wxTextCtrl(void) : streambuf() | |
45 | { | |
6de97a3b | 46 | if (allocate()) setp(base(),ebuf()); |
13289f04 | 47 | |
112892b9 | 48 | m_modified = FALSE; |
6de97a3b | 49 | } |
c801d85f | 50 | |
debe6624 | 51 | wxTextCtrl::wxTextCtrl( wxWindow *parent, wxWindowID id, const wxString &value, |
484e45bf | 52 | const wxPoint &pos, const wxSize &size, |
6de97a3b | 53 | int style, const wxValidator& validator, const wxString &name ) : streambuf() |
c801d85f | 54 | { |
6de97a3b | 55 | if (allocate()) setp(base(),ebuf()); |
13289f04 | 56 | |
112892b9 | 57 | m_modified = FALSE; |
6de97a3b RR |
58 | Create( parent, id, value, pos, size, style, validator, name ); |
59 | } | |
c801d85f | 60 | |
debe6624 | 61 | bool wxTextCtrl::Create( wxWindow *parent, wxWindowID id, const wxString &value, |
484e45bf | 62 | const wxPoint &pos, const wxSize &size, |
6de97a3b | 63 | int style, const wxValidator& validator, const wxString &name ) |
c801d85f KB |
64 | { |
65 | m_needParent = TRUE; | |
484e45bf | 66 | |
c801d85f | 67 | PreCreation( parent, id, pos, size, style, name ); |
484e45bf | 68 | |
6de97a3b RR |
69 | SetValidator( validator ); |
70 | ||
13289f04 | 71 | bool bMultiLine = (style & wxTE_MULTILINE) != 0; |
5796ed40 | 72 | if ( bMultiLine ) |
47908e25 | 73 | { |
13289f04 VZ |
74 | // a multi-line edit control: create a vertical scrollbar by default and |
75 | // horizontal if requested | |
76 | bool bHasHScrollbar = (style & wxHSCROLL) != 0; | |
77 | ||
78 | // create our control... | |
c67daf87 | 79 | m_text = gtk_text_new( (GtkAdjustment *) NULL, (GtkAdjustment *) NULL ); |
13289f04 VZ |
80 | |
81 | // ... and put into the upper left hand corner of the table | |
82 | m_widget = gtk_table_new(bHasHScrollbar ? 2 : 1, 2, FALSE); | |
83 | gtk_table_attach(GTK_TABLE(m_widget), m_text, 0, 1, 0, 1, | |
84 | GTK_FILL | GTK_EXPAND, | |
85 | GTK_FILL | GTK_EXPAND | GTK_SHRINK, | |
86 | 0, 0); | |
87 | ||
88 | // put the horizontal scrollbar in the lower left hand corner | |
903f689b RR |
89 | if (bHasHScrollbar) |
90 | { | |
13289f04 VZ |
91 | GtkWidget *hscrollbar = gtk_hscrollbar_new(GTK_TEXT(m_text)->hadj); |
92 | gtk_table_attach(GTK_TABLE(m_widget), hscrollbar, 0, 1, 1, 2, | |
93 | GTK_EXPAND | GTK_FILL, | |
94 | GTK_FILL, | |
95 | 0, 0); | |
96 | gtk_widget_show(hscrollbar); | |
97 | } | |
98 | ||
99 | // finally, put the vertical scrollbar in the upper right corner | |
100 | GtkWidget *vscrollbar = gtk_vscrollbar_new(GTK_TEXT(m_text)->vadj); | |
101 | gtk_table_attach(GTK_TABLE(m_widget), vscrollbar, 1, 2, 0, 1, | |
102 | GTK_FILL, | |
103 | GTK_EXPAND | GTK_FILL | GTK_SHRINK, | |
104 | 0, 0); | |
903f689b | 105 | gtk_widget_show( vscrollbar ); |
13289f04 | 106 | } |
903f689b RR |
107 | else |
108 | { | |
13289f04 VZ |
109 | // a single-line text control: no need for scrollbars |
110 | m_widget = | |
111 | m_text = gtk_entry_new(); | |
112 | } | |
484e45bf | 113 | |
c801d85f KB |
114 | wxSize newSize = size; |
115 | if (newSize.x == -1) newSize.x = 80; | |
116 | if (newSize.y == -1) newSize.y = 26; | |
117 | SetSize( newSize.x, newSize.y ); | |
484e45bf | 118 | |
c801d85f | 119 | PostCreation(); |
484e45bf | 120 | |
903f689b RR |
121 | if (bMultiLine) |
122 | { | |
13289f04 VZ |
123 | gtk_widget_realize(m_text); |
124 | gtk_widget_show(m_text); | |
125 | } | |
126 | ||
484e45bf | 127 | // we want to be notified about text changes |
13289f04 | 128 | gtk_signal_connect(GTK_OBJECT(m_text), "changed", |
484e45bf VZ |
129 | GTK_SIGNAL_FUNC(gtk_text_changed_callback), |
130 | (gpointer)this); | |
131 | ||
7f4dc78d RR |
132 | if (!value.IsNull()) |
133 | { | |
134 | gint tmp = 0; | |
13289f04 | 135 | gtk_editable_insert_text( GTK_EDITABLE(m_text), value, value.Length(), &tmp ); |
6de97a3b | 136 | } |
484e45bf | 137 | |
5796ed40 | 138 | if (style & wxTE_READONLY) |
112892b9 RR |
139 | { |
140 | } | |
141 | else | |
142 | { | |
903f689b | 143 | if (bMultiLine) |
13289f04 | 144 | gtk_text_set_editable( GTK_TEXT(m_text), 1 ); |
6de97a3b | 145 | } |
484e45bf | 146 | |
c801d85f | 147 | Show( TRUE ); |
484e45bf | 148 | |
c801d85f | 149 | return TRUE; |
6de97a3b | 150 | } |
c801d85f KB |
151 | |
152 | wxString wxTextCtrl::GetValue(void) const | |
153 | { | |
154 | wxString tmp; | |
155 | if (m_windowStyle & wxTE_MULTILINE) | |
156 | { | |
13289f04 VZ |
157 | gint len = gtk_text_get_length( GTK_TEXT(m_text) ); |
158 | tmp = gtk_editable_get_chars( GTK_EDITABLE(m_text), 0, len ); | |
c801d85f KB |
159 | } |
160 | else | |
161 | { | |
13289f04 | 162 | tmp = gtk_entry_get_text( GTK_ENTRY(m_text) ); |
6de97a3b | 163 | } |
c801d85f | 164 | return tmp; |
6de97a3b | 165 | } |
c801d85f KB |
166 | |
167 | void wxTextCtrl::SetValue( const wxString &value ) | |
168 | { | |
169 | wxString tmp = ""; | |
170 | if (!value.IsNull()) tmp = value; | |
171 | if (m_windowStyle & wxTE_MULTILINE) | |
172 | { | |
13289f04 VZ |
173 | gint len = gtk_text_get_length( GTK_TEXT(m_text) ); |
174 | gtk_editable_delete_text( GTK_EDITABLE(m_text), 0, len ); | |
c801d85f | 175 | len = 0; |
13289f04 | 176 | gtk_editable_insert_text( GTK_EDITABLE(m_text), tmp, tmp.Length(), &len ); |
c801d85f KB |
177 | } |
178 | else | |
179 | { | |
13289f04 | 180 | gtk_entry_set_text( GTK_ENTRY(m_text), tmp ); |
6de97a3b RR |
181 | } |
182 | } | |
c801d85f KB |
183 | |
184 | void wxTextCtrl::WriteText( const wxString &text ) | |
185 | { | |
186 | if (text.IsNull()) return; | |
484e45bf | 187 | |
c801d85f KB |
188 | if (m_windowStyle & wxTE_MULTILINE) |
189 | { | |
13289f04 VZ |
190 | gint len = gtk_text_get_length( GTK_TEXT(m_text) ); |
191 | gtk_editable_insert_text( GTK_EDITABLE(m_text), text, text.Length(), &len ); | |
c801d85f KB |
192 | } |
193 | else | |
194 | { | |
13289f04 | 195 | gtk_entry_append_text( GTK_ENTRY(m_text), text ); |
6de97a3b RR |
196 | } |
197 | } | |
c801d85f | 198 | |
112892b9 | 199 | bool wxTextCtrl::LoadFile( const wxString &WXUNUSED(file) ) |
c801d85f | 200 | { |
6de97a3b | 201 | wxFAIL_MSG( "wxTextCtrl::LoadFile not implemented" ); |
2ad3a34e | 202 | |
112892b9 | 203 | return FALSE; |
6de97a3b | 204 | } |
c801d85f | 205 | |
112892b9 | 206 | bool wxTextCtrl::SaveFile( const wxString &WXUNUSED(file) ) |
c801d85f | 207 | { |
6de97a3b | 208 | wxFAIL_MSG( "wxTextCtrl::SaveFile not implemented" ); |
2ad3a34e | 209 | |
112892b9 | 210 | return FALSE; |
6de97a3b | 211 | } |
c801d85f | 212 | |
112892b9 | 213 | /* |
debe6624 | 214 | wxString wxTextCtrl::GetLineText( long lineNo ) const |
c801d85f | 215 | { |
6de97a3b | 216 | } |
c801d85f | 217 | |
112892b9 | 218 | |
c801d85f KB |
219 | void wxTextCtrl::OnDropFiles( wxDropFilesEvent &event ) |
220 | { | |
6de97a3b | 221 | } |
c801d85f | 222 | |
debe6624 | 223 | long wxTextCtrl::PositionToXY( long pos, long *x, long *y ) const |
c801d85f | 224 | { |
6de97a3b | 225 | } |
c801d85f | 226 | |
debe6624 | 227 | long wxTextCtrl::XYToPosition( long x, long y ) |
c801d85f | 228 | { |
6de97a3b | 229 | } |
c801d85f KB |
230 | |
231 | int wxTextCtrl::GetNumberOfLines(void) | |
232 | { | |
6de97a3b | 233 | } |
c801d85f KB |
234 | |
235 | */ | |
debe6624 | 236 | void wxTextCtrl::SetInsertionPoint( long pos ) |
c801d85f KB |
237 | { |
238 | int tmp = (int) pos; | |
239 | if (m_windowStyle & wxTE_MULTILINE) | |
13289f04 | 240 | gtk_text_set_point( GTK_TEXT(m_text), tmp ); |
c801d85f | 241 | else |
13289f04 | 242 | gtk_entry_set_position( GTK_ENTRY(m_text), tmp ); |
6de97a3b | 243 | } |
c801d85f KB |
244 | |
245 | void wxTextCtrl::SetInsertionPointEnd(void) | |
246 | { | |
247 | int pos = 0; | |
248 | if (m_windowStyle & wxTE_MULTILINE) | |
13289f04 | 249 | pos = gtk_text_get_length( GTK_TEXT(m_text) ); |
c801d85f | 250 | else |
13289f04 | 251 | pos = GTK_ENTRY(m_text)->text_length; |
c801d85f | 252 | SetInsertionPoint( pos-1 ); |
6de97a3b | 253 | } |
c801d85f | 254 | |
debe6624 | 255 | void wxTextCtrl::SetEditable( bool editable ) |
c801d85f KB |
256 | { |
257 | if (m_windowStyle & wxTE_MULTILINE) | |
13289f04 | 258 | gtk_text_set_editable( GTK_TEXT(m_text), editable ); |
c801d85f | 259 | else |
13289f04 | 260 | gtk_entry_set_editable( GTK_ENTRY(m_text), editable ); |
6de97a3b | 261 | } |
c801d85f | 262 | |
debe6624 | 263 | void wxTextCtrl::SetSelection( long from, long to ) |
c801d85f | 264 | { |
13289f04 | 265 | gtk_editable_select_region( GTK_EDITABLE(m_text), (gint)from, (gint)to ); |
6de97a3b | 266 | } |
c801d85f | 267 | |
debe6624 | 268 | void wxTextCtrl::ShowPosition( long WXUNUSED(pos) ) |
c801d85f | 269 | { |
1a5a8367 | 270 | wxFAIL_MSG(_("wxTextCtrl::ShowPosition not implemented")); |
6de97a3b | 271 | } |
c801d85f KB |
272 | |
273 | long wxTextCtrl::GetInsertionPoint(void) const | |
274 | { | |
13289f04 | 275 | return (long) GTK_EDITABLE(m_text)->current_pos; |
6de97a3b | 276 | } |
c801d85f KB |
277 | |
278 | long wxTextCtrl::GetLastPosition(void) const | |
279 | { | |
280 | int pos = 0; | |
281 | if (m_windowStyle & wxTE_MULTILINE) | |
13289f04 | 282 | pos = gtk_text_get_length( GTK_TEXT(m_text) ); |
c801d85f | 283 | else |
13289f04 | 284 | pos = GTK_ENTRY(m_text)->text_length; |
c801d85f | 285 | return (long)pos-1; |
6de97a3b | 286 | } |
c801d85f | 287 | |
debe6624 | 288 | void wxTextCtrl::Remove( long from, long to ) |
c801d85f | 289 | { |
13289f04 | 290 | gtk_editable_delete_text( GTK_EDITABLE(m_text), (gint)from, (gint)to ); |
6de97a3b | 291 | } |
c801d85f | 292 | |
debe6624 | 293 | void wxTextCtrl::Replace( long from, long to, const wxString &value ) |
c801d85f | 294 | { |
13289f04 | 295 | gtk_editable_delete_text( GTK_EDITABLE(m_text), (gint)from, (gint)to ); |
c801d85f KB |
296 | if (value.IsNull()) return; |
297 | gint pos = (gint)to; | |
13289f04 | 298 | gtk_editable_insert_text( GTK_EDITABLE(m_text), value, value.Length(), &pos ); |
6de97a3b | 299 | } |
c801d85f KB |
300 | |
301 | void wxTextCtrl::Cut(void) | |
302 | { | |
13289f04 | 303 | gtk_editable_cut_clipboard( GTK_EDITABLE(m_text), 0 ); |
6de97a3b | 304 | } |
c801d85f KB |
305 | |
306 | void wxTextCtrl::Copy(void) | |
307 | { | |
13289f04 | 308 | gtk_editable_copy_clipboard( GTK_EDITABLE(m_text), 0 ); |
6de97a3b | 309 | } |
c801d85f KB |
310 | |
311 | void wxTextCtrl::Paste(void) | |
312 | { | |
13289f04 | 313 | gtk_editable_paste_clipboard( GTK_EDITABLE(m_text), 0 ); |
6de97a3b | 314 | } |
c801d85f KB |
315 | |
316 | void wxTextCtrl::Delete(void) | |
317 | { | |
318 | SetValue( "" ); | |
6de97a3b | 319 | } |
c801d85f | 320 | |
903f689b | 321 | void wxTextCtrl::OnChar( wxKeyEvent &key_event ) |
c801d85f | 322 | { |
903f689b RR |
323 | if ((key_event.KeyCode() == WXK_RETURN) && (m_windowStyle & wxPROCESS_ENTER)) |
324 | { | |
325 | wxCommandEvent event(wxEVT_COMMAND_TEXT_ENTER, m_windowId); | |
326 | event.SetEventObject(this); | |
903f689b RR |
327 | if (GetEventHandler()->ProcessEvent(event)) return; |
328 | } | |
329 | else if (key_event.KeyCode() == WXK_TAB) | |
330 | { | |
331 | wxNavigationKeyEvent event; | |
332 | event.SetDirection( key_event.m_shiftDown ); | |
333 | event.SetWindowChange(FALSE); | |
334 | event.SetEventObject(this); | |
335 | ||
336 | if (GetEventHandler()->ProcessEvent(event)) return; | |
337 | } | |
338 | key_event.Skip(); | |
6de97a3b | 339 | } |
c801d85f | 340 | |
46dc76ba | 341 | int wxTextCtrl::overflow( int WXUNUSED(c) ) |
c801d85f | 342 | { |
c801d85f KB |
343 | int len = pptr() - pbase(); |
344 | char *txt = new char[len+1]; | |
345 | strncpy(txt, pbase(), len); | |
346 | txt[len] = '\0'; | |
347 | (*this) << txt; | |
348 | setp(pbase(), epptr()); | |
349 | delete[] txt; | |
350 | return EOF; | |
6de97a3b | 351 | } |
c801d85f KB |
352 | |
353 | int wxTextCtrl::sync(void) | |
354 | { | |
c801d85f KB |
355 | int len = pptr() - pbase(); |
356 | char *txt = new char[len+1]; | |
357 | strncpy(txt, pbase(), len); | |
358 | txt[len] = '\0'; | |
359 | (*this) << txt; | |
360 | setp(pbase(), epptr()); | |
361 | delete[] txt; | |
362 | return 0; | |
6de97a3b | 363 | } |
c801d85f KB |
364 | |
365 | int wxTextCtrl::underflow(void) | |
366 | { | |
367 | return EOF; | |
6de97a3b | 368 | } |
c801d85f KB |
369 | |
370 | wxTextCtrl& wxTextCtrl::operator<<(const wxString& s) | |
371 | { | |
372 | WriteText(s); | |
373 | return *this; | |
374 | } | |
375 | ||
debe6624 | 376 | wxTextCtrl& wxTextCtrl::operator<<(float f) |
c801d85f KB |
377 | { |
378 | static char buf[100]; | |
379 | sprintf(buf, "%.2f", f); | |
380 | WriteText(buf); | |
381 | return *this; | |
382 | } | |
383 | ||
debe6624 | 384 | wxTextCtrl& wxTextCtrl::operator<<(double d) |
c801d85f KB |
385 | { |
386 | static char buf[100]; | |
387 | sprintf(buf, "%.2f", d); | |
388 | WriteText(buf); | |
389 | return *this; | |
390 | } | |
391 | ||
debe6624 | 392 | wxTextCtrl& wxTextCtrl::operator<<(int i) |
c801d85f KB |
393 | { |
394 | static char buf[100]; | |
395 | sprintf(buf, "%i", i); | |
396 | WriteText(buf); | |
397 | return *this; | |
398 | } | |
399 | ||
debe6624 | 400 | wxTextCtrl& wxTextCtrl::operator<<(long i) |
c801d85f KB |
401 | { |
402 | static char buf[100]; | |
403 | sprintf(buf, "%ld", i); | |
404 | WriteText(buf); | |
405 | return *this; | |
406 | } | |
407 | ||
408 | wxTextCtrl& wxTextCtrl::operator<<(const char c) | |
409 | { | |
410 | char buf[2]; | |
411 | ||
412 | buf[0] = c; | |
413 | buf[1] = 0; | |
414 | WriteText(buf); | |
415 | return *this; | |
416 | } | |
417 | ||
30dea054 | 418 | GtkWidget* wxTextCtrl::GetConnectWidget(void) |
e3e65dac RR |
419 | { |
420 | return GTK_WIDGET(m_text); | |
6de97a3b | 421 | } |
e3e65dac | 422 | |
903f689b RR |
423 | bool wxTextCtrl::IsOwnGtkWindow( GdkWindow *window ) |
424 | { | |
425 | if (m_windowStyle & wxTE_MULTILINE) | |
426 | return (window == GTK_TEXT(m_text)->text_area); | |
427 | else | |
428 | return (window == GTK_ENTRY(m_text)->text_area); | |
429 | } | |
e3e65dac | 430 | |
868a2826 RR |
431 | void wxTextCtrl::SetFont( const wxFont &font ) |
432 | { | |
3f659fd6 RR |
433 | if (((wxFont*)&font)->Ok()) |
434 | m_font = font; | |
435 | else | |
436 | m_font = *wxSWISS_FONT; | |
868a2826 RR |
437 | |
438 | GtkStyle *style = (GtkStyle*) NULL; | |
439 | if (!m_hasOwnStyle) | |
440 | { | |
441 | m_hasOwnStyle = TRUE; | |
442 | style = gtk_style_copy( gtk_widget_get_style( m_text ) ); | |
443 | } | |
444 | else | |
445 | { | |
446 | style = gtk_widget_get_style( m_text ); | |
447 | } | |
448 | ||
449 | gdk_font_unref( style->font ); | |
450 | style->font = gdk_font_ref( m_font.GetInternalFont( 1.0 ) ); | |
451 | ||
452 | gtk_widget_set_style( m_text, style ); | |
453 | } | |
e3e65dac | 454 | |
5796ed40 | 455 |