1 ///////////////////////////////////////////////////////////////////////////// 
   2 // Name:        msw/textctrl.cpp 
   4 // Author:      Julian Smart 
   8 // Copyright:   (c) Julian Smart and Markus Holzem 
   9 // Licence:     wxWindows license 
  10 ///////////////////////////////////////////////////////////////////////////// 
  12 // ============================================================================ 
  14 // ============================================================================ 
  17     #pragma implementation "textctrl.h" 
  20 // ---------------------------------------------------------------------------- 
  22 // ---------------------------------------------------------------------------- 
  24 // For compilers that support precompilation, includes "wx.h". 
  25 #include "wx/wxprec.h" 
  34     #include "wx/textctrl.h" 
  35     #include "wx/settings.h" 
  43 #include "wx/module.h" 
  46     #include "wx/clipbrd.h" 
  49 #include "wx/textfile.h" 
  53 #include "wx/msw/private.h" 
  57 #include <sys/types.h> 
  59 #if wxUSE_RICHEDIT && (!defined(__GNUWIN32_OLD__) || defined(__CYGWIN10__)) 
  63 // old mingw32 doesn't define this 
  65     #define CFM_CHARSET 0x08000000 
  69     #define CFM_BACKCOLOR 0x04000000 
  72 // cygwin does not have these defined for richedit 
  74     #define ENM_LINK 0x04000000 
  77 #ifndef EM_AUTOURLDETECT 
  78     #define EM_AUTOURLDETECT (WM_USER + 91) 
  82     #define EN_LINK 0x070b 
  84     typedef struct _enlink
 
  95     #define SF_UNICODE 0x0010 
  98 // ---------------------------------------------------------------------------- 
 100 // ---------------------------------------------------------------------------- 
 104 DWORD CALLBACK 
wxRichEditStreamIn(DWORD dwCookie
, BYTE 
*buf
, LONG cb
, LONG 
*pcb
); 
 106 #endif // wxUSE_RICHEDIT 
 108 // ---------------------------------------------------------------------------- 
 110 // ---------------------------------------------------------------------------- 
 114 // this module initializes RichEdit DLL if needed 
 115 class wxRichEditModule 
: public wxModule
 
 118     virtual bool OnInit(); 
 119     virtual void OnExit(); 
 121     // get the version currently loaded, -1 if none 
 122     static int GetLoadedVersion() { return ms_verRichEdit
; } 
 124     // load the richedit DLL of at least of required version 
 125     static bool Load(int version 
= 1); 
 128     // the handle to richedit DLL and the version of the DLL loaded 
 129     static HINSTANCE ms_hRichEdit
; 
 131     // the DLL version loaded or -1 if none 
 132     static int ms_verRichEdit
; 
 134     DECLARE_DYNAMIC_CLASS(wxRichEditModule
) 
 137 HINSTANCE 
wxRichEditModule::ms_hRichEdit 
= (HINSTANCE
)NULL
; 
 138 int       wxRichEditModule::ms_verRichEdit 
= -1; 
 140 IMPLEMENT_DYNAMIC_CLASS(wxRichEditModule
, wxModule
) 
 142 #endif // wxUSE_RICHEDIT 
 144 // ---------------------------------------------------------------------------- 
 145 // event tables and other macros 
 146 // ---------------------------------------------------------------------------- 
 148 IMPLEMENT_DYNAMIC_CLASS(wxTextCtrl
, wxControl
) 
 150 BEGIN_EVENT_TABLE(wxTextCtrl
, wxControl
) 
 151     EVT_CHAR(wxTextCtrl::OnChar
) 
 152     EVT_DROP_FILES(wxTextCtrl::OnDropFiles
) 
 154     EVT_MENU(wxID_CUT
, wxTextCtrl::OnCut
) 
 155     EVT_MENU(wxID_COPY
, wxTextCtrl::OnCopy
) 
 156     EVT_MENU(wxID_PASTE
, wxTextCtrl::OnPaste
) 
 157     EVT_MENU(wxID_UNDO
, wxTextCtrl::OnUndo
) 
 158     EVT_MENU(wxID_REDO
, wxTextCtrl::OnRedo
) 
 160     EVT_UPDATE_UI(wxID_CUT
, wxTextCtrl::OnUpdateCut
) 
 161     EVT_UPDATE_UI(wxID_COPY
, wxTextCtrl::OnUpdateCopy
) 
 162     EVT_UPDATE_UI(wxID_PASTE
, wxTextCtrl::OnUpdatePaste
) 
 163     EVT_UPDATE_UI(wxID_UNDO
, wxTextCtrl::OnUpdateUndo
) 
 164     EVT_UPDATE_UI(wxID_REDO
, wxTextCtrl::OnUpdateRedo
) 
 166     EVT_ERASE_BACKGROUND(wxTextCtrl::OnEraseBackground
) 
 170 // ============================================================================ 
 172 // ============================================================================ 
 174 // ---------------------------------------------------------------------------- 
 176 // ---------------------------------------------------------------------------- 
 178 void wxTextCtrl::Init() 
 185 bool wxTextCtrl::Create(wxWindow 
*parent
, wxWindowID id
, 
 186                         const wxString
& value
, 
 190                         const wxValidator
& validator
, 
 191                         const wxString
& name
) 
 193     // base initialization 
 194     if ( !CreateBase(parent
, id
, pos
, size
, style
, validator
, name
) ) 
 198         parent
->AddChild(this); 
 200     // translate wxWin style flags to MSW ones, checking for consistency while 
 202     long msStyle 
= ES_LEFT 
| WS_TABSTOP
; 
 204     if ( m_windowStyle 
& wxCLIP_SIBLINGS 
) 
 205         msStyle 
|= WS_CLIPSIBLINGS
; 
 207     if ( m_windowStyle 
& wxTE_MULTILINE 
) 
 209         wxASSERT_MSG( !(m_windowStyle 
& wxTE_PROCESS_ENTER
), 
 210                       wxT("wxTE_PROCESS_ENTER style is ignored for multiline text controls (they always process it)") ); 
 212         msStyle 
|= ES_MULTILINE 
| ES_WANTRETURN
; 
 213         if ((m_windowStyle 
& wxTE_NO_VSCROLL
) == 0) 
 214             msStyle 
|= WS_VSCROLL
; 
 215         m_windowStyle 
|= wxTE_PROCESS_ENTER
; 
 219         // there is really no reason to not have this style for single line 
 221         msStyle 
|= ES_AUTOHSCROLL
; 
 224     if ( m_windowStyle 
& wxHSCROLL 
) 
 225         msStyle 
|= WS_HSCROLL 
| ES_AUTOHSCROLL
; 
 227     if ( m_windowStyle 
& wxTE_READONLY 
) 
 228         msStyle 
|= ES_READONLY
; 
 230     if ( m_windowStyle 
& wxTE_PASSWORD 
) 
 231         msStyle 
|= ES_PASSWORD
; 
 233     if ( m_windowStyle 
& wxTE_AUTO_SCROLL 
) 
 234         msStyle 
|= ES_AUTOHSCROLL
; 
 236     if ( m_windowStyle 
& wxTE_NOHIDESEL 
) 
 237         msStyle 
|= ES_NOHIDESEL
; 
 239     // we always want the characters and the arrows 
 240     m_lDlgCode 
= DLGC_WANTCHARS 
| DLGC_WANTARROWS
; 
 242     // we may have several different cases: 
 243     // 1. normal case: both TAB and ENTER are used for dialog navigation 
 244     // 2. ctrl which wants TAB for itself: ENTER is used to pass to the next 
 245     //    control in the dialog 
 246     // 3. ctrl which wants ENTER for itself: TAB is used for dialog navigation 
 247     // 4. ctrl which wants both TAB and ENTER: Ctrl-ENTER is used to pass to 
 249     if ( m_windowStyle 
& wxTE_PROCESS_ENTER 
) 
 250         m_lDlgCode 
|= DLGC_WANTMESSAGE
; 
 251     if ( m_windowStyle 
& wxTE_PROCESS_TAB 
) 
 252         m_lDlgCode 
|= DLGC_WANTTAB
; 
 254     // do create the control - either an EDIT or RICHEDIT 
 255     wxString windowClass 
= wxT("EDIT"); 
 258     if ( m_windowStyle 
& wxTE_RICH 
) 
 260         static bool s_errorGiven 
= FALSE
;   // MT-FIXME 
 262         // only give the error msg once if the DLL can't be loaded 
 265             // first try to load the RichEdit DLL (will do nothing if already 
 267             if ( !wxRichEditModule::Load() ) 
 269                 wxLogError(_("Impossible to create a rich edit control, using simple text control instead. Please reinstall riched32.dll")); 
 277             msStyle 
|= ES_AUTOVSCROLL
; 
 279             m_verRichEdit 
= wxRichEditModule::GetLoadedVersion(); 
 280             if ( m_verRichEdit 
== 1 ) 
 282                 windowClass 
= wxT("RICHEDIT"); 
 286 #ifndef RICHEDIT_CLASS 
 287                 wxString RICHEDIT_CLASS
; 
 288                 RICHEDIT_CLASS
.Printf(_T("RichEdit%d0"), m_verRichEdit
); 
 290                 RICHEDIT_CLASS 
+= _T('W'); 
 292                 RICHEDIT_CLASS 
+= _T('A'); 
 293 #endif // Unicode/ANSI 
 294 #endif // !RICHEDIT_CLASS 
 296                 windowClass 
= RICHEDIT_CLASS
; 
 300 #endif // wxUSE_RICHEDIT 
 302     // we need to turn '\n's into "\r\n"s for the multiline controls 
 304     if ( m_windowStyle 
& wxTE_MULTILINE 
) 
 306         valueWin 
= wxTextFile::Translate(value
, wxTextFileType_Dos
); 
 313     if ( !MSWCreateControl(windowClass
, msStyle
, pos
, size
, valueWin
) ) 
 316     SetBackgroundColour(wxSystemSettings::GetSystemColour(wxSYS_COLOUR_WINDOW
)); 
 321         // have to enable events manually 
 322         LPARAM mask 
= ENM_CHANGE 
| ENM_DROPFILES 
| ENM_SELCHANGE 
| ENM_UPDATE
; 
 324         if ( m_windowStyle 
& wxTE_AUTO_URL 
) 
 328             ::SendMessage(GetHwnd(), EM_AUTOURLDETECT
, TRUE
, 0); 
 331         ::SendMessage(GetHwnd(), EM_SETEVENTMASK
, 0, mask
); 
 333 #endif // wxUSE_RICHEDIT 
 338 // Make sure the window style (etc.) reflects the HWND style (roughly) 
 339 void wxTextCtrl::AdoptAttributesFromHWND() 
 341     wxWindow::AdoptAttributesFromHWND(); 
 343     HWND hWnd 
= GetHwnd(); 
 344     long style 
= ::GetWindowLong(hWnd
, GWL_STYLE
); 
 346     // retrieve the style to see whether this is an edit or richedit ctrl 
 348     wxString classname 
= wxGetWindowClass(GetHWND()); 
 350     if ( classname
.IsSameAs(_T("EDIT"), FALSE 
/* no case */) ) 
 357         if ( wxSscanf(classname
, _T("RichEdit%d0%c"), &m_verRichEdit
, &c
) != 2 ) 
 359             wxLogDebug(_T("Unknown edit control '%s'."), classname
.c_str()); 
 364 #endif // wxUSE_RICHEDIT 
 366     if (style 
& ES_MULTILINE
) 
 367         m_windowStyle 
|= wxTE_MULTILINE
; 
 368     if (style 
& ES_PASSWORD
) 
 369         m_windowStyle 
|= wxTE_PASSWORD
; 
 370     if (style 
& ES_READONLY
) 
 371         m_windowStyle 
|= wxTE_READONLY
; 
 372     if (style 
& ES_WANTRETURN
) 
 373         m_windowStyle 
|= wxTE_PROCESS_ENTER
; 
 376 // ---------------------------------------------------------------------------- 
 377 // set/get the controls text 
 378 // ---------------------------------------------------------------------------- 
 380 wxString 
wxTextCtrl::GetValue() const 
 382     // we can't use wxGetWindowText() (i.e. WM_GETTEXT internally) for 
 383     // retrieving more than 64Kb under Win9x 
 389         int len 
= GetWindowTextLength(GetHwnd()); 
 392             // alloc one extra WORD as needed by the control 
 393             wxChar 
*p 
= str
.GetWriteBuf(++len
); 
 396             textRange
.chrg
.cpMin 
= 0; 
 397             textRange
.chrg
.cpMax 
= -1; 
 398             textRange
.lpstrText 
= p
; 
 400             (void)SendMessage(GetHwnd(), EM_GETTEXTRANGE
, 0, (LPARAM
)&textRange
); 
 402             // believe it or not, but EM_GETTEXTRANGE uses just CR ('\r') for 
 403             // the newlines which is neither Unix nor Windows style (Win95 with 
 404             // riched20.dll shows this behaviour) - convert it to something 
 408                 if ( *p 
== _T('\r') ) 
 414         //else: no text at all, leave the string empty 
 418 #endif // wxUSE_RICHEDIT 
 420     // WM_GETTEXT uses standard DOS CR+LF (\r\n) convention - convert to the 
 421     // same one as above for consitency 
 422     wxString str 
= wxGetWindowText(GetHWND()); 
 424     return wxTextFile::Translate(str
, wxTextFileType_Unix
); 
 427 void wxTextCtrl::SetValue(const wxString
& value
) 
 429     // if the text is long enough, it's faster to just set it instead of first 
 430     // comparing it with the old one (chances are that it will be different 
 431     // anyhow, this comparison is there to avoid flicker for small single-line 
 432     // edit controls mostly) 
 433     if ( (value
.length() > 0x400) || (value 
!= GetValue()) ) 
 435         // it is simpler to do this but it could be more efficient to reproduce 
 436         // WriteText() logic here 
 441         // for compatibility, don't move the cursor when doing SetValue() 
 442         SetInsertionPoint(0); 
 446 #if wxUSE_RICHEDIT && !wxUSE_UNICODE 
 448 DWORD CALLBACK 
wxRichEditStreamIn(DWORD dwCookie
, BYTE 
*buf
, LONG cb
, LONG 
*pcb
) 
 452     wchar_t *wbuf 
= (wchar_t *)buf
; 
 453     const wchar_t *wpc 
= *(const wchar_t **)dwCookie
; 
 458         cb 
-= sizeof(wchar_t); 
 459         (*pcb
) += sizeof(wchar_t); 
 462     *(const wchar_t **)dwCookie 
= wpc
; 
 467 extern long wxEncodingToCodepage(wxFontEncoding encoding
); // from strconv.cpp 
 469 bool wxTextCtrl::StreamIn(const wxString
& value
, wxFontEncoding encoding
) 
 471     // we have to use EM_STREAMIN to force richedit control 2.0+ to show any 
 472     // text in the non default charset - otherwise it thinks it knows better 
 473     // than we do and always shows it in the default one 
 475     // first get the Windows code page for this encoding 
 476     long codepage 
= wxEncodingToCodepage(encoding
); 
 477     if ( codepage 
== -1 ) 
 483     // next translate to Unicode using this code page 
 484     int len 
= ::MultiByteToWideChar(codepage
, 0, value
, -1, NULL
, 0); 
 485     wxWCharBuffer 
wchBuf(len
); 
 486     if ( !::MultiByteToWideChar(codepage
, 0, value
, -1, 
 487                                 (wchar_t *)wchBuf
.data(), len
) ) 
 489         wxLogLastError(_T("MultiByteToWideChar")); 
 492     // finally, stream it in the control 
 493     const wchar_t *wpc 
= wchBuf
; 
 497     eds
.dwCookie 
= (DWORD
)&wpc
; 
 498     // the cast below is needed for broken (very) old mingw32 headers 
 499     eds
.pfnCallback 
= (EDITSTREAMCALLBACK
)wxRichEditStreamIn
; 
 501     if ( !::SendMessage(GetHwnd(), EM_STREAMIN
, 
 502                         SF_TEXT 
| SF_UNICODE 
| SFF_SELECTION
, 
 503                         (LPARAM
)&eds
) || eds
.dwError 
) 
 505         wxLogLastError(_T("EM_STREAMIN")); 
 513 #endif // wxUSE_RICHEDIT 
 515 void wxTextCtrl::WriteText(const wxString
& value
) 
 517     wxString valueDos 
= wxTextFile::Translate(value
, wxTextFileType_Dos
); 
 520     // there are several complications with the rich edit controls here 
 524         // first, ensure that the new text will be in the default style 
 525         if ( !m_defaultStyle
.IsDefault() ) 
 528             GetSelection(&start
, &end
); 
 529             SetStyle(start
, end
, m_defaultStyle 
); 
 533         // next check if the text we're inserting must be shown in a non 
 534         // default charset -- this only works for RichEdit > 1.0 
 535         if ( GetRichVersion() > 1 ) 
 537             wxFont font 
= m_defaultStyle
.GetFont(); 
 543                wxFontEncoding encoding 
= font
.GetEncoding(); 
 544                if ( encoding 
!= wxFONTENCODING_SYSTEM 
) 
 546                    done 
= StreamIn(valueDos
, encoding
); 
 550 #endif // !wxUSE_UNICODE 
 554 #endif // wxUSE_RICHEDIT 
 556         ::SendMessage(GetHwnd(), EM_REPLACESEL
, 0, (LPARAM
)valueDos
.c_str()); 
 562 void wxTextCtrl::AppendText(const wxString
& text
) 
 564     SetInsertionPointEnd(); 
 569 void wxTextCtrl::Clear() 
 571     ::SetWindowText(GetHwnd(), wxT("")); 
 574 // ---------------------------------------------------------------------------- 
 575 // Clipboard operations 
 576 // ---------------------------------------------------------------------------- 
 578 void wxTextCtrl::Copy() 
 582         HWND hWnd 
= GetHwnd(); 
 583         SendMessage(hWnd
, WM_COPY
, 0, 0L); 
 587 void wxTextCtrl::Cut() 
 591         HWND hWnd 
= GetHwnd(); 
 592         SendMessage(hWnd
, WM_CUT
, 0, 0L); 
 596 void wxTextCtrl::Paste() 
 600         HWND hWnd 
= GetHwnd(); 
 601         SendMessage(hWnd
, WM_PASTE
, 0, 0L); 
 605 bool wxTextCtrl::CanCopy() const 
 607     // Can copy if there's a selection 
 609     GetSelection(& from
, & to
); 
 610     return (from 
!= to
) ; 
 613 bool wxTextCtrl::CanCut() const 
 615     // Can cut if there's a selection 
 617     GetSelection(& from
, & to
); 
 618     return (from 
!= to
) && (IsEditable()); 
 621 bool wxTextCtrl::CanPaste() const 
 629         UINT cf 
= 0; // 0 == any format 
 631         return ::SendMessage(GetHwnd(), EM_CANPASTE
, cf
, 0) != 0; 
 633 #endif // wxUSE_RICHEDIT 
 635     // Standard edit control: check for straight text on clipboard 
 636     if ( !::OpenClipboard(GetHwndOf(wxTheApp
->GetTopWindow())) ) 
 639     bool isTextAvailable 
= ::IsClipboardFormatAvailable(CF_TEXT
) != 0; 
 642     return isTextAvailable
; 
 645 // ---------------------------------------------------------------------------- 
 647 // ---------------------------------------------------------------------------- 
 649 void wxTextCtrl::SetEditable(bool editable
) 
 651     HWND hWnd 
= GetHwnd(); 
 652     SendMessage(hWnd
, EM_SETREADONLY
, (WPARAM
)!editable
, (LPARAM
)0L); 
 655 void wxTextCtrl::SetInsertionPoint(long pos
) 
 657     SetSelection(pos
, pos
); 
 663         static const wxChar 
*nothing 
= _T(""); 
 664         SendMessage(GetHwnd(), EM_REPLACESEL
, 0, (LPARAM
)nothing
); 
 668 void wxTextCtrl::SetInsertionPointEnd() 
 670     long pos 
= GetLastPosition(); 
 671     SetInsertionPoint(pos
); 
 674 long wxTextCtrl::GetInsertionPoint() const 
 682         SendMessage(GetHwnd(), EM_EXGETSEL
, 0, (LPARAM
) &range
); 
 685 #endif // wxUSE_RICHEDIT 
 687     DWORD Pos 
= (DWORD
)SendMessage(GetHwnd(), EM_GETSEL
, 0, 0L); 
 691 long wxTextCtrl::GetLastPosition() const 
 693     HWND hWnd 
= GetHwnd(); 
 695     // Will always return a number > 0 (according to docs) 
 696     int noLines 
= (int)SendMessage(hWnd
, EM_GETLINECOUNT
, (WPARAM
)0, (LPARAM
)0L); 
 698     // This gets the char index for the _beginning_ of the last line 
 699     int charIndex 
= (int)SendMessage(hWnd
, EM_LINEINDEX
, (WPARAM
)(noLines
-1), (LPARAM
)0L); 
 701     // Get number of characters in the last line. We'll add this to the character 
 702     // index for the last line, 1st position. 
 703     int lineLength 
= (int)SendMessage(hWnd
, EM_LINELENGTH
, (WPARAM
)charIndex
, (LPARAM
)0L); 
 705     return (long)(charIndex 
+ lineLength
); 
 708 // If the return values from and to are the same, there is no 
 710 void wxTextCtrl::GetSelection(long* from
, long* to
) const 
 716         ::SendMessage(GetHwnd(), EM_EXGETSEL
, 0, (LPARAM
) &charRange
); 
 718         *from 
= charRange
.cpMin
; 
 719         *to 
= charRange
.cpMax
; 
 722 #endif // !wxUSE_RICHEDIT 
 724         DWORD dwStart
, dwEnd
; 
 725         ::SendMessage(GetHwnd(), EM_GETSEL
, (WPARAM
)&dwStart
, (LPARAM
)&dwEnd
); 
 732 wxString 
wxTextCtrl::GetStringSelection() const 
 734     // the base class version works correctly for the rich text controls 
 735     // because there the lines are terminated with just '\r' which means that 
 736     // the string length is not changed in the result of the translations doen 
 737     // in GetValue() but for the normal ones when we replace "\r\n" with '\n' 
 738     // we break the indices 
 741         return wxTextCtrlBase::GetStringSelection(); 
 742 #endif // wxUSE_RICHEDIT 
 745     GetSelection(&from
, &to
); 
 750         str 
= wxGetWindowText(GetHWND()).Mid(from
, to 
- from
); 
 752         // and now that we have the correct selection, convert it to the 
 754         str 
= wxTextFile::Translate(str
, wxTextFileType_Unix
); 
 760 bool wxTextCtrl::IsEditable() const 
 762     long style 
= ::GetWindowLong(GetHwnd(), GWL_STYLE
); 
 764     return ((style 
& ES_READONLY
) == 0); 
 767 // ---------------------------------------------------------------------------- 
 769 // ---------------------------------------------------------------------------- 
 771 void wxTextCtrl::SetSelection(long from
, long to
) 
 773     DoSetSelection(from
, to
); 
 776 void wxTextCtrl::DoSetSelection(long from
, long to
, bool scrollCaret
) 
 778     // if from and to are both -1, it means (in wxWindows) that all text should 
 779     // be selected - translate into Windows convention 
 780     if ( (from 
== -1) && (to 
== -1) ) 
 786     HWND hWnd 
= GetHwnd(); 
 795         SendMessage(hWnd
, EM_EXSETSEL
, 0, (LPARAM
) &range
); 
 798 #endif // wxUSE_RICHEDIT 
 800         SendMessage(hWnd
, EM_SETSEL
, (WPARAM
)from
, (LPARAM
)to
); 
 805         SendMessage(hWnd
, EM_SCROLLCARET
, (WPARAM
)0, (LPARAM
)0); 
 808     // WPARAM is 0: selection is scrolled into view 
 809     SendMessage(hWnd
, EM_SETSEL
, (WPARAM
)0, (LPARAM
)MAKELONG(from
, to
)); 
 813 // ---------------------------------------------------------------------------- 
 815 // ---------------------------------------------------------------------------- 
 817 void wxTextCtrl::Replace(long from
, long to
, const wxString
& value
) 
 819     // Set selection and remove it 
 820     DoSetSelection(from
, to
, FALSE
); 
 822     SendMessage(GetHwnd(), EM_REPLACESEL
, 
 828                 (LPARAM
)value
.c_str()); 
 831 void wxTextCtrl::Remove(long from
, long to
) 
 833     Replace(from
, to
, _T("")); 
 836 bool wxTextCtrl::LoadFile(const wxString
& file
) 
 838     if ( wxTextCtrlBase::LoadFile(file
) ) 
 840         // update the size limit if needed 
 849 bool wxTextCtrl::IsModified() const 
 851     return (SendMessage(GetHwnd(), EM_GETMODIFY
, 0, 0) != 0); 
 854 // Makes 'unmodified' 
 855 void wxTextCtrl::DiscardEdits() 
 857     SendMessage(GetHwnd(), EM_SETMODIFY
, FALSE
, 0L); 
 860 int wxTextCtrl::GetNumberOfLines() const 
 862     return (int)SendMessage(GetHwnd(), EM_GETLINECOUNT
, (WPARAM
)0, (LPARAM
)0); 
 865 long wxTextCtrl::XYToPosition(long x
, long y
) const 
 867     HWND hWnd 
= GetHwnd(); 
 869     // This gets the char index for the _beginning_ of this line 
 870     int charIndex 
= (int)SendMessage(hWnd
, EM_LINEINDEX
, (WPARAM
)y
, (LPARAM
)0); 
 871     return (long)(x 
+ charIndex
); 
 874 bool wxTextCtrl::PositionToXY(long pos
, long *x
, long *y
) const 
 876     HWND hWnd 
= GetHwnd(); 
 878     // This gets the line number containing the character 
 883         lineNo 
= (int)SendMessage(hWnd
, EM_EXLINEFROMCHAR
, 0, (LPARAM
)pos
); 
 886 #endif // wxUSE_RICHEDIT 
 887         lineNo 
= (int)SendMessage(hWnd
, EM_LINEFROMCHAR
, (WPARAM
)pos
, 0); 
 895     // This gets the char index for the _beginning_ of this line 
 896     int charIndex 
= (int)SendMessage(hWnd
, EM_LINEINDEX
, (WPARAM
)lineNo
, (LPARAM
)0); 
 897     if ( charIndex 
== -1 ) 
 902     // The X position must therefore be the different between pos and charIndex 
 904         *x 
= (long)(pos 
- charIndex
); 
 911 void wxTextCtrl::ShowPosition(long pos
) 
 913     HWND hWnd 
= GetHwnd(); 
 915     // To scroll to a position, we pass the number of lines and characters 
 916     // to scroll *by*. This means that we need to: 
 917     // (1) Find the line position of the current line. 
 918     // (2) Find the line position of pos. 
 919     // (3) Scroll by (pos - current). 
 920     // For now, ignore the horizontal scrolling. 
 922     // Is this where scrolling is relative to - the line containing the caret? 
 923     // Or is the first visible line??? Try first visible line. 
 924 //    int currentLineLineNo1 = (int)SendMessage(hWnd, EM_LINEFROMCHAR, (WPARAM)-1, (LPARAM)0L); 
 926     int currentLineLineNo 
= (int)SendMessage(hWnd
, EM_GETFIRSTVISIBLELINE
, (WPARAM
)0, (LPARAM
)0L); 
 928     int specifiedLineLineNo 
= (int)SendMessage(hWnd
, EM_LINEFROMCHAR
, (WPARAM
)pos
, (LPARAM
)0L); 
 930     int linesToScroll 
= specifiedLineLineNo 
- currentLineLineNo
; 
 932     if (linesToScroll 
!= 0) 
 933       (void)SendMessage(hWnd
, EM_LINESCROLL
, (WPARAM
)0, (LPARAM
)linesToScroll
); 
 936 int wxTextCtrl::GetLineLength(long lineNo
) const 
 938     long charIndex 
= XYToPosition(0, lineNo
); 
 939     int len 
= (int)SendMessage(GetHwnd(), EM_LINELENGTH
, charIndex
, 0); 
 943 wxString 
wxTextCtrl::GetLineText(long lineNo
) const 
 945     size_t len 
= (size_t)GetLineLength(lineNo
) + 1; 
 947     // there must be at least enough place for the length WORD in the 
 952     wxChar 
*buf 
= str
.GetWriteBuf(len
); 
 954     *(WORD 
*)buf 
= (WORD
)len
; 
 955     len 
= (size_t)::SendMessage(GetHwnd(), EM_GETLINE
, lineNo
, (LPARAM
)buf
); 
 958     str
.UngetWriteBuf(len
); 
 963 void wxTextCtrl::SetMaxLength(unsigned long len
) 
 965     ::SendMessage(GetHwnd(), EM_LIMITTEXT
, len
, 0); 
 968 // ---------------------------------------------------------------------------- 
 970 // ---------------------------------------------------------------------------- 
 972 void wxTextCtrl::Undo() 
 976         ::SendMessage(GetHwnd(), EM_UNDO
, 0, 0); 
 980 void wxTextCtrl::Redo() 
 984         // Same as Undo, since Undo undoes the undo, i.e. a redo. 
 985         ::SendMessage(GetHwnd(), EM_UNDO
, 0, 0); 
 989 bool wxTextCtrl::CanUndo() const 
 991     return (::SendMessage(GetHwnd(), EM_CANUNDO
, 0, 0) != 0); 
 994 bool wxTextCtrl::CanRedo() const 
 996     return (::SendMessage(GetHwnd(), EM_CANUNDO
, 0, 0) != 0); 
 999 // ---------------------------------------------------------------------------- 
1000 // implemenation details 
1001 // ---------------------------------------------------------------------------- 
1003 void wxTextCtrl::Command(wxCommandEvent 
& event
) 
1005     SetValue(event
.GetString()); 
1006     ProcessCommand (event
); 
1009 void wxTextCtrl::OnDropFiles(wxDropFilesEvent
& event
) 
1011     // By default, load the first file into the text window. 
1012     if (event
.GetNumberOfFiles() > 0) 
1014         LoadFile(event
.GetFiles()[0]); 
1018 // ---------------------------------------------------------------------------- 
1019 // kbd input processing 
1020 // ---------------------------------------------------------------------------- 
1022 bool wxTextCtrl::MSWShouldPreProcessMessage(WXMSG
* pMsg
) 
1024     MSG 
*msg 
= (MSG 
*)pMsg
; 
1026     // check for our special keys here: if we don't do it and the parent frame 
1027     // uses them as accelerators, they wouldn't work at all, so we disable 
1028     // usual preprocessing for them 
1029     if ( msg
->message 
== WM_KEYDOWN 
) 
1031         WORD vkey 
= msg
->wParam
; 
1032         if ( (HIWORD(msg
->lParam
) & KF_ALTDOWN
) == KF_ALTDOWN 
) 
1034             if ( vkey 
== VK_BACK 
) 
1039             if ( wxIsCtrlDown() ) 
1053             else if ( wxIsShiftDown() ) 
1055                 if ( vkey 
== VK_INSERT 
|| vkey 
== VK_DELETE 
) 
1061     return wxControl::MSWShouldPreProcessMessage(pMsg
); 
1064 void wxTextCtrl::OnChar(wxKeyEvent
& event
) 
1066     switch ( event
.KeyCode() ) 
1069             if ( !(m_windowStyle 
& wxTE_MULTILINE
) ) 
1071                 wxCommandEvent 
event(wxEVT_COMMAND_TEXT_ENTER
, m_windowId
); 
1072                 InitCommandEvent(event
); 
1073                 event
.SetString(GetValue()); 
1074                 if ( GetEventHandler()->ProcessEvent(event
) ) 
1077             //else: multiline controls need Enter for themselves 
1082             // always produce navigation event - even if we process TAB 
1083             // ourselves the fact that we got here means that the user code 
1084             // decided to skip processing of this TAB - probably to let it 
1085             // do its default job. 
1087                 wxNavigationKeyEvent eventNav
; 
1088                 eventNav
.SetDirection(!event
.ShiftDown()); 
1089                 eventNav
.SetWindowChange(event
.ControlDown()); 
1090                 eventNav
.SetEventObject(this); 
1092                 if ( GetParent()->GetEventHandler()->ProcessEvent(eventNav
) ) 
1098     // no, we didn't process it 
1102 bool wxTextCtrl::MSWCommand(WXUINT param
, WXWORD 
WXUNUSED(id
)) 
1109                 wxFocusEvent 
event(param 
== EN_KILLFOCUS 
? wxEVT_KILL_FOCUS
 
1112                 event
.SetEventObject( this ); 
1113                 GetEventHandler()->ProcessEvent(event
); 
1119                 wxCommandEvent 
event(wxEVT_COMMAND_TEXT_UPDATED
, m_windowId
); 
1120                 InitCommandEvent(event
); 
1121                 event
.SetString(GetValue()); 
1122                 ProcessCommand(event
); 
1127             // the text size limit has been hit - increase it 
1128             if ( !AdjustSpaceLimit() ) 
1130                 wxCommandEvent 
event(wxEVT_COMMAND_TEXT_MAXLEN
, m_windowId
); 
1131                 InitCommandEvent(event
); 
1132                 event
.SetString(GetValue()); 
1133                 ProcessCommand(event
); 
1137             // the other notification messages are not processed 
1151 WXHBRUSH 
wxTextCtrl::OnCtlColor(WXHDC pDC
, WXHWND 
WXUNUSED(pWnd
), WXUINT 
WXUNUSED(nCtlColor
), 
1157                                WXUINT 
WXUNUSED(message
), 
1158                                WXWPARAM 
WXUNUSED(wParam
), 
1159                                WXLPARAM 
WXUNUSED(lParam
) 
1166         HBRUSH hbrush 
= Ctl3dCtlColorEx(message
, wParam
, lParam
); 
1167         return (WXHBRUSH
) hbrush
; 
1169 #endif // wxUSE_CTL3D 
1172     if (GetParent()->GetTransparentBackground()) 
1173         SetBkMode(hdc
, TRANSPARENT
); 
1175         SetBkMode(hdc
, OPAQUE
); 
1177     wxColour colBack 
= GetBackgroundColour(); 
1179     if (!IsEnabled() && (GetWindowStyle() & wxTE_MULTILINE
) == 0) 
1180         colBack 
= wxSystemSettings::GetSystemColour(wxSYS_COLOUR_3DFACE
); 
1182     ::SetBkColor(hdc
, wxColourToRGB(colBack
)); 
1183     ::SetTextColor(hdc
, wxColourToRGB(GetForegroundColour())); 
1185     wxBrush 
*brush 
= wxTheBrushList
->FindOrCreateBrush(colBack
, wxSOLID
); 
1187     return (WXHBRUSH
)brush
->GetResourceHandle(); 
1190 // In WIN16, need to override normal erasing because 
1191 // Ctl3D doesn't use the wxWindows background colour. 
1193 void wxTextCtrl::OnEraseBackground(wxEraseEvent
& event
) 
1195     wxColour 
col(m_backgroundColour
); 
1199         col 
= wxSystemSettings::GetSystemColour(wxSYS_COLOUR_WINDOW
); 
1203     ::GetClientRect(GetHwnd(), &rect
); 
1205     COLORREF ref 
= PALETTERGB(col
.Red(), 
1208     HBRUSH hBrush 
= ::CreateSolidBrush(ref
); 
1210         wxLogLastError(wxT("CreateSolidBrush")); 
1212     HDC hdc 
= (HDC
)event
.GetDC()->GetHDC(); 
1214     int mode 
= ::SetMapMode(hdc
, MM_TEXT
); 
1216     ::FillRect(hdc
, &rect
, hBrush
); 
1217     ::DeleteObject(hBrush
); 
1218     ::SetMapMode(hdc
, mode
); 
1223 bool wxTextCtrl::AdjustSpaceLimit() 
1226     unsigned int limit 
= ::SendMessage(GetHwnd(), EM_GETLIMITTEXT
, 0, 0); 
1228     // HACK: we try to automatically extend the limit for the amount of text 
1229     //       to allow (interactively) entering more than 64Kb of text under 
1230     //       Win9x but we shouldn't reset the text limit which was previously 
1231     //       set explicitly with SetMaxLength() 
1233     //       we could solve this by storing the limit we set in wxTextCtrl but 
1234     //       to save space we prefer to simply test here the actual limit 
1235     //       value: we consider that SetMaxLength() can only be called for 
1237     if ( limit 
< 0x8000 ) 
1239         // we've got more text than limit set by SetMaxLength() 
1243     unsigned int len 
= ::GetWindowTextLength(GetHwnd()); 
1246         limit 
= len 
+ 0x8000;    // 32Kb 
1251             // as a nice side effect, this also allows passing limit > 64Kb 
1252             ::SendMessage(GetHwnd(), EM_EXLIMITTEXT
, 0, limit
); 
1255 #endif // wxUSE_RICHEDIT 
1257             if ( limit 
> 0xffff ) 
1259                 // this will set it to a platform-dependent maximum (much more 
1260                 // than 64Kb under NT) 
1264             ::SendMessage(GetHwnd(), EM_LIMITTEXT
, limit
, 0); 
1269     // we changed the limit 
1273 bool wxTextCtrl::AcceptsFocus() const 
1275     // we don't want focus if we can't be edited 
1276     return IsEditable() && wxControl::AcceptsFocus(); 
1279 wxSize 
wxTextCtrl::DoGetBestSize() const 
1282     wxGetCharSize(GetHWND(), &cx
, &cy
, &GetFont()); 
1284     int wText 
= DEFAULT_ITEM_WIDTH
; 
1286     int hText 
= EDIT_HEIGHT_FROM_CHAR_HEIGHT(cy
); 
1287     if ( m_windowStyle 
& wxTE_MULTILINE 
) 
1289         hText 
*= wxMax(GetNumberOfLines(), 5); 
1291     //else: for single line control everything is ok 
1293     return wxSize(wText
, hText
); 
1296 // ---------------------------------------------------------------------------- 
1297 // standard handlers for standard edit menu events 
1298 // ---------------------------------------------------------------------------- 
1300 void wxTextCtrl::OnCut(wxCommandEvent
& WXUNUSED(event
)) 
1305 void wxTextCtrl::OnCopy(wxCommandEvent
& WXUNUSED(event
)) 
1310 void wxTextCtrl::OnPaste(wxCommandEvent
& WXUNUSED(event
)) 
1315 void wxTextCtrl::OnUndo(wxCommandEvent
& WXUNUSED(event
)) 
1320 void wxTextCtrl::OnRedo(wxCommandEvent
& WXUNUSED(event
)) 
1325 void wxTextCtrl::OnUpdateCut(wxUpdateUIEvent
& event
) 
1327     event
.Enable( CanCut() ); 
1330 void wxTextCtrl::OnUpdateCopy(wxUpdateUIEvent
& event
) 
1332     event
.Enable( CanCopy() ); 
1335 void wxTextCtrl::OnUpdatePaste(wxUpdateUIEvent
& event
) 
1337     event
.Enable( CanPaste() ); 
1340 void wxTextCtrl::OnUpdateUndo(wxUpdateUIEvent
& event
) 
1342     event
.Enable( CanUndo() ); 
1345 void wxTextCtrl::OnUpdateRedo(wxUpdateUIEvent
& event
) 
1347     event
.Enable( CanRedo() ); 
1350 // the rest of the file only deals with the rich edit controls 
1353 // ---------------------------------------------------------------------------- 
1354 // EN_LINK processing 
1355 // ---------------------------------------------------------------------------- 
1357 bool wxTextCtrl::MSWOnNotify(int idCtrl
, WXLPARAM lParam
, WXLPARAM 
*result
) 
1359     NMHDR 
*hdr 
= (NMHDR
* )lParam
; 
1360     if ( hdr
->code 
== EN_LINK 
) 
1362         ENLINK 
*enlink 
= (ENLINK 
*)hdr
; 
1364         switch ( enlink
->msg 
) 
1367                 // ok, so it is hardcoded - do we really nee to customize it? 
1368                 ::SetCursor(GetHcursorOf(wxCursor(wxCURSOR_HAND
))); 
1373             case WM_LBUTTONDOWN
: 
1375             case WM_LBUTTONDBLCLK
: 
1376             case WM_RBUTTONDOWN
: 
1378             case WM_RBUTTONDBLCLK
: 
1379                 // send a mouse event 
1381                     static const wxEventType eventsMouse
[] = 
1392                     // the event ids are consecutive 
1394                         evtMouse(eventsMouse
[enlink
->msg 
- WM_MOUSEMOVE
]); 
1396                     InitMouseEvent(evtMouse
, 
1397                                    GET_X_LPARAM(enlink
->lParam
), 
1398                                    GET_Y_LPARAM(enlink
->lParam
), 
1401                     wxTextUrlEvent 
event(m_windowId
, evtMouse
, 
1403                                          enlink
->chrg
.cpMax
); 
1405                     InitCommandEvent(event
); 
1407                     *result 
= ProcessCommand(event
); 
1419 // ---------------------------------------------------------------------------- 
1420 // colour setting for the rich edit controls 
1421 // ---------------------------------------------------------------------------- 
1423 // Watcom C++ doesn't define this 
1425 #define SCF_ALL 0x0004 
1428 bool wxTextCtrl::SetBackgroundColour(const wxColour
& colour
) 
1430     if ( !wxTextCtrlBase::SetBackgroundColour(colour
) ) 
1432         // colour didn't really change 
1438         // rich edit doesn't use WM_CTLCOLOR, hence we need to send 
1439         // EM_SETBKGNDCOLOR additionally 
1440         ::SendMessage(GetHwnd(), EM_SETBKGNDCOLOR
, 0, wxColourToRGB(colour
)); 
1446 bool wxTextCtrl::SetForegroundColour(const wxColour
& colour
) 
1448     if ( !wxTextCtrlBase::SetForegroundColour(colour
) ) 
1450         // colour didn't really change 
1456         // change the colour of everything 
1459         cf
.cbSize 
= sizeof(cf
); 
1460         cf
.dwMask 
= CFM_COLOR
; 
1461         cf
.crTextColor 
= wxColourToRGB(colour
); 
1462         ::SendMessage(GetHwnd(), EM_SETCHARFORMAT
, SCF_ALL
, (LPARAM
)&cf
); 
1468 // ---------------------------------------------------------------------------- 
1469 // styling support for rich edit controls 
1470 // ---------------------------------------------------------------------------- 
1472 bool wxTextCtrl::SetStyle(long start
, long end
, const wxTextAttr
& style
) 
1476         // can't do it with normal text control 
1480     // the rich text control doesn't handle setting background colour, so don't 
1481     // even try if it's the only thing we want to change 
1482     if ( wxRichEditModule::GetLoadedVersion() < 2 && 
1483          !style
.HasFont() && !style
.HasTextColour() ) 
1485         // nothing to do: return TRUE if there was really nothing to do and 
1486         // FALSE if we failed to set bg colour 
1487         return !style
.HasBackgroundColour(); 
1490     // order the range if needed 
1498     // we can only change the format of the selection, so select the range we 
1499     // want and restore the old selection later 
1500     long startOld
, endOld
; 
1501     GetSelection(&startOld
, &endOld
); 
1503     // but do we really have to change the selection? 
1504     bool changeSel 
= start 
!= startOld 
|| end 
!= endOld
; 
1508         DoSetSelection(start
, end
, FALSE
); 
1511     // initialize CHARFORMAT struct 
1518     cf
.cbSize 
= sizeof(cf
); 
1520     if ( style
.HasFont() ) 
1522         // VZ: CFM_CHARSET doesn't seem to do anything at all in RichEdit 2.0 
1523         //     but using it doesn't seem to hurt neither so leaving it for now 
1525         cf
.dwMask 
|= CFM_FACE 
| CFM_SIZE 
| CFM_CHARSET 
| 
1526                      CFM_ITALIC 
| CFM_BOLD 
| CFM_UNDERLINE
; 
1528         // fill in data from LOGFONT but recalculate lfHeight because we need 
1529         // the real height in twips and not the negative number which 
1530         // wxFillLogFont() returns (this is correct in general and works with 
1531         // the Windows font mapper, but not here) 
1533         wxFillLogFont(&lf
, &style
.GetFont()); 
1534         cf
.yHeight 
= 20*style
.GetFont().GetPointSize(); // 1 pt = 20 twips 
1535         cf
.bCharSet 
= lf
.lfCharSet
; 
1536         cf
.bPitchAndFamily 
= lf
.lfPitchAndFamily
; 
1537         wxStrncpy( cf
.szFaceName
, lf
.lfFaceName
, WXSIZEOF(cf
.szFaceName
) ); 
1539         // also deal with underline/italic/bold attributes: note that we must 
1540         // always set CFM_ITALIC &c bits in dwMask, even if we don't set the 
1541         // style to allow clearing it 
1544             cf
.dwEffects 
|= CFE_ITALIC
; 
1547         if ( lf
.lfWeight 
== FW_BOLD 
) 
1549             cf
.dwEffects 
|= CFE_BOLD
; 
1552         if ( lf
.lfUnderline 
) 
1554             cf
.dwEffects 
|= CFE_UNDERLINE
; 
1557         // strikeout fonts are not supported by wxWindows 
1560     if ( style
.HasTextColour() ) 
1562         cf
.dwMask 
|= CFM_COLOR
; 
1563         cf
.crTextColor 
= wxColourToRGB(style
.GetTextColour()); 
1567     if ( wxRichEditModule::GetLoadedVersion() > 1 && style
.HasBackgroundColour() ) 
1569         cf
.dwMask 
|= CFM_BACKCOLOR
; 
1570         cf
.crBackColor 
= wxColourToRGB(style
.GetBackgroundColour()); 
1572 #endif // wxUSE_RICHEDIT2 
1574     // do format the selection 
1575     bool ok 
= ::SendMessage(GetHwnd(), EM_SETCHARFORMAT
, 
1576                             SCF_SELECTION
, (LPARAM
)&cf
) != 0; 
1579         wxLogDebug(_T("SendMessage(EM_SETCHARFORMAT, SCF_SELECTION) failed")); 
1584         // restore the original selection 
1585         DoSetSelection(startOld
, endOld
, FALSE
); 
1591 // ---------------------------------------------------------------------------- 
1593 // ---------------------------------------------------------------------------- 
1595 bool wxRichEditModule::OnInit() 
1597     // don't do anything - we will load it when needed 
1601 void wxRichEditModule::OnExit() 
1605         FreeLibrary(ms_hRichEdit
); 
1610 bool wxRichEditModule::Load(int version
) 
1612     wxCHECK_MSG( version 
>= 1 && version 
<= 3, FALSE
, 
1613                  _T("incorrect richedit control version requested") ); 
1615     if ( version 
<= ms_verRichEdit 
) 
1617         // we've already got this or better 
1623         ::FreeLibrary(ms_hRichEdit
); 
1626     // always try load riched20.dll first - like this we won't have to reload 
1627     // it later if we're first asked for RE 1 and then for RE 2 or 3 
1628     wxString dllname 
= _T("riched20.dll"); 
1629     ms_hRichEdit 
= ::LoadLibrary(dllname
); 
1630     ms_verRichEdit 
= 2; // no way to tell if it's 2 or 3, assume 2 
1632     if ( !ms_hRichEdit 
&& (version 
== 1) ) 
1634         // fall back to RE 1 
1635         dllname 
= _T("riched32.dll"); 
1636         ms_hRichEdit 
= ::LoadLibrary(dllname
); 
1640     if ( !ms_hRichEdit 
) 
1642         wxLogSysError(_("Could not load Rich Edit DLL '%s'"), dllname
.c_str()); 
1644         ms_verRichEdit 
= -1; 
1652 #endif // wxUSE_RICHEDIT 
1654 #endif // wxUSE_TEXTCTRL