1 ///////////////////////////////////////////////////////////////////////////// 
   4 // Author:      Julian Smart 
   8 // Copyright:   (c) Julian Smart and Markus Holzem 
   9 // Licence:     wxWindows license 
  10 ///////////////////////////////////////////////////////////////////////////// 
  13 #pragma implementation "textctrl.h" 
  16 // For compilers that support precompilation, includes "wx.h". 
  17 #include "wx/wxprec.h" 
  24 #include "wx/textctrl.h" 
  25 #include "wx/settings.h" 
  33 #include "wx/clipbrd.h" 
  36 #include "wx/msw/private.h" 
  50 #include <sys/types.h> 
  56 #if defined(__BORLANDC__) && !defined(__WIN32__) 
  59 #if !defined(__GNUWIN32__) && !defined(__SALFORDC__) 
  62 #define farmalloc malloc 
  69 #if wxUSE_RICHEDIT && !defined(__GNUWIN32__) 
  73 #if !USE_SHARED_LIBRARY 
  74 IMPLEMENT_DYNAMIC_CLASS(wxTextCtrl
, wxControl
) 
  76 BEGIN_EVENT_TABLE(wxTextCtrl
, wxControl
) 
  77   EVT_CHAR(wxTextCtrl::OnChar
) 
  78   EVT_DROP_FILES(wxTextCtrl::OnDropFiles
) 
  79   EVT_ERASE_BACKGROUND(wxTextCtrl::OnEraseBackground
) 
  81 #endif // USE_SHARED_LIBRARY 
  84 wxTextCtrl::wxTextCtrl() 
  85 #ifndef NO_TEXT_WINDOW_STREAM 
  94 bool wxTextCtrl::Create(wxWindow 
*parent
, wxWindowID id
, 
  95                         const wxString
& value
, 
  97                         const wxSize
& size
, long style
, 
  98                         const wxValidator
& validator
, 
 103   SetValidator(validator
); 
 104   if (parent
) parent
->AddChild(this); 
 106   m_windowStyle 
= style
; 
 108   // Should this be taken from the system colours? 
 109 //  SetBackgroundColour(wxColour(255, 255, 255)); 
 111   SetBackgroundColour(wxSystemSettings::GetSystemColour(wxSYS_COLOUR_WINDOW
)); 
 113   SetForegroundColour(parent
->GetForegroundColour()) ; 
 116     m_windowId 
= (int)NewControlId(); 
 125   long msStyle 
= ES_LEFT 
| WS_VISIBLE 
| WS_CHILD 
| WS_TABSTOP
; 
 126   if (m_windowStyle 
& wxTE_MULTILINE
) 
 128     msStyle 
|= ES_MULTILINE 
| ES_WANTRETURN 
| WS_VSCROLL 
; // WS_BORDER 
 129     m_windowStyle 
|= wxTE_PROCESS_ENTER
; 
 132     msStyle 
|= ES_AUTOHSCROLL 
; 
 134   if (m_windowStyle 
& wxTE_READONLY
) 
 135     msStyle 
|= ES_READONLY
; 
 137   if (m_windowStyle 
& wxHSCROLL
) 
 138     msStyle 
|= (WS_HSCROLL 
| ES_AUTOHSCROLL
) ; 
 139   if (m_windowStyle 
& wxTE_PASSWORD
) // hidden input 
 140     msStyle 
|= ES_PASSWORD
; 
 142   const char *windowClass 
= "EDIT"; 
 145   if ( m_windowStyle 
& wxTE_MULTILINE 
) 
 147     msStyle 
|= ES_AUTOVSCROLL
; 
 149     windowClass 
= "RichEdit" ; 
 156   WXDWORD exStyle 
= Determine3DEffects(WS_EX_CLIENTEDGE
, &want3D
) ; 
 158   // If we're in Win95, and we want a simple 2D border, 
 159   // then make it an EDIT control instead. 
 161   if (m_windowStyle 
& wxSIMPLE_BORDER
) 
 163     windowClass 
= "EDIT"; 
 168   // Even with extended styles, need to combine with WS_BORDER 
 169   // for them to look right. 
 170   if ( want3D 
|| wxStyleHasBorder(m_windowStyle
) ) 
 171     msStyle 
|= WS_BORDER
; 
 173   m_hWnd 
= (WXHWND
)::CreateWindowEx(exStyle
, windowClass
, NULL
, 
 175                         0, 0, 0, 0, (HWND
) ((wxWindow
*)parent
)->GetHWND(), (HMENU
)m_windowId
, 
 176                         wxGetInstance(), NULL
); 
 178   wxCHECK_MSG( m_hWnd
, FALSE
, "Failed to create text ctrl" ); 
 183     Ctl3dSubclassCtl((HWND
)m_hWnd
); 
 191     // Have to enable events 
 192     ::SendMessage((HWND
)m_hWnd
, EM_SETEVENTMASK
, 0, 
 193                   ENM_CHANGE 
| ENM_DROPFILES 
| ENM_SELCHANGE 
| ENM_UPDATE
); 
 197   SubclassWin(GetHWND()); 
 199   if ( parent
->GetFont().Ok() && parent
->GetFont().Ok() ) 
 201     SetFont(parent
->GetFont()); 
 205     SetFont(wxSystemSettings::GetSystemFont(wxSYS_SYSTEM_FONT
)); 
 208   SetSize(x
, y
, width
, height
); 
 210   // Causes a crash for Symantec C++ and WIN32 for some reason 
 211 #if !(defined(__SC__) && defined(__WIN32__)) 
 212   if ( !value
.IsEmpty() ) 
 221 // Make sure the window style (etc.) reflects the HWND style (roughly) 
 222 void wxTextCtrl::AdoptAttributesFromHWND() 
 224   wxWindow::AdoptAttributesFromHWND(); 
 226   HWND hWnd 
= (HWND
) GetHWND(); 
 227   long style 
= GetWindowLong((HWND
) hWnd
, GWL_STYLE
); 
 229   // retrieve the style to see whether this is an edit or richedit ctrl 
 234   GetClassName((HWND
) hWnd
, buf
, 256); 
 237   GetClassNameW((HWND
) hWnd
, buf
, 256); 
 240   GetClassName((HWND
) hWnd
, buf
, 256); 
 242   GetClassNameA((HWND
) hWnd
, buf
, 256); 
 256   if (style 
& ES_MULTILINE
) 
 257     m_windowStyle 
|= wxTE_MULTILINE
; 
 258   if (style 
& ES_PASSWORD
) 
 259     m_windowStyle 
|= wxTE_PASSWORD
; 
 260   if (style 
& ES_READONLY
) 
 261     m_windowStyle 
|= wxTE_READONLY
; 
 262   if (style 
& ES_WANTRETURN
) 
 263     m_windowStyle 
|= wxTE_PROCESS_ENTER
; 
 266 void wxTextCtrl::SetupColours() 
 268   SetBackgroundColour(wxSystemSettings::GetSystemColour(wxSYS_COLOUR_WINDOW
)); 
 269   SetForegroundColour(GetParent()->GetForegroundColour()); 
 272 wxString 
wxTextCtrl::GetValue() const 
 274     int length 
= GetWindowTextLength((HWND
) GetHWND()); 
 275     char *s 
= new char[length
+1]; 
 276     GetWindowText((HWND
) GetHWND(), s
, length
+1); 
 282 void wxTextCtrl::SetValue(const wxString
& value
) 
 284   // If newlines are denoted by just 10, must stick 13 in front. 
 286   int len 
= value
.Length(); 
 288   for (i 
= 0; i 
< len
; i 
++) 
 290     if ((i 
> 0) && (value
[i
] == 10) && (value
[i
-1] != 13)) 
 295     char *tmp 
= new char[len 
+ singletons 
+ 1]; 
 297     for (i 
= 0; i 
< len
; i 
++) 
 299       if ((i 
> 0) && (value
[i
] == 10) && (value
[i
-1] != 13)) 
 308     SetWindowText((HWND
) GetHWND(), tmp
); 
 312     SetWindowText((HWND
) GetHWND(), (const char *)value
); 
 315 void wxTextCtrl::SetSize(int x
, int y
, int width
, int height
, int sizeFlags
) 
 317   int currentX
, currentY
; 
 318   GetPosition(¤tX
, ¤tY
); 
 324   if (x 
== -1 || (sizeFlags 
& wxSIZE_ALLOW_MINUS_ONE
)) 
 326   if (y 
== -1 || (sizeFlags 
& wxSIZE_ALLOW_MINUS_ONE
)) 
 329   AdjustForParentClientOrigin(x1
, y1
, sizeFlags
); 
 331   int cx
; // button font dimensions 
 334   wxGetCharSize(GetHWND(), &cx
, &cy
, & this->GetFont()); 
 336   int control_width
, control_height
, control_x
, control_y
; 
 338   // If we're prepared to use the existing size, then... 
 339   if (width 
== -1 && height 
== -1 && ((sizeFlags 
& wxSIZE_AUTO
) != wxSIZE_AUTO
)) 
 344   // Deal with default size (using -1 values) 
 346     w1 
= DEFAULT_ITEM_WIDTH
; 
 353   // Calculations may have made text size too small 
 354   if (control_height 
<= 0) 
 355     control_height 
= EDIT_HEIGHT_FROM_CHAR_HEIGHT(cy
); 
 357   if (control_width 
<= 0) 
 358     control_width 
= DEFAULT_ITEM_WIDTH
; 
 360   MoveWindow((HWND
) GetHWND(), (int)control_x
, (int)control_y
, 
 361                               (int)control_width
, (int)control_height
, TRUE
); 
 364 // Clipboard operations 
 365 void wxTextCtrl::Copy() 
 367   HWND hWnd 
= (HWND
) GetHWND(); 
 368   SendMessage(hWnd
, WM_COPY
, 0, 0L); 
 371 void wxTextCtrl::Cut() 
 373   HWND hWnd 
= (HWND
) GetHWND(); 
 374   SendMessage(hWnd
, WM_CUT
, 0, 0L); 
 377 void wxTextCtrl::Paste() 
 379   HWND hWnd 
= (HWND
) GetHWND(); 
 380   SendMessage(hWnd
, WM_PASTE
, 0, 0L); 
 383 void wxTextCtrl::SetEditable(bool editable
) 
 385   HWND hWnd 
= (HWND
) GetHWND(); 
 386   SendMessage(hWnd
, EM_SETREADONLY
, (WPARAM
)!editable
, (LPARAM
)0L); 
 389 void wxTextCtrl::SetInsertionPoint(long pos
) 
 391   HWND hWnd 
= (HWND
) GetHWND(); 
 399     SendMessage(hWnd
, EM_EXSETSEL
, 0, (LPARAM
) &range
); 
 400     SendMessage(hWnd
, EM_SCROLLCARET
, (WPARAM
)0, (LPARAM
)0); 
 405     SendMessage(hWnd
, EM_SETSEL
, pos
, pos
); 
 406     SendMessage(hWnd
, EM_SCROLLCARET
, (WPARAM
)0, (LPARAM
)0); 
 409   SendMessage(hWnd
, EM_SETSEL
, 0, MAKELPARAM(pos
, pos
)); 
 412   SendMessage(hWnd
, EM_REPLACESEL
, 0, (LPARAM
)nothing
); 
 415 void wxTextCtrl::SetInsertionPointEnd() 
 417   long pos 
= GetLastPosition(); 
 418   SetInsertionPoint(pos
); 
 421 long wxTextCtrl::GetInsertionPoint() const 
 429     SendMessage((HWND
) GetHWND(), EM_EXGETSEL
, 0, (LPARAM
) &range
); 
 434   DWORD Pos
=(DWORD
)SendMessage((HWND
) GetHWND(), EM_GETSEL
, 0, 0L); 
 438 long wxTextCtrl::GetLastPosition() const 
 440     HWND hWnd 
= (HWND
) GetHWND(); 
 442     // Will always return a number > 0 (according to docs) 
 443     int noLines 
= (int)SendMessage(hWnd
, EM_GETLINECOUNT
, (WPARAM
)0, (LPARAM
)0L); 
 445     // This gets the char index for the _beginning_ of the last line 
 446     int charIndex 
= (int)SendMessage(hWnd
, EM_LINEINDEX
, (WPARAM
)(noLines
-1), (LPARAM
)0L); 
 448     // Get number of characters in the last line. We'll add this to the character 
 449     // index for the last line, 1st position. 
 450     int lineLength 
= (int)SendMessage(hWnd
, EM_LINELENGTH
, (WPARAM
)charIndex
, (LPARAM
)0L); 
 452     return (long)(charIndex 
+ lineLength
); 
 455 void wxTextCtrl::Replace(long from
, long to
, const wxString
& value
) 
 457     HWND hWnd 
= (HWND
) GetHWND(); 
 458     long fromChar 
= from
; 
 461     // Set selection and remove it 
 463     SendMessage(hWnd
, EM_SETSEL
, fromChar
, toChar
); 
 465     SendMessage(hWnd
, EM_SETSEL
, (WPARAM
)0, (LPARAM
)MAKELONG(fromChar
, toChar
)); 
 467     SendMessage(hWnd
, WM_CUT
, (WPARAM
)0, (LPARAM
)0); 
 469     // Now replace with 'value', by pasting. 
 470     wxSetClipboardData(wxDF_TEXT
, (wxObject 
*) (const char *)value
, 0, 0); 
 472     // Paste into edit control 
 473     SendMessage(hWnd
, WM_PASTE
, (WPARAM
)0, (LPARAM
)0L); 
 476 void wxTextCtrl::Remove(long from
, long to
) 
 478     HWND hWnd 
= (HWND
) GetHWND(); 
 479     long fromChar 
= from
; 
 482     // Cut all selected text 
 484     SendMessage(hWnd
, EM_SETSEL
, fromChar
, toChar
); 
 486     SendMessage(hWnd
, EM_SETSEL
, (WPARAM
)0, (LPARAM
)MAKELONG(fromChar
, toChar
)); 
 488     SendMessage(hWnd
, WM_CUT
, (WPARAM
)0, (LPARAM
)0); 
 491 void wxTextCtrl::SetSelection(long from
, long to
) 
 493     HWND hWnd 
= (HWND
) GetHWND(); 
 494     long fromChar 
= from
; 
 496     // if from and to are both -1, it means 
 497     // (in wxWindows) that all text should be selected. 
 498     // This translates into Windows convention 
 499     if ((from 
== -1) && (to 
== -1)) 
 506     SendMessage(hWnd
, EM_SETSEL
, (WPARAM
)fromChar
, (LPARAM
)toChar
); 
 507     SendMessage(hWnd
, EM_SCROLLCARET
, (WPARAM
)0, (LPARAM
)0); 
 509     // WPARAM is 0: selection is scrolled into view 
 510     SendMessage(hWnd
, EM_SETSEL
, (WPARAM
)0, (LPARAM
)MAKELONG(fromChar
, toChar
)); 
 514 bool wxTextCtrl::LoadFile(const wxString
& file
) 
 516   if (!wxFileExists(WXSTRINGCAST file
)) 
 523 //  ifstream input(WXSTRINGCAST file, ios::nocreate | ios::in); 
 524   ifstream 
input(WXSTRINGCAST file
, ios::in
); 
 528       // Previously a SETSEL/REPLACESEL call-pair were done to insert 
 529       // line by line into the control. Apart from being very slow this 
 530       // was limited to 32K of text by the external interface presenting 
 531       // positions as signed shorts. Now load in one chunk... 
 532       // Note use of 'farmalloc' as in Borland 3.1 'size_t' is 16-bits... 
 535       struct _stat stat_buf
; 
 536       if (stat((char*) (const char*) file
, &stat_buf
) < 0) 
 539       struct stat stat_buf
; 
 540       if (stat(file
, &stat_buf
) < 0) 
 544 //      char *tmp_buffer = (char*)farmalloc(stat_buf.st_size+1); 
 545       // This may need to be a bigger buffer than the file size suggests, 
 546       // if it's a UNIX file. Give it an extra 1000 just in case. 
 547       char *tmp_buffer 
= (char*)farmalloc((size_t)(stat_buf
.st_size
+1+1000)); 
 550       while (!input
.eof() && input
.peek() != EOF
) 
 552         input
.getline(wxBuffer
, 500); 
 553   int len 
= strlen(wxBuffer
); 
 555   wxBuffer
[len
+1] = 10; 
 557   strcpy(tmp_buffer
+pos
, wxBuffer
); 
 558   pos 
+= strlen(wxBuffer
); 
 562 //      SendMessage((HWND) GetHWND(), WM_SETTEXT, 0, (LPARAM)tmp_buffer); 
 563       SetWindowText((HWND
) GetHWND(), tmp_buffer
); 
 564       SendMessage((HWND
) GetHWND(), EM_SETMODIFY
, FALSE
, 0L); 
 572 // If file is null, try saved file name first 
 573 // Returns TRUE if succeeds. 
 574 bool wxTextCtrl::SaveFile(const wxString
& file
) 
 576     wxString 
theFile(file
); 
 579         theFile 
= m_fileName
; 
 584     m_fileName 
= theFile
; 
 586     ofstream 
output((char*) (const char*) theFile
); 
 590     // This will only save 64K max 
 591     unsigned long nbytes 
= SendMessage((HWND
) GetHWND(), WM_GETTEXTLENGTH
, 0, 0); 
 592     char *tmp_buffer 
= (char*)farmalloc((size_t)(nbytes
+1)); 
 593     SendMessage((HWND
) GetHWND(), WM_GETTEXT
, (WPARAM
)(nbytes
+1), (LPARAM
)tmp_buffer
); 
 594     char *pstr 
= tmp_buffer
; 
 596     // Convert \r\n to just \n 
 605     SendMessage((HWND
) GetHWND(), EM_SETMODIFY
, FALSE
, 0L); 
 610 void wxTextCtrl::WriteText(const wxString
& text
) 
 613     int len 
= text
.Length(); 
 614     char *newtext 
= new char[(len
*2)+1]; 
 624       newtext
[j
] = text
[i
]; 
 629     SendMessage((HWND
) GetHWND(), EM_REPLACESEL
, 0, (LPARAM
)newtext
); 
 633 void wxTextCtrl::Clear() 
 635 //    SendMessage((HWND) GetHWND(), WM_SETTEXT, 0, (LPARAM)""); 
 636     SetWindowText((HWND
) GetHWND(), ""); 
 639 bool wxTextCtrl::IsModified() const 
 641     return (SendMessage((HWND
) GetHWND(), EM_GETMODIFY
, 0, 0) != 0); 
 644 // Makes 'unmodified' 
 645 void wxTextCtrl::DiscardEdits() 
 647   SendMessage((HWND
) GetHWND(), EM_SETMODIFY
, FALSE
, 0L); 
 651  * Some of the following functions are yet to be implemented 
 655 int wxTextCtrl::GetNumberOfLines() const 
 657     return (int)SendMessage((HWND
) GetHWND(), EM_GETLINECOUNT
, (WPARAM
)0, (LPARAM
)0); 
 660 long wxTextCtrl::XYToPosition(long x
, long y
) const 
 662     HWND hWnd 
= (HWND
) GetHWND(); 
 664     // This gets the char index for the _beginning_ of this line 
 665     int charIndex 
= (int)SendMessage(hWnd
, EM_LINEINDEX
, (WPARAM
)y
, (LPARAM
)0); 
 666     return (long)(x 
+ charIndex
); 
 669 void wxTextCtrl::PositionToXY(long pos
, long *x
, long *y
) const 
 671     HWND hWnd 
= (HWND
) GetHWND(); 
 673     // This gets the line number containing the character 
 674     int lineNo 
= (int)SendMessage(hWnd
, EM_LINEFROMCHAR
, (WPARAM
)pos
, (LPARAM
)0); 
 675     // This gets the char index for the _beginning_ of this line 
 676     int charIndex 
= (int)SendMessage(hWnd
, EM_LINEINDEX
, (WPARAM
)lineNo
, (LPARAM
)0); 
 677     // The X position must therefore be the different between pos and charIndex 
 678     *x 
= (long)(pos 
- charIndex
); 
 682 void wxTextCtrl::ShowPosition(long pos
) 
 684     HWND hWnd 
= (HWND
) GetHWND(); 
 686     // To scroll to a position, we pass the number of lines and characters 
 687     // to scroll *by*. This means that we need to: 
 688     // (1) Find the line position of the current line. 
 689     // (2) Find the line position of pos. 
 690     // (3) Scroll by (pos - current). 
 691     // For now, ignore the horizontal scrolling. 
 693     // Is this where scrolling is relative to - the line containing the caret? 
 694     // Or is the first visible line??? Try first visible line. 
 695 //    int currentLineLineNo1 = (int)SendMessage(hWnd, EM_LINEFROMCHAR, (WPARAM)-1, (LPARAM)0L); 
 697     int currentLineLineNo 
= (int)SendMessage(hWnd
, EM_GETFIRSTVISIBLELINE
, (WPARAM
)0, (LPARAM
)0L); 
 699     int specifiedLineLineNo 
= (int)SendMessage(hWnd
, EM_LINEFROMCHAR
, (WPARAM
)pos
, (LPARAM
)0L); 
 701     int linesToScroll 
= specifiedLineLineNo 
- currentLineLineNo
; 
 704     wxDebugMsg("Caret line: %d; Current visible line: %d; Specified line: %d; lines to scroll: %d\n", 
 705       currentLineLineNo1, currentLineLineNo, specifiedLineLineNo, linesToScroll); 
 708     if (linesToScroll 
!= 0) 
 709       (void)SendMessage(hWnd
, EM_LINESCROLL
, (WPARAM
)0, (LPARAM
)MAKELPARAM(linesToScroll
, 0)); 
 712 int wxTextCtrl::GetLineLength(long lineNo
) const 
 714     long charIndex 
= XYToPosition(0, lineNo
); 
 715     HWND hWnd 
= (HWND
) GetHWND(); 
 716     int len 
= (int)SendMessage(hWnd
, EM_LINELENGTH
, (WPARAM
)charIndex
, (LPARAM
)0); 
 720 wxString 
wxTextCtrl::GetLineText(long lineNo
) const 
 722     HWND hWnd 
= (HWND
) GetHWND(); 
 723     *(WORD 
*)wxBuffer 
= 512; 
 724     int noChars 
= (int)SendMessage(hWnd
, EM_GETLINE
, (WPARAM
)lineNo
, (LPARAM
)wxBuffer
); 
 725     wxBuffer
[noChars
] = 0; 
 726   return wxString(wxBuffer
); 
 733 void wxTextCtrl::Command(wxCommandEvent 
& event
) 
 735   SetValue (event
.GetString()); 
 736   ProcessCommand (event
); 
 739 void wxTextCtrl::OnDropFiles(wxDropFilesEvent
& event
) 
 741   // By default, load the first file into the text window. 
 742   if (event
.GetNumberOfFiles() > 0) 
 744     LoadFile(event
.GetFiles()[0]); 
 748 // The streambuf code was partly taken from chapter 3 by Jerry Schwarz of 
 749 // AT&T's "C++ Lanuage System Release 3.0 Library Manual" - Stein Somers 
 751 //========================================================================= 
 752 // Called then the buffer is full (gcc 2.6.3)  
 753 // or when "endl" is output (Borland 4.5) 
 754 //========================================================================= 
 755 // Class declaration using multiple inheritance doesn't work properly for 
 756 // Borland. See note in textctrl.h. 
 757 #ifndef NO_TEXT_WINDOW_STREAM 
 758 int wxTextCtrl::overflow(int c
) 
 760   // Make sure there is a holding area 
 761   // this is not needed in <iostream> usage as it automagically allocates 
 762   // it, but does someone want to emulate it for safety's sake? 
 764   if ( allocate()==EOF 
) 
 766     wxLogError("Streambuf allocation failed"); 
 771   // Verify that there are no characters in get area 
 772   if ( gptr() && gptr() < egptr() ) 
 774      wxError("Who's trespassing my get area?","Internal error"); 
 781   // Make sure there is a put area 
 784 /* This doesn't seem to be fatal so comment out error message */ 
 785 //    wxError("Put area not opened","Internal error"); 
 788           setp( base(), base() ); 
 790           setp( pbase(), pbase() ); 
 794   // Determine how many characters have been inserted but no consumed 
 795   int plen 
= pptr() - pbase(); 
 797   // Now Jerry relies on the fact that the buffer is at least 2 chars 
 798   // long, but the holding area "may be as small as 1" ??? 
 799   // And we need an additional \0, so let's keep this inefficient but 
 802   // If c!=EOF, it is a character that must also be comsumed 
 803   int xtra 
= c
==EOF
? 0 : 1; 
 805   // Write temporary C-string to wxTextWindow 
 807   char *txt 
= new char[plen
+xtra
+1]; 
 808   memcpy(txt
, pbase(), plen
); 
 809   txt
[plen
] = (char)c
;     // append c 
 810   txt
[plen
+xtra
] = '\0';   // append '\0' or overwrite c 
 811     // If the put area already contained \0, output will be truncated there 
 817   setp(pbase(), epptr()); 
 819 #if defined(__WATCOMC__) 
 821 #elif defined(zapeof)     // HP-UX (all cfront based?) 
 824   return c
!=EOF 
? c 
: 0;  // this should make everybody happy 
 828   int len = pptr() - pbase(); 
 829   char *txt = new char[len+1]; 
 830   strncpy(txt, pbase(), len); 
 833   setp(pbase(), epptr()); 
 839 //========================================================================= 
 840 // called then "endl" is output (gcc) or then explicit sync is done (Borland) 
 841 //========================================================================= 
 842 int wxTextCtrl::sync() 
 844   // Verify that there are no characters in get area 
 845   if ( gptr() && gptr() < egptr() ) 
 847      wxError("Who's trespassing my get area?","Internal error"); 
 851   if ( pptr() && pptr() > pbase() ) return overflow(EOF
); 
 855   int len = pptr() - pbase(); 
 856   char *txt = new char[len+1]; 
 857   strncpy(txt, pbase(), len); 
 860   setp(pbase(), epptr()); 
 866 //========================================================================= 
 867 // Should not be called by a "ostream". Used by a "istream" 
 868 //========================================================================= 
 869 int wxTextCtrl::underflow() 
 875 wxTextCtrl
& wxTextCtrl::operator<<(const wxString
& s
) 
 881 wxTextCtrl
& wxTextCtrl::operator<<(float f
) 
 884     str
.Printf("%.2f", f
); 
 889 wxTextCtrl
& wxTextCtrl::operator<<(double d
) 
 892     str
.Printf("%.2f", d
); 
 897 wxTextCtrl
& wxTextCtrl::operator<<(int i
) 
 905 wxTextCtrl
& wxTextCtrl::operator<<(long i
) 
 908     str
.Printf("%ld", i
); 
 913 wxTextCtrl
& wxTextCtrl::operator<<(const char c
) 
 923 WXHBRUSH 
wxTextCtrl::OnCtlColor(WXHDC pDC
, WXHWND pWnd
, WXUINT nCtlColor
, 
 924       WXUINT message
, WXWPARAM wParam
, WXLPARAM lParam
) 
 929     HBRUSH hbrush 
= Ctl3dCtlColorEx(message
, wParam
, lParam
); 
 930     return (WXHBRUSH
) hbrush
; 
 934   if (GetParent()->GetTransparentBackground()) 
 935     SetBkMode((HDC
) pDC
, TRANSPARENT
); 
 937     SetBkMode((HDC
) pDC
, OPAQUE
); 
 939   ::SetBkColor((HDC
) pDC
, RGB(GetBackgroundColour().Red(), GetBackgroundColour().Green(), GetBackgroundColour().Blue())); 
 940   ::SetTextColor((HDC
) pDC
, RGB(GetForegroundColour().Red(), GetForegroundColour().Green(), GetForegroundColour().Blue())); 
 942   wxBrush 
*backgroundBrush 
= wxTheBrushList
->FindOrCreateBrush(GetBackgroundColour(), wxSOLID
); 
 944   // Note that this will be cleaned up in wxApp::OnIdle, if backgroundBrush 
 945   // has a zero usage count. 
 946   // NOT NOW - will be cleaned up at end of app. 
 947 //  backgroundBrush->RealizeResource(); 
 948   return (WXHBRUSH
) backgroundBrush
->GetResourceHandle(); 
 951 void wxTextCtrl::OnChar(wxKeyEvent
& event
) 
 953     switch( event
.KeyCode() ) 
 955         // Fix by Marcel Rasche to allow Alt-Ctrl insertion of special characters 
 964                 char c 
= (char)event
.KeyCode(); 
 970             wxASSERT_MSG( m_windowStyle 
& wxTE_PROCESS_ENTER
, 
 971                           "this text ctrl should never receive return" ); 
 973                 wxCommandEvent 
event(wxEVT_COMMAND_TEXT_ENTER
, m_windowId
); 
 974                 event
.SetEventObject( this ); 
 975                 if ( GetEventHandler()->ProcessEvent(event
) ) 
 980             // only produce navigation event if we don't process TAB ourself 
 981             if ( !(m_windowStyle 
& wxTE_PROCESS_TAB
) ) 
 983                 wxNavigationKeyEvent event
; 
 984                 event
.SetDirection(!(::GetKeyState(VK_SHIFT
) & 0x100)); 
 985                 event
.SetWindowChange(FALSE
); 
 986                 event
.SetEventObject(this); 
 988                 if ( GetEventHandler()->ProcessEvent(event
) ) 
 993     // don't just call event.Skip() because this will cause TABs and ENTERs 
 994     // be passed upwards and we don't always want this - instead process it 
 999 long wxTextCtrl::MSWGetDlgCode() 
1001     // we always want the characters and the arrows 
1002     long lRc 
= DLGC_WANTCHARS 
| DLGC_WANTARROWS
; 
1004     // we may have several different cases: 
1005     // 1. normal case: both TAB and ENTER are used for dialog navigation 
1006     // 2. ctrl which wants TAB for itself: ENTER is used to pass to the next 
1007     //    control in the dialog 
1008     // 3. ctrl which wants ENTER for itself: TAB is used for dialog navigation 
1009     // 4. ctrl which wants both TAB and ENTER: Ctrl-ENTER is used to pass to 
1011     if ( m_windowStyle 
& wxTE_PROCESS_ENTER 
) 
1012         lRc 
|= DLGC_WANTMESSAGE
; 
1013     if ( m_windowStyle 
& wxTE_PROCESS_TAB 
) 
1014         lRc 
|= DLGC_WANTTAB
; 
1019 void wxTextCtrl::OnEraseBackground(wxEraseEvent
& event
) 
1021     if ( m_windowStyle 
& wxTE_MULTILINE 
) 
1023         // No flicker - only problem is we probably can't change the background 
1027         ::GetClientRect((HWND) GetHWND(), &rect); 
1029         HBRUSH hBrush = ::CreateSolidBrush(PALETTERGB(GetBackgroundColour().Red(), GetBackgroundColour().Green(), GetBackgroundColour().Blue())); 
1030         int mode = ::SetMapMode((HDC) event.GetDC()->GetHDC(), MM_TEXT); 
1032         ::FillRect ((HDC) event.GetDC()->GetHDC(), &rect, hBrush); 
1033         ::DeleteObject(hBrush); 
1034         ::SetMapMode((HDC) event.GetDC()->GetHDC(), mode); 
1037 //        wxWindow::OnEraseBackground(event); 
1040 bool wxTextCtrl::MSWCommand(WXUINT param
, WXWORD 
WXUNUSED(id
)) 
1044   wxDebugMsg("Edit control %d: ", (int)id); 
1048       wxDebugMsg("EN_SETFOCUS\n"); 
1051       wxDebugMsg("EN_KILLFOCUS\n"); 
1054       wxDebugMsg("EN_CHANGE\n"); 
1057       wxDebugMsg("EN_UPDATE\n"); 
1060       wxDebugMsg("EN_ERRSPACE\n"); 
1063       wxDebugMsg("EN_MAXTEXT\n"); 
1066       wxDebugMsg("EN_HSCROLL\n"); 
1069       wxDebugMsg("EN_VSCROLL\n"); 
1072       wxDebugMsg("Unknown EDIT notification\n"); 
1081         wxFocusEvent 
event(param 
== EN_KILLFOCUS 
? wxEVT_KILL_FOCUS
 
1084         event
.SetEventObject( this ); 
1085         GetEventHandler()->ProcessEvent(event
); 
1091         wxCommandEvent 
event(wxEVT_COMMAND_TEXT_UPDATED
, m_windowId
); 
1092         wxString 
val(GetValue()); 
1093         if ( !val
.IsNull() ) 
1094           event
.m_commandString 
= WXSTRINGCAST val
; 
1095         event
.SetEventObject( this ); 
1096         ProcessCommand(event
); 
1100     // the other notification messages are not processed 
1115 // For Rich Edit controls. Do we need it? 
1118 bool wxTextCtrl::MSWNotify(WXWPARAM wParam
, WXLPARAM lParam
) 
1120   wxCommandEvent 
event(0, m_windowId
); 
1122   NMHDR 
*hdr1 
= (NMHDR 
*) lParam
; 
1123   switch ( hdr1
->code 
) 
1125     // Insert case code here 
1127       return wxControl::MSWNotify(wParam
, lParam
); 
1131   event
.SetEventObject( this ); 
1132   event
.SetEventType(eventType
); 
1134   if ( !GetEventHandler()->ProcessEvent(event
) )