1 /////////////////////////////////////////////////////////////////////////////
4 // Author: David Webster
8 // Copyright: (c) David Webster
9 // Licence: wxWindows licence
10 /////////////////////////////////////////////////////////////////////////////
12 // ----------------------------------------------------------------------------
14 // ----------------------------------------------------------------------------
16 // For compilers that support precompilation, includes "wx.h".
17 #include "wx/wxprec.h"
20 #include "wx/textctrl.h"
21 #include "wx/scrolwin.h"
22 #include "wx/settings.h"
30 #include "wx/clipbrd.h"
33 #include "wx/textfile.h"
35 #include "wx/os2/private.h"
39 #include <sys/types.h>
47 #if !defined(MLE_INDEX)
53 // ----------------------------------------------------------------------------
54 // event tables and other macros
55 // ----------------------------------------------------------------------------
57 IMPLEMENT_DYNAMIC_CLASS(wxTextCtrl
, wxControl
)
59 BEGIN_EVENT_TABLE(wxTextCtrl
, wxControl
)
60 EVT_CHAR(wxTextCtrl::OnChar
)
61 EVT_DROP_FILES(wxTextCtrl::OnDropFiles
)
63 EVT_MENU(wxID_CUT
, wxTextCtrl::OnCut
)
64 EVT_MENU(wxID_COPY
, wxTextCtrl::OnCopy
)
65 EVT_MENU(wxID_PASTE
, wxTextCtrl::OnPaste
)
66 EVT_MENU(wxID_UNDO
, wxTextCtrl::OnUndo
)
67 EVT_MENU(wxID_REDO
, wxTextCtrl::OnRedo
)
69 EVT_UPDATE_UI(wxID_CUT
, wxTextCtrl::OnUpdateCut
)
70 EVT_UPDATE_UI(wxID_COPY
, wxTextCtrl::OnUpdateCopy
)
71 EVT_UPDATE_UI(wxID_PASTE
, wxTextCtrl::OnUpdatePaste
)
72 EVT_UPDATE_UI(wxID_UNDO
, wxTextCtrl::OnUpdateUndo
)
73 EVT_UPDATE_UI(wxID_REDO
, wxTextCtrl::OnUpdateRedo
)
77 // ============================================================================
79 // ============================================================================
81 // ----------------------------------------------------------------------------
83 // ----------------------------------------------------------------------------
85 wxTextCtrl::wxTextCtrl()
89 bool wxTextCtrl::Create(
92 , const wxString
& rsValue
97 , const wxValidator
& rValidator
99 , const wxString
& rsName
105 // Base initialization
107 if ( !CreateBase( pParent
119 wxPoint vPos
= rPos
; // The OS/2 position
123 pParent
->AddChild(this);
124 hParent
= GetWinHwnd(pParent
);
126 // OS2 uses normal coordinates, no bassackwards Windows ones
128 if (pParent
->IsKindOf(CLASSINFO(wxGenericScrolledWindow
)) ||
129 pParent
->IsKindOf(CLASSINFO(wxScrolledWindow
))
132 wxWindow
* pGrandParent
= NULL
;
134 pGrandParent
= pParent
->GetParent();
136 nTempy
= pGrandParent
->GetSize().y
- (vPos
.y
+ rSize
.y
);
138 nTempy
= pParent
->GetSize().y
- (vPos
.y
+ rSize
.y
);
141 nTempy
= pParent
->GetSize().y
- (vPos
.y
+ rSize
.y
);
148 ::WinQueryWindowRect(HWND_DESKTOP
, &vRect
);
149 hParent
= HWND_DESKTOP
;
150 vPos
.y
= vRect
.yTop
- (vPos
.y
+ rSize
.y
);
153 m_windowStyle
= lStyle
;
155 long lSstyle
= WS_VISIBLE
| WS_TABSTOP
;
158 // Single and multiline edit fields are two different controls in PM
160 if ( m_windowStyle
& wxTE_MULTILINE
)
163 m_windowStyle
|= wxTE_PROCESS_ENTER
;
165 if ((m_windowStyle
& wxTE_NO_VSCROLL
) == 0)
166 lSstyle
|= MLS_VSCROLL
;
167 if (m_windowStyle
& wxHSCROLL
)
168 lSstyle
|= MLS_HSCROLL
;
169 if (m_windowStyle
& wxTE_READONLY
)
170 lSstyle
|= MLS_READONLY
;
176 if (m_windowStyle
& wxHSCROLL
)
177 lSstyle
|= ES_AUTOSCROLL
;
178 if (m_windowStyle
& wxTE_READONLY
)
179 lSstyle
|= ES_READONLY
;
180 if (m_windowStyle
& wxTE_PASSWORD
) // hidden input
181 lSstyle
|= ES_UNREADABLE
;
183 if ( pParent
->IsKindOf(CLASSINFO(wxGenericScrolledWindow
)) ||
184 pParent
->IsKindOf(CLASSINFO(wxScrolledWindow
))
186 lSstyle
|= WS_CLIPSIBLINGS
;
189 m_hWnd
= (WXHWND
)::WinCreateWindow( (HWND
)GetHwndOf(pParent
) // Parent window handle
190 ,WC_MLE
// Window class
191 ,(PSZ
)rsValue
.c_str() // Initial Text
192 ,(ULONG
)lSstyle
// Style flags
193 ,(LONG
)0 // X pos of origin
194 ,(LONG
)0 // Y pos of origin
195 ,(LONG
)0 // field width
196 ,(LONG
)0 // field height
197 ,(HWND
)GetHwndOf(pParent
) // owner window handle (same as parent
198 ,HWND_TOP
// initial z position
199 ,(ULONG
)vId
// Window identifier
200 ,NULL
// no control data
201 ,NULL
// no Presentation parameters
206 m_hWnd
= (WXHWND
)::WinCreateWindow( (HWND
)GetHwndOf(pParent
) // Parent window handle
207 ,WC_ENTRYFIELD
// Window class
208 ,(PSZ
)rsValue
.c_str() // Initial Text
209 ,(ULONG
)lSstyle
// Style flags
210 ,(LONG
)0 // X pos of origin
211 ,(LONG
)0 // Y pos of origin
212 ,(LONG
)0 // field width
213 ,(LONG
)0 // field height
214 ,(HWND
)GetHwndOf(pParent
) // owner window handle (same as parent
215 ,HWND_TOP
// initial z position
216 ,(ULONG
)vId
// Window identifier
217 ,NULL
// no control data
218 ,NULL
// no Presentation parameters
227 SubclassWin(GetHWND());
230 // Set font, position, size and initial value
232 wxFont
& vFontParent
= pParent
->GetFont();
234 if (vFontParent
.Ok())
236 SetFont(vFontParent
);
240 SetFont(wxSystemSettings::GetSystemFont(wxSYS_SYSTEM_FONT
));
242 if (!rsValue
.IsEmpty())
253 } // end of wxTextCtrl::Create
256 // Make sure the window style (etc.) reflects the HWND style (roughly)
258 void wxTextCtrl::AdoptAttributesFromHWND()
260 HWND hWnd
= GetHwnd();
261 LONG lStyle
= ::WinQueryWindowULong(hWnd
, QWL_STYLE
);
263 wxWindow::AdoptAttributesFromHWND();
267 m_windowStyle
|= wxTE_MULTILINE
;
268 if (lStyle
& MLS_READONLY
)
269 m_windowStyle
|= wxTE_READONLY
;
273 if (lStyle
& ES_UNREADABLE
)
274 m_windowStyle
|= wxTE_PASSWORD
;
275 if (lStyle
& ES_READONLY
)
276 m_windowStyle
|= wxTE_READONLY
;
278 } // end of wxTextCtrl::AdoptAttributesFromHWND
280 void wxTextCtrl::SetupColours()
282 wxColour vBkgndColour
;
284 vBkgndColour
= wxSystemSettings::GetSystemColour(wxSYS_COLOUR_WINDOW
);
285 SetBackgroundColour(vBkgndColour
);
286 SetForegroundColour(GetParent()->GetForegroundColour());
287 } // end of wxTextCtrl::SetupColours
289 // ----------------------------------------------------------------------------
290 // set/get the controls text
291 // ----------------------------------------------------------------------------
293 wxString
wxTextCtrl::GetValue() const
295 wxString sStr
= wxGetWindowText(GetHWND());
296 char* zStr
= (char*)sStr
.c_str();
298 for ( ; *zStr
; zStr
++ )
301 // this will replace \r\n with just \n
310 } // end of wxTextCtrl::GetValue
312 void wxTextCtrl::SetValue(
313 const wxString
& rsValue
317 // If the text is long enough, it's faster to just set it instead of first
318 // comparing it with the old one (chances are that it will be different
319 // anyhow, this comparison is there to avoid flicker for small single-line
320 // edit controls mostly)
322 if ((rsValue
.length() > 0x400) || (rsValue
!= GetValue()))
324 ::WinSetWindowText(GetHwnd(), rsValue
.c_str());
327 } // end of wxTextCtrl::SetValue
329 void wxTextCtrl::WriteText(
330 const wxString
& rsValue
333 ::WinSetWindowText(GetHwnd(), rsValue
.c_str());
335 } // end of wxTextCtrl::WriteText
337 void wxTextCtrl::AppendText(
338 const wxString
& rsText
341 SetInsertionPointEnd();
343 } // end of wxTextCtrl::AppendText
345 void wxTextCtrl::Clear()
347 ::WinSetWindowText(GetHwnd(), "");
348 } // end of wxTextCtrl::Clear
350 // ----------------------------------------------------------------------------
351 // Clipboard operations
352 // ----------------------------------------------------------------------------
354 void wxTextCtrl::Copy()
358 HWND hWnd
= GetHwnd();
360 ::WinSendMsg(hWnd
, MLM_COPY
, 0, 0);
362 ::WinSendMsg(hWnd
, EM_COPY
, 0, 0);
364 } // end of wxTextCtrl::Copy
366 void wxTextCtrl::Cut()
370 HWND hWnd
= GetHwnd();
373 ::WinSendMsg(hWnd
, MLM_CUT
, 0, 0);
375 ::WinSendMsg(hWnd
, EM_CUT
, 0, 0);
377 } // end of wxTextCtrl::Cut
379 void wxTextCtrl::Paste()
383 HWND hWnd
= GetHwnd();
385 ::WinSendMsg(hWnd
, EM_PASTE
, 0, 0);
387 } // end of wxTextCtrl::Paste
389 bool wxTextCtrl::CanCopy() const
392 // Can copy if there's a selection
397 GetSelection(&lFrom
, &lTo
);
398 return (lFrom
!= lTo
);
399 } // end of wxTextCtrl::CanCopy
401 bool wxTextCtrl::CanCut() const
404 // Can cut if there's a selection
409 GetSelection(&lFrom
, &lTo
);
410 return (lFrom
!= lTo
);
411 } // end of wxTextCtrl::CanCut
413 bool wxTextCtrl::CanPaste() const
415 bool bIsTextAvailable
= FALSE
;
421 // Check for straight text on clipboard
423 if (::WinOpenClipbrd(vHabmain
))
425 bIsTextAvailable
= (::WinQueryClipbrdData(vHabmain
, CF_TEXT
) != 0);
426 ::WinCloseClipbrd(vHabmain
);
428 return bIsTextAvailable
;
429 } // end of wxTextCtrl::CanPaste
431 // ----------------------------------------------------------------------------
433 // ----------------------------------------------------------------------------
435 void wxTextCtrl::SetEditable(
439 HWND hWnd
= GetHwnd();
442 ::WinSendMsg(hWnd
, MLM_SETREADONLY
, MPFROMLONG(!bEditable
), (MPARAM
)0);
444 ::WinSendMsg(hWnd
, EM_SETREADONLY
, MPFROMLONG(!bEditable
), (MPARAM
)0);
445 } // end of wxTextCtrl::SetEditable
447 void wxTextCtrl::SetInsertionPoint(
451 HWND hWnd
= GetHwnd();
454 ::WinSendMsg(hWnd
, MLM_SETSEL
, (MPARAM
)lPos
, (MPARAM
)lPos
);
456 ::WinSendMsg(hWnd
, EM_SETSEL
, MPFROM2SHORT((USHORT
)lPos
, (USHORT
)lPos
), (MPARAM
)0);
457 } // end of wxTextCtrl::SetInsertionPoint
459 void wxTextCtrl::SetInsertionPointEnd()
461 long lPos
= GetLastPosition();
463 SetInsertionPoint(lPos
);
464 } // end of wxTextCtrl::SetInsertionPointEnd
466 long wxTextCtrl::GetInsertionPoint() const
471 dwPos
= (WXDWORD
)::WinSendMsg(GetHwnd(), MLM_QUERYSEL
, (MPARAM
)MLFQS_MINSEL
, 0);
474 dwPos
= (WXDWORD
)::WinSendMsg(GetHwnd(), EM_QUERYSEL
, 0, 0);
475 dwPos
= SHORT1FROMMP((MPARAM
)dwPos
); // the first 16 bit value is the min pos
477 return (dwPos
& 0xFFFF);
478 } // end of wxTextCtrl::GetInsertionPoint
480 long wxTextCtrl::GetLastPosition() const
482 HWND hWnd
= GetHwnd();
491 // This just gets the total text length. The last will be this value
493 lLineLength
= (long)::WinSendMsg(hWnd
, MLM_QUERYTEXTLENGTH
, 0, 0);
500 vParams
.fsStatus
= WPM_CCHTEXT
;
501 if (::WinSendMsg( GetHwnd()
502 ,WM_QUERYWINDOWPARAMS
507 lLineLength
= (long)vParams
.cchText
;
512 return(lCharIndex
+ lLineLength
);
513 } // end of wxTextCtrl::GetLastPosition
515 // If the return values from and to are the same, there is no
517 void wxTextCtrl::GetSelection(
525 dwPos
= (WXDWORD
)::WinSendMsg(GetHwnd(), MLM_QUERYSEL
, (MPARAM
)MLFQS_MINSEL
, 0);
528 dwPos
= (WXDWORD
)::WinSendMsg(GetHwnd(), EM_QUERYSEL
, 0, 0);
530 *plFrom
= SHORT1FROMMP((MPARAM
)dwPos
); // the first 16 bit value is the min pos
531 *plTo
= SHORT2FROMMP((MPARAM
)dwPos
); // the first 16 bit value is the min pos
532 } // end of wxTextCtrl::GetSelection
534 bool wxTextCtrl::IsEditable() const
537 return((bool)LONGFROMMR(::WinSendMsg(GetHwnd(), MLM_QUERYREADONLY
, 0, 0)));
539 return((bool)LONGFROMMR(::WinSendMsg(GetHwnd(), EM_QUERYREADONLY
, 0, 0)));
540 } // end of wxTextCtrl::IsEditable
542 // ----------------------------------------------------------------------------
544 // ----------------------------------------------------------------------------
546 void wxTextCtrl::Replace(
549 , const wxString
& rsValue
553 HWND hWnd
= GetHwnd();
554 long lFromChar
= lFrom
;
558 // Set selection and remove it
562 ::WinSendMsg(hWnd
, MLM_SETSEL
, MPFROM2SHORT((USHORT
)lFrom
, (USHORT
)lTo
), 0);
563 ::WinSendMsg(hWnd
, MLM_CUT
, 0, 0);
567 ::WinSendMsg(hWnd
, EM_SETSEL
, MPFROM2SHORT((USHORT
)lFrom
, (USHORT
)lTo
), 0);
568 ::WinSendMsg(hWnd
, EM_CUT
, 0, 0);
572 // Now replace with 'value', by pasting.
574 wxSetClipboardData(wxDF_TEXT
, (wxObject
*) (const wxChar
*)rsValue
, 0, 0);
576 // Paste into edit control
578 ::WinSendMsg(hWnd
, MLM_PASTE
, (MPARAM
)0, (MPARAM
)0);
580 ::WinSendMsg(hWnd
, EM_PASTE
, (MPARAM
)0, (MPARAM
)0);
582 wxFAIL_MSG("wxTextCtrl::Replace not implemented if wxUSE_CLIPBOARD is 0.");
584 } // end of wxTextCtrl::Replace
586 void wxTextCtrl::Remove(
591 HWND hWnd
= GetHwnd();
592 long lFromChar
= lFrom
;
597 ::WinSendMsg(hWnd
, MLM_SETSEL
, MPFROM2SHORT((USHORT
)lFrom
, (USHORT
)lTo
), 0);
598 ::WinSendMsg(hWnd
, MLM_CUT
, 0, 0);
602 ::WinSendMsg(hWnd
, EM_SETSEL
, MPFROM2SHORT((USHORT
)lFrom
, (USHORT
)lTo
), 0);
603 ::WinSendMsg(hWnd
, EM_CUT
, 0, 0);
605 } // end of wxTextCtrl::Remove
607 void wxTextCtrl::SetSelection(
612 HWND hWnd
= GetHwnd();
613 long lFromChar
= lFrom
;
617 // If from and to are both -1, it means (in wxWindows) that all text should
618 // be selected. Translate into Windows convention
620 if ((lFrom
== -1L) && (lTo
== -1L))
626 ::WinSendMsg(hWnd
, MLM_SETSEL
, (MPARAM
)lFromChar
, (MPARAM
)lToChar
);
628 ::WinSendMsg(hWnd
, EM_SETSEL
, MPFROM2SHORT((USHORT
)lFromChar
, (USHORT
)lToChar
), (MPARAM
)0);
629 } // end of wxTextCtrl::SetSelection
631 bool wxTextCtrl::LoadFile(
632 const wxString
& rsFile
635 if ( wxTextCtrlBase::LoadFile(rsFile
) )
638 // Update the size limit if needed
644 } // end of wxTextCtrl::LoadFile
646 bool wxTextCtrl::IsModified() const
651 bRc
= (bool)LONGFROMMR(::WinSendMsg(GetHwnd(), MLM_QUERYCHANGED
, 0, 0));
653 bRc
= (bool)LONGFROMMR(::WinSendMsg(GetHwnd(), EM_QUERYCHANGED
, 0, 0));
655 } // end of wxTextCtrl::IsModified
658 // Makes 'unmodified'
660 void wxTextCtrl::DiscardEdits()
663 ::WinSendMsg(GetHwnd(), MLM_SETCHANGED
, MPFROMLONG(FALSE
), 0);
666 // EM controls do not have a SETCHANGED but issuing a query should reset it
668 ::WinSendMsg(GetHwnd(), EM_QUERYCHANGED
, 0, 0);
669 } // end of wxTextCtrl::DiscardEdits
671 int wxTextCtrl::GetNumberOfLines() const
676 nNumLines
= (int)::WinSendMsg(GetHwnd(), MLM_QUERYLINECOUNT
, 0, 0);
680 } // end of wxTextCtrl::GetNumberOfLines
682 long wxTextCtrl::XYToPosition(
687 HWND hWnd
= GetHwnd();
688 long lCharIndex
= 0L;
693 lLen
= (long)::WinSendMsg(GetHwnd(), MLM_QUERYLINELENGTH
, 0, 0);
694 lCharIndex
= ((lLen
* lY
) + lX
);
699 } // end of wxTextCtrl::XYToPosition
701 bool wxTextCtrl::PositionToXY(
707 HWND hWnd
= GetHwnd();
712 nLineNo
= (long)::WinSendMsg(hWnd
, MLM_LINEFROMCHAR
, (MPARAM
)lPos
, 0);
723 // This gets the char index for the _beginning_ of this line
729 lLineWidth
= (long)::WinSendMsg(hWnd
, MLM_QUERYLINELENGTH
, (MPARAM
)0, (MPARAM
)0);
730 lCharIndex
= (nLineNo
+ 1) * lLineWidth
;
736 vParams
.fsStatus
= WPM_CCHTEXT
;
737 if (::WinSendMsg( hWnd
738 ,WM_QUERYWINDOWPARAMS
743 lCharIndex
= vParams
.cchText
;
749 if (lCharIndex
== -1)
755 // The X position must therefore be the difference between pos and charIndex
758 *plX
= lPos
- lCharIndex
;
763 } // end of wxTextCtrl::PositionToXY
765 void wxTextCtrl::ShowPosition(
769 HWND hWnd
= GetHwnd();
770 long lCurrentLineLineNo
= 0L;
772 // To scroll to a position, we pass the number of lines and characters
773 // to scroll *by*. This means that we need to:
774 // (1) Find the line position of the current line.
775 // (2) Find the line position of pos.
776 // (3) Scroll by (pos - current).
777 // For now, ignore the horizontal scrolling.
780 // Is this where scrolling is relative to - the line containing the caret?
781 // Or is the first visible line??? Try first visible line.
786 // In PM this is the actual char position
788 lCurrentLineLineNo
= (long)::WinSendMsg(hWnd
, MLM_QUERYFIRSTCHAR
, (MPARAM
)0, (MPARAM
)0);
791 // This will cause a scroll to the selected position
793 ::WinSendMsg(hWnd
, MLM_SETSEL
, (MPARAM
)lCurrentLineLineNo
, (MPARAM
)lCurrentLineLineNo
);
795 } // end of wxTextCtrl::ShowPosition
797 int wxTextCtrl::GetLineLength(
804 lLen
= (long)::WinSendMsg(GetHwnd(), MLM_QUERYLINELENGTH
, 0, 0);
809 vParams
.fsStatus
= WPM_CCHTEXT
;
810 if (::WinSendMsg( GetHwnd()
811 ,WM_QUERYWINDOWPARAMS
816 lLen
= vParams
.cchText
;
822 } // end ofwxTextCtrl::GetLineLength
824 wxString
wxTextCtrl::GetLineText(
828 long lLen
= (long)GetLineLength((long)lLineNo
) + 1;
833 // There must be at least enough place for the length WORD in the
836 lLen
+= sizeof(WORD
);
837 zBuf
= new char[lLen
];
844 lLen
= (long)::WinSendMsg(GetHwnd(), MLM_QUERYLINELENGTH
, 0, 0);
845 lIndex
= lLen
* lLineNo
;
847 ::WinSendMsg(GetHwnd(), MLM_SETSEL
, (MPARAM
)lIndex
, (MPARAM
)lIndex
);
848 ::WinSendMsg(GetHwnd(), MLM_SETIMPORTEXPORT
, MPFROMP(zBuf
), MPFROMSHORT((USHORT
)sizeof(zBuf
)));
849 lBuflen
= (long)::WinSendMsg(GetHwnd(), MLM_QUERYFORMATTEXTLENGTH
, MPFROMLONG(lIndex
), MPFROMLONG(-1));
850 lCopied
= (long)::WinSendMsg(GetHwnd(), MLM_EXPORT
, MPFROMP(&lIndex
), MPFROMP(&lBuflen
));
851 zBuf
[lCopied
] = '\0';
857 vParams
.fsStatus
= WPM_CCHTEXT
;
858 if (::WinSendMsg( GetHwnd()
859 ,WM_QUERYWINDOWPARAMS
863 memcpy(zBuf
, vParams
.pszText
, vParams
.cchText
);
864 zBuf
[vParams
.cchText
] = '\0';
869 } // end of wxTextCtrl::GetLineText
871 // ----------------------------------------------------------------------------
873 // ----------------------------------------------------------------------------
875 void wxTextCtrl::Undo()
880 ::WinSendMsg(GetHwnd(), MLM_UNDO
, 0, 0);
881 // Simple entryfields cannot be undone
883 } // end of wxTextCtrl::Undo
885 void wxTextCtrl::Redo()
890 ::WinSendMsg(GetHwnd(), MLM_UNDO
, 0, 0);
891 // Simple entryfields cannot be undone
893 } // end of wxTextCtrl::Redo
895 bool wxTextCtrl::CanUndo() const
900 bOk
= (::WinSendMsg(GetHwnd(), MLM_QUERYUNDO
, 0, 0) != 0);
902 bOk
= FALSE
; // can't undo regular edit fields in PM
904 } // end of wxTextCtrl::CanUndo
906 bool wxTextCtrl::CanRedo() const
911 bOk
= (::WinSendMsg(GetHwnd(), MLM_QUERYUNDO
, 0, 0) != 0);
913 bOk
= FALSE
; // can't undo regular edit fields in PM
915 } // end of wxTextCtrl::CanRedo
917 // ----------------------------------------------------------------------------
918 // implemenation details
919 // ----------------------------------------------------------------------------
921 void wxTextCtrl::Command(
922 wxCommandEvent
& rEvent
925 SetValue(rEvent
.GetString());
926 ProcessCommand (rEvent
);
927 } // end of wxTextCtrl::Command
929 void wxTextCtrl::OnDropFiles(
930 wxDropFilesEvent
& rEvent
933 // By default, load the first file into the text window.
934 if (rEvent
.GetNumberOfFiles() > 0)
936 LoadFile(rEvent
.GetFiles()[0]);
938 } // end of wxTextCtrl::OnDropFiles
940 WXHBRUSH
wxTextCtrl::OnCtlColor(
949 HPS hPS
= (HPS
)hWxDC
;
950 wxBrush
* pBrush
= NULL
;
951 wxColour vColBack
= GetBackgroundColour();
952 wxColour vColFore
= GetForegroundColour();
953 wxBrush
* pBackgroundBrush
= wxTheBrushList
->FindOrCreateBrush( GetBackgroundColour()
959 HBRUSH hBrush
= NULLHANDLE
;
963 if (GetParent()->GetTransparentBackground())
964 ::GpiSetBackMix(hPS
, BM_LEAVEALONE
);
966 ::GpiSetBackMix(hPS
, BM_OVERPAINT
);
967 if (!IsEnabled() && (GetWindowStyle() & wxTE_MULTILINE
) == 0)
968 vColBack
= wxSystemSettings::GetSystemColour(wxSYS_COLOUR_3DFACE
);
969 ::GpiSetBackColor(hPS
, vColBack
.GetPixel());
970 ::GpiSetColor(hPS
, vColFore
.GetPixel());
971 return (WXHBRUSH
)pBackgroundBrush
->GetResourceHandle();
972 } // end of wxTextCtrl::OnCtlColor
974 void wxTextCtrl::OnChar(
978 switch (rEvent
.KeyCode())
981 if ( !(m_windowStyle
& wxTE_MULTILINE
) )
983 wxCommandEvent
vEvent(wxEVT_COMMAND_TEXT_ENTER
, m_windowId
);
985 vEvent
.SetEventObject(this);
986 if ( GetEventHandler()->ProcessEvent(vEvent
))
989 //else: multiline controls need Enter for themselves
994 // always produce navigation event - even if we process TAB
995 // ourselves the fact that we got here means that the user code
996 // decided to skip processing of this TAB - probably to let it
997 // do its default job.
999 // NB: Notice that Ctrl-Tab is handled elsewhere and Alt-Tab is
1000 // handled by Windows
1002 wxNavigationKeyEvent vEventNav
;
1004 vEventNav
.SetDirection(!rEvent
.ShiftDown());
1005 vEventNav
.SetWindowChange(FALSE
);
1006 vEventNav
.SetEventObject(this);
1008 if ( GetEventHandler()->ProcessEvent(vEventNav
) )
1014 } // end of wxTextCtrl::OnChar
1016 bool wxTextCtrl::OS2Command(
1018 , WXWORD
WXUNUSED(vId
)
1026 wxFocusEvent
vEvent( uParam
== EN_KILLFOCUS
? wxEVT_KILL_FOCUS
1031 vEvent
.SetEventObject(this);
1032 GetEventHandler()->ProcessEvent(vEvent
);
1038 wxCommandEvent
vEvent( wxEVT_COMMAND_TEXT_UPDATED
1042 InitCommandEvent(vEvent
);
1043 vEvent
.SetString((char*)GetValue().c_str());
1044 ProcessCommand(vEvent
);
1050 // The text size limit has been hit - increase it
1056 case EN_INSERTMODETOGGLE
:
1067 } // end of wxTextCtrl::OS2Command
1069 void wxTextCtrl::AdjustSpaceLimit()
1071 unsigned int uLen
= 0;
1072 unsigned int uLimit
= 0;
1074 uLen
= ::WinQueryWindowTextLength(GetHwnd());
1077 uLimit
= (unsigned int)::WinSendMsg( GetHwnd()
1088 vParams
.fsStatus
= WPM_CBCTLDATA
;
1089 vParams
.cbCtlData
= sizeof(ENTRYFDATA
);
1091 if (::WinSendMsg( GetHwnd()
1092 ,WM_QUERYWINDOWPARAMS
1097 pEfd
= (ENTRYFDATA
*)vParams
.pCtlData
;
1098 uLimit
= (unsigned int)pEfd
->cchEditLimit
;
1101 uLimit
= 32; //PM's default
1105 uLimit
= uLen
+ 0x8000; // 32Kb
1106 if (uLimit
> 0xffff)
1111 ::WinSendMsg(GetHwnd(), MLM_SETTEXTLIMIT
, MPFROMLONG(uLimit
), 0);
1113 ::WinSendMsg(GetHwnd(), EM_SETTEXTLIMIT
, MPFROMLONG(uLimit
), 0);
1115 } // end of wxTextCtrl::AdjustSpaceLimit
1117 bool wxTextCtrl::AcceptsFocus() const
1120 // We don't want focus if we can't be edited
1122 return IsEditable() && wxControl::AcceptsFocus();
1123 } // end of wxTextCtrl::Command
1125 wxSize
wxTextCtrl::DoGetBestSize() const
1130 wxGetCharSize(GetHWND(), &nCx
, &nCy
, (wxFont
*)&GetFont());
1132 int wText
= DEFAULT_ITEM_WIDTH
;
1133 int hText
= EDIT_HEIGHT_FROM_CHAR_HEIGHT(nCy
);
1135 if (m_windowStyle
& wxTE_MULTILINE
)
1137 hText
*= wxMin(GetNumberOfLines(), 5);
1139 //else: for single line control everything is ok
1140 return wxSize(wText
, hText
);
1141 } // end of wxTextCtrl::DoGetBestSize
1143 // ----------------------------------------------------------------------------
1144 // standard handlers for standard edit menu events
1145 // ----------------------------------------------------------------------------
1147 void wxTextCtrl::OnCut(
1148 wxCommandEvent
& rEvent
1152 } // end of wxTextCtrl::OnCut
1154 void wxTextCtrl::OnCopy(
1155 wxCommandEvent
& rEvent
1159 } // end of wxTextCtrl::OnCopy
1161 void wxTextCtrl::OnPaste(
1162 wxCommandEvent
& rEvent
1166 } // end of wxTextCtrl::OnPaste
1168 void wxTextCtrl::OnUndo(
1169 wxCommandEvent
& rEvent
1173 } // end of wxTextCtrl::OnUndo
1175 void wxTextCtrl::OnRedo(
1176 wxCommandEvent
& rEvent
1180 } // end of wxTextCtrl::OnRedo
1182 void wxTextCtrl::OnUpdateCut(
1183 wxUpdateUIEvent
& rEvent
1186 rEvent
.Enable(CanCut());
1187 } // end of wxTextCtrl::OnUpdateCut
1189 void wxTextCtrl::OnUpdateCopy(
1190 wxUpdateUIEvent
& rEvent
1193 rEvent
.Enable(CanCopy());
1194 } // end of wxTextCtrl::OnUpdateCopy
1196 void wxTextCtrl::OnUpdatePaste(
1197 wxUpdateUIEvent
& rEvent
1200 rEvent
.Enable(CanPaste());
1201 } // end of wxTextCtrl::OnUpdatePaste
1203 void wxTextCtrl::OnUpdateUndo(
1204 wxUpdateUIEvent
& rEvent
1207 rEvent
.Enable(CanUndo());
1208 } // end of wxTextCtrl::OnUpdateUndo
1210 void wxTextCtrl::OnUpdateRedo(
1211 wxUpdateUIEvent
& rEvent
1214 rEvent
.Enable(CanRedo());
1215 } // end of wxTextCtrl::OnUpdateRedo
1217 bool wxTextCtrl::SetBackgroundColour(
1218 const wxColour
& rColour
1222 ::WinSendMsg(GetHwnd(), MLM_SETBACKCOLOR
, (MPARAM
)rColour
.GetPixel(), MLE_INDEX
);
1224 } // end of wxTextCtrl::SetBackgroundColour
1226 bool wxTextCtrl::SetForegroundColour(
1227 const wxColour
& rColour
1231 ::WinSendMsg(GetHwnd(), MLM_SETTEXTCOLOR
, (MPARAM
)rColour
.GetPixel(), MLE_INDEX
);
1233 } // end of wxTextCtrl::SetForegroundColour
1235 bool wxTextCtrl::SetStyle(
1238 , const wxTextAttr
& rStyle
1241 HWND hWnd
= GetHwnd();
1252 // We can only change the format of the selection, so select the range we
1253 // want and restore the old selection later
1258 GetSelection( &lStartOld
1263 // But do we really have to change the selection?
1265 bool bChangeSel
= lStart
!= lStartOld
||
1271 ::WinSendMsg(hWnd
, MLM_SETSEL
, MPFROM2SHORT((USHORT
)lStart
, (USHORT
)lEnd
), 0);
1273 ::WinSendMsg(hWnd
, EM_SETSEL
, MPFROM2SHORT((USHORT
)lStart
, (USHORT
)lEnd
), 0);
1277 // TODO:: finish this part
1280 } // end of wxTextCtrl::SetStyle