]> git.saurik.com Git - wxWidgets.git/blame - src/os2/textctrl.cpp
Fixed idle event processing in wxMotif.
[wxWidgets.git] / src / os2 / textctrl.cpp
CommitLineData
0e320a79
DW
1/////////////////////////////////////////////////////////////////////////////
2// Name: textctrl.cpp
3// Purpose: wxTextCtrl
d90895ac 4// Author: David Webster
0e320a79 5// Modified by:
d90895ac 6// Created: 10/17/99
0e320a79 7// RCS-ID: $Id$
d90895ac
DW
8// Copyright: (c) David Webster
9// Licence: wxWindows licence
0e320a79
DW
10/////////////////////////////////////////////////////////////////////////////
11
d90895ac
DW
12// ----------------------------------------------------------------------------
13// headers
14// ----------------------------------------------------------------------------
15
16// For compilers that support precompilation, includes "wx.h".
17#include "wx/wxprec.h"
18
19#ifndef WX_PRECOMP
20 #include "wx/textctrl.h"
19193a2c 21 #include "wx/scrolwin.h"
d90895ac
DW
22 #include "wx/settings.h"
23 #include "wx/brush.h"
24 #include "wx/utils.h"
25 #include "wx/log.h"
0e320a79
DW
26#endif
27
d90895ac
DW
28#if wxUSE_CLIPBOARD
29 #include "wx/app.h"
30 #include "wx/clipbrd.h"
31#endif
0e320a79 32
d90895ac
DW
33#include "wx/textfile.h"
34
35#include "wx/os2/private.h"
36
37#include <string.h>
38#include <stdlib.h>
39#include <sys/types.h>
0e320a79 40
d90895ac
DW
41#if wxUSE_IOSTREAMH
42# include <fstream.h>
0e320a79 43#else
d90895ac 44# include <fstream>
0e320a79
DW
45#endif
46
19193a2c 47#if !defined(MLE_INDEX)
a92d9721
SN
48#define MLE_INDEX 0
49#define MLE_RGB 1
50#endif
51
d90895ac
DW
52
53// ----------------------------------------------------------------------------
54// event tables and other macros
55// ----------------------------------------------------------------------------
56
0e320a79
DW
57IMPLEMENT_DYNAMIC_CLASS(wxTextCtrl, wxControl)
58
59BEGIN_EVENT_TABLE(wxTextCtrl, wxControl)
d90895ac
DW
60 EVT_CHAR(wxTextCtrl::OnChar)
61 EVT_DROP_FILES(wxTextCtrl::OnDropFiles)
62
0e320a79
DW
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)
68
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)
74END_EVENT_TABLE()
0e320a79 75
d90895ac
DW
76
77// ============================================================================
78// implementation
79// ============================================================================
80
81// ----------------------------------------------------------------------------
82// creation
83// ----------------------------------------------------------------------------
84
0e320a79 85wxTextCtrl::wxTextCtrl()
0e320a79 86{
0e320a79
DW
87}
88
2b5f62a0
VZ
89wxTextCtrl::~wxTextCtrl()
90{
91}
92
3bd418ca
DW
93bool wxTextCtrl::Create(
94 wxWindow* pParent
95, wxWindowID vId
96, const wxString& rsValue
97, const wxPoint& rPos
98, const wxSize& rSize
99, long lStyle
3bd418ca 100, const wxValidator& rValidator
3bd418ca
DW
101, const wxString& rsName
102)
103{
39c26ef2
DW
104 HWND hParent;
105 int nTempy;
cfcebdb1 106
3bd418ca
DW
107 //
108 // Base initialization
109 //
110 if ( !CreateBase( pParent
111 ,vId
112 ,rPos
113 ,rSize
114 ,lStyle
3bd418ca 115 ,rValidator
3bd418ca
DW
116 ,rsName
117 ))
d90895ac 118 return FALSE;
0e320a79 119
45bfc84b 120 wxPoint vPos = rPos; // The OS/2 position
5d44b24e 121 SWP vSwp;
45bfc84b 122
3bd418ca 123 if (pParent )
45bfc84b 124 {
3bd418ca 125 pParent->AddChild(this);
45bfc84b 126 }
0e320a79 127
3bd418ca 128 m_windowStyle = lStyle;
70a2c656 129 m_bIsMLE = FALSE;
d90895ac 130
3bd418ca
DW
131 long lSstyle = WS_VISIBLE | WS_TABSTOP;
132
133 //
134 // Single and multiline edit fields are two different controls in PM
135 //
d90895ac
DW
136 if ( m_windowStyle & wxTE_MULTILINE )
137 {
9c88ff79 138 lSstyle |= MLS_BORDER | MLS_WORDWRAP;
3bd418ca 139 m_bIsMLE = TRUE;
d90895ac 140
d90895ac 141 if ((m_windowStyle & wxTE_NO_VSCROLL) == 0)
3bd418ca
DW
142 lSstyle |= MLS_VSCROLL;
143 if (m_windowStyle & wxHSCROLL)
144 lSstyle |= MLS_HSCROLL;
145 if (m_windowStyle & wxTE_READONLY)
146 lSstyle |= MLS_READONLY;
d90895ac 147 }
0e320a79 148 else
3bd418ca 149 {
987da0d4 150 lSstyle |= ES_LEFT | ES_AUTOSCROLL | ES_MARGIN;
3bd418ca
DW
151
152 if (m_windowStyle & wxHSCROLL)
153 lSstyle |= ES_AUTOSCROLL;
154 if (m_windowStyle & wxTE_READONLY)
155 lSstyle |= ES_READONLY;
156 if (m_windowStyle & wxTE_PASSWORD) // hidden input
157 lSstyle |= ES_UNREADABLE;
158 }
5d44b24e 159
3bd418ca
DW
160 if (m_bIsMLE)
161 {
162 m_hWnd = (WXHWND)::WinCreateWindow( (HWND)GetHwndOf(pParent) // Parent window handle
163 ,WC_MLE // Window class
164 ,(PSZ)rsValue.c_str() // Initial Text
165 ,(ULONG)lSstyle // Style flags
716ed225
DW
166 ,(LONG)0 // X pos of origin
167 ,(LONG)0 // Y pos of origin
168 ,(LONG)0 // field width
169 ,(LONG)0 // field height
3bd418ca
DW
170 ,(HWND)GetHwndOf(pParent) // owner window handle (same as parent
171 ,HWND_TOP // initial z position
172 ,(ULONG)vId // Window identifier
173 ,NULL // no control data
174 ,NULL // no Presentation parameters
175 );
176 }
177 else
178 {
179 m_hWnd = (WXHWND)::WinCreateWindow( (HWND)GetHwndOf(pParent) // Parent window handle
180 ,WC_ENTRYFIELD // Window class
181 ,(PSZ)rsValue.c_str() // Initial Text
182 ,(ULONG)lSstyle // Style flags
716ed225
DW
183 ,(LONG)0 // X pos of origin
184 ,(LONG)0 // Y pos of origin
185 ,(LONG)0 // field width
186 ,(LONG)0 // field height
3bd418ca
DW
187 ,(HWND)GetHwndOf(pParent) // owner window handle (same as parent
188 ,HWND_TOP // initial z position
189 ,(ULONG)vId // Window identifier
190 ,NULL // no control data
191 ,NULL // no Presentation parameters
192 );
193 }
194
195 if (m_hWnd == 0)
196 {
197 return FALSE;
198 }
199
d90895ac
DW
200 SubclassWin(GetHWND());
201
3bd418ca
DW
202 //
203 // Set font, position, size and initial value
204 //
70a2c656 205 wxFont* pTextFont = new wxFont( 8
2c1e8f2e
DW
206 ,wxMODERN
207 ,wxNORMAL
208 ,wxNORMAL
209 );
210 SetFont(*pTextFont);
3bd418ca
DW
211 if (!rsValue.IsEmpty())
212 {
213 SetValue(rsValue);
214 }
215 SetupColours();
5d44b24e
DW
216 //
217 // If X and/or Y are not zero the difference is the compensation value
218 // for margins for OS/2 controls.
219 //
220 ::WinQueryWindowPos(m_hWnd, &vSwp);
221 SetXComp(vSwp.x);
222 SetYComp(vSwp.y);
70a2c656
DW
223 SetSize( vPos.x - GetXComp()
224 ,vPos.y - GetYComp()
3bd418ca
DW
225 ,rSize.x
226 ,rSize.y
227 );
b3260bce 228 delete pTextFont;
0e320a79 229 return TRUE;
3bd418ca 230} // end of wxTextCtrl::Create
0e320a79 231
3bd418ca 232//
d90895ac 233// Make sure the window style (etc.) reflects the HWND style (roughly)
3bd418ca 234//
d90895ac
DW
235void wxTextCtrl::AdoptAttributesFromHWND()
236{
3bd418ca
DW
237 HWND hWnd = GetHwnd();
238 LONG lStyle = ::WinQueryWindowULong(hWnd, QWL_STYLE);
d90895ac 239
3bd418ca 240 wxWindow::AdoptAttributesFromHWND();
d90895ac 241
3bd418ca
DW
242 if (m_bIsMLE)
243 {
244 m_windowStyle |= wxTE_MULTILINE;
245 if (lStyle & MLS_READONLY)
246 m_windowStyle |= wxTE_READONLY;
247 }
248 else
249 {
250 if (lStyle & ES_UNREADABLE)
251 m_windowStyle |= wxTE_PASSWORD;
252 if (lStyle & ES_READONLY)
253 m_windowStyle |= wxTE_READONLY;
254 }
c4955899 255} // end of wxTextCtrl::AdoptAttributesFromHWND
d90895ac 256
b9b1d6c8
DW
257WXDWORD wxTextCtrl::OS2GetStyle(
258 long lStyle
259, WXDWORD* pdwExstyle
260) const
261{
262 //
263 // Default border for the text controls is the sunken one
264 //
265 if ((lStyle & wxBORDER_MASK) == wxBORDER_DEFAULT )
266 {
267 lStyle |= wxBORDER_SUNKEN;
268 }
269
270 long dwStyle = wxControl::OS2GetStyle( lStyle
271 ,pdwExstyle
272 );
273
274 dwStyle = WS_VISIBLE | WS_TABSTOP;
275
276 //
277 // Single and multiline edit fields are two different controls in PM
278 //
279 if ( m_windowStyle & wxTE_MULTILINE )
280 {
281 dwStyle |= MLS_BORDER | MLS_WORDWRAP;
282 if ((m_windowStyle & wxTE_NO_VSCROLL) == 0)
283 dwStyle |= MLS_VSCROLL;
284 if (m_windowStyle & wxHSCROLL)
285 dwStyle |= MLS_HSCROLL;
286 if (m_windowStyle & wxTE_READONLY)
287 dwStyle |= MLS_READONLY;
288 }
289 else
290 {
291 dwStyle |= ES_LEFT | ES_AUTOSCROLL | ES_MARGIN;
292 if (m_windowStyle & wxHSCROLL)
293 dwStyle |= ES_AUTOSCROLL;
294 if (m_windowStyle & wxTE_READONLY)
295 dwStyle |= ES_READONLY;
296 if (m_windowStyle & wxTE_PASSWORD) // hidden input
297 dwStyle |= ES_UNREADABLE;
298 }
299 return dwStyle;
300} // end of wxTextCtrl::OS2GetStyle
301
302void wxTextCtrl::SetWindowStyleFlag(
303 long lStyle
304)
305{
306 wxControl::SetWindowStyleFlag(lStyle);
307} // end of wxTextCtrl::SetWindowStyleFlag
308
d90895ac
DW
309void wxTextCtrl::SetupColours()
310{
c4955899
DW
311 wxColour vBkgndColour;
312
a756f210 313 vBkgndColour = wxSystemSettings::GetColour(wxSYS_COLOUR_WINDOW);
c4955899 314 SetBackgroundColour(vBkgndColour);
d90895ac 315 SetForegroundColour(GetParent()->GetForegroundColour());
9c88ff79
DW
316 if (m_bIsMLE)
317 {
318 ::WinSendMsg( GetHwnd()
319 ,MLM_SETTEXTCOLOR
320 ,(MPARAM)GetParent()->GetForegroundColour().GetPixel()
321 ,(MPARAM)MLE_RGB
322 );
323 }
c4955899 324} // end of wxTextCtrl::SetupColours
d90895ac
DW
325
326// ----------------------------------------------------------------------------
327// set/get the controls text
328// ----------------------------------------------------------------------------
329
0e320a79
DW
330wxString wxTextCtrl::GetValue() const
331{
c4955899
DW
332 wxString sStr = wxGetWindowText(GetHWND());
333 char* zStr = (char*)sStr.c_str();
d90895ac 334
c4955899
DW
335 for ( ; *zStr; zStr++ )
336 {
337 //
338 // this will replace \r\n with just \n
339 //
340 if (*zStr == '\n')
341 *zStr = '\0';
342 if (*zStr == '\r')
343 *zStr = '\n';
344 }
c4955899
DW
345 return sStr;
346} // end of wxTextCtrl::GetValue
0e320a79 347
c4955899
DW
348void wxTextCtrl::SetValue(
349 const wxString& rsValue
350)
0e320a79 351{
c4955899
DW
352 //
353 // If the text is long enough, it's faster to just set it instead of first
354 // comparing it with the old one (chances are that it will be different
355 // anyhow, this comparison is there to avoid flicker for small single-line
356 // edit controls mostly)
357 //
358 if ((rsValue.length() > 0x400) || (rsValue != GetValue()))
359 {
360 ::WinSetWindowText(GetHwnd(), rsValue.c_str());
361 AdjustSpaceLimit();
362 }
363} // end of wxTextCtrl::SetValue
d90895ac 364
c4955899
DW
365void wxTextCtrl::WriteText(
366 const wxString& rsValue
367)
368{
b3260bce
DW
369 if (m_bIsMLE)
370 ::WinSendMsg(GetHwnd(), MLM_INSERT, MPARAM((PCHAR)rsValue.c_str()), MPARAM(0));
371 else
372 ::WinSetWindowText(GetHwnd(), rsValue.c_str());
d90895ac 373 AdjustSpaceLimit();
c4955899 374} // end of wxTextCtrl::WriteText
0e320a79 375
c4955899
DW
376void wxTextCtrl::AppendText(
377 const wxString& rsText
378)
d90895ac 379{
d90895ac 380 SetInsertionPointEnd();
c4955899
DW
381 WriteText(rsText);
382} // end of wxTextCtrl::AppendText
d90895ac
DW
383
384void wxTextCtrl::Clear()
385{
c4955899
DW
386 ::WinSetWindowText(GetHwnd(), "");
387} // end of wxTextCtrl::Clear
d90895ac 388
a086de98
DW
389bool wxTextCtrl::EmulateKeyPress(
390 const wxKeyEvent& rEvent
391)
392{
393 SetFocus();
394 return(wxTextCtrlBase::EmulateKeyPress(rEvent));
395} // end of wxTextCtrl::EmulateKeyPress
396
d90895ac 397// ----------------------------------------------------------------------------
0e320a79 398// Clipboard operations
d90895ac
DW
399// ----------------------------------------------------------------------------
400
0e320a79
DW
401void wxTextCtrl::Copy()
402{
d90895ac
DW
403 if (CanCopy())
404 {
405 HWND hWnd = GetHwnd();
3bd418ca
DW
406 if (m_bIsMLE)
407 ::WinSendMsg(hWnd, MLM_COPY, 0, 0);
408 else
409 ::WinSendMsg(hWnd, EM_COPY, 0, 0);
d90895ac 410 }
3bd418ca 411} // end of wxTextCtrl::Copy
0e320a79
DW
412
413void wxTextCtrl::Cut()
414{
d90895ac
DW
415 if (CanCut())
416 {
417 HWND hWnd = GetHwnd();
3bd418ca
DW
418
419 if (m_bIsMLE)
420 ::WinSendMsg(hWnd, MLM_CUT, 0, 0);
421 else
422 ::WinSendMsg(hWnd, EM_CUT, 0, 0);
d90895ac 423 }
c4955899 424} // end of wxTextCtrl::Cut
0e320a79
DW
425
426void wxTextCtrl::Paste()
427{
d90895ac
DW
428 if (CanPaste())
429 {
c4955899
DW
430 HWND hWnd = GetHwnd();
431
432 ::WinSendMsg(hWnd, EM_PASTE, 0, 0);
d90895ac 433 }
c4955899 434} // end of wxTextCtrl::Paste
d90895ac
DW
435
436bool wxTextCtrl::CanCopy() const
437{
c4955899 438 //
d90895ac 439 // Can copy if there's a selection
c4955899
DW
440 //
441 long lFrom = 0L;
442 long lTo = 0L;
443
444 GetSelection(&lFrom, &lTo);
445 return (lFrom != lTo);
446} // end of wxTextCtrl::CanCopy
d90895ac
DW
447
448bool wxTextCtrl::CanCut() const
449{
c4955899 450 //
d90895ac 451 // Can cut if there's a selection
c4955899
DW
452 //
453 long lFrom = 0L;
454 long lTo = 0L;
455
456 GetSelection(&lFrom, &lTo);
457 return (lFrom != lTo);
458} // end of wxTextCtrl::CanCut
d90895ac
DW
459
460bool wxTextCtrl::CanPaste() const
461{
3bd418ca
DW
462 bool bIsTextAvailable = FALSE;
463
d90895ac
DW
464 if (!IsEditable())
465 return FALSE;
466
3bd418ca
DW
467 //
468 // Check for straight text on clipboard
469 //
470 if (::WinOpenClipbrd(vHabmain))
d90895ac 471 {
3bd418ca
DW
472 bIsTextAvailable = (::WinQueryClipbrdData(vHabmain, CF_TEXT) != 0);
473 ::WinCloseClipbrd(vHabmain);
d90895ac 474 }
3bd418ca
DW
475 return bIsTextAvailable;
476} // end of wxTextCtrl::CanPaste
0e320a79 477
d90895ac
DW
478// ----------------------------------------------------------------------------
479// Accessors
480// ----------------------------------------------------------------------------
481
c4955899
DW
482void wxTextCtrl::SetEditable(
483 bool bEditable
484)
0e320a79 485{
c4955899
DW
486 HWND hWnd = GetHwnd();
487
488 if (m_bIsMLE)
489 ::WinSendMsg(hWnd, MLM_SETREADONLY, MPFROMLONG(!bEditable), (MPARAM)0);
490 else
491 ::WinSendMsg(hWnd, EM_SETREADONLY, MPFROMLONG(!bEditable), (MPARAM)0);
492} // end of wxTextCtrl::SetEditable
0e320a79 493
c4955899
DW
494void wxTextCtrl::SetInsertionPoint(
495 long lPos
496)
0e320a79 497{
c4955899
DW
498 HWND hWnd = GetHwnd();
499
500 if (m_bIsMLE)
501 ::WinSendMsg(hWnd, MLM_SETSEL, (MPARAM)lPos, (MPARAM)lPos);
502 else
503 ::WinSendMsg(hWnd, EM_SETSEL, MPFROM2SHORT((USHORT)lPos, (USHORT)lPos), (MPARAM)0);
504} // end of wxTextCtrl::SetInsertionPoint
0e320a79
DW
505
506void wxTextCtrl::SetInsertionPointEnd()
507{
c4955899
DW
508 long lPos = GetLastPosition();
509
6e348b12
DW
510 //
511 // We must not do anything if the caret is already there because calling
512 // SetInsertionPoint() thaws the controls if Freeze() had been called even
513 // if it doesn't actually move the caret anywhere and so the simple fact of
514 // doing it results in horrible flicker when appending big amounts of text
515 // to the control in a few chunks (see DoAddText() test in the text sample)
516 //
517 if (GetInsertionPoint() == GetLastPosition())
518 return;
c4955899
DW
519 SetInsertionPoint(lPos);
520} // end of wxTextCtrl::SetInsertionPointEnd
0e320a79
DW
521
522long wxTextCtrl::GetInsertionPoint() const
523{
3bd418ca
DW
524 WXDWORD dwPos = 0L;
525
526 if (m_bIsMLE)
527 dwPos = (WXDWORD)::WinSendMsg(GetHwnd(), MLM_QUERYSEL, (MPARAM)MLFQS_MINSEL, 0);
528 else
529 {
530 dwPos = (WXDWORD)::WinSendMsg(GetHwnd(), EM_QUERYSEL, 0, 0);
531 dwPos = SHORT1FROMMP((MPARAM)dwPos); // the first 16 bit value is the min pos
532 }
533 return (dwPos & 0xFFFF);
534} // end of wxTextCtrl::GetInsertionPoint
0e320a79
DW
535
536long wxTextCtrl::GetLastPosition() const
537{
3bd418ca
DW
538 HWND hWnd = GetHwnd();
539 long lCharIndex;
540 long lLineLength;
0e320a79 541
3bd418ca
DW
542 if (m_bIsMLE)
543 {
544 lCharIndex = 0;
d90895ac 545
3bd418ca
DW
546 //
547 // This just gets the total text length. The last will be this value
548 //
549 lLineLength = (long)::WinSendMsg(hWnd, MLM_QUERYTEXTLENGTH, 0, 0);
550 }
551 else
552 {
553 WNDPARAMS vParams;
554
555 lCharIndex = 0;
556 vParams.fsStatus = WPM_CCHTEXT;
557 if (::WinSendMsg( GetHwnd()
558 ,WM_QUERYWINDOWPARAMS
559 ,&vParams
560 ,0
561 ))
562 {
563 lLineLength = (long)vParams.cchText;
564 }
565 else
566 lLineLength = 0;
567 }
568 return(lCharIndex + lLineLength);
569} // end of wxTextCtrl::GetLastPosition
0e320a79 570
d90895ac
DW
571// If the return values from and to are the same, there is no
572// selection.
c4955899
DW
573void wxTextCtrl::GetSelection(
574 long* plFrom
575, long* plTo
576) const
0e320a79 577{
c4955899 578 WXDWORD dwPos;
0e320a79 579
c4955899
DW
580 if (m_bIsMLE)
581 dwPos = (WXDWORD)::WinSendMsg(GetHwnd(), MLM_QUERYSEL, (MPARAM)MLFQS_MINSEL, 0);
582 else
583 {
584 dwPos = (WXDWORD)::WinSendMsg(GetHwnd(), EM_QUERYSEL, 0, 0);
585 }
586 *plFrom = SHORT1FROMMP((MPARAM)dwPos); // the first 16 bit value is the min pos
587 *plTo = SHORT2FROMMP((MPARAM)dwPos); // the first 16 bit value is the min pos
588} // end of wxTextCtrl::GetSelection
0e320a79 589
d90895ac
DW
590bool wxTextCtrl::IsEditable() const
591{
c4955899
DW
592 if (m_bIsMLE)
593 return((bool)LONGFROMMR(::WinSendMsg(GetHwnd(), MLM_QUERYREADONLY, 0, 0)));
594 else
595 return((bool)LONGFROMMR(::WinSendMsg(GetHwnd(), EM_QUERYREADONLY, 0, 0)));
596} // end of wxTextCtrl::IsEditable
0e320a79 597
d90895ac
DW
598// ----------------------------------------------------------------------------
599// Editing
600// ----------------------------------------------------------------------------
601
c4955899
DW
602void wxTextCtrl::Replace(
603 long lFrom
604, long lTo
605, const wxString& rsValue
606)
0e320a79 607{
d90895ac 608#if wxUSE_CLIPBOARD
c4955899
DW
609 HWND hWnd = GetHwnd();
610 long lFromChar = lFrom;
611 long lToChar = lTo;
0e320a79 612
c4955899 613 //
d90895ac 614 // Set selection and remove it
c4955899
DW
615 //
616 if (m_bIsMLE)
617 {
618 ::WinSendMsg(hWnd, MLM_SETSEL, MPFROM2SHORT((USHORT)lFrom, (USHORT)lTo), 0);
619 ::WinSendMsg(hWnd, MLM_CUT, 0, 0);
620 }
621 else
622 {
623 ::WinSendMsg(hWnd, EM_SETSEL, MPFROM2SHORT((USHORT)lFrom, (USHORT)lTo), 0);
624 ::WinSendMsg(hWnd, EM_CUT, 0, 0);
625 }
0e320a79 626
c4955899 627 //
d90895ac 628 // Now replace with 'value', by pasting.
c4955899
DW
629 //
630 wxSetClipboardData(wxDF_TEXT, (wxObject *) (const wxChar *)rsValue, 0, 0);
0e320a79 631
d90895ac 632 // Paste into edit control
c4955899
DW
633 if (m_bIsMLE)
634 ::WinSendMsg(hWnd, MLM_PASTE, (MPARAM)0, (MPARAM)0);
635 else
636 ::WinSendMsg(hWnd, EM_PASTE, (MPARAM)0, (MPARAM)0);
d90895ac
DW
637#else
638 wxFAIL_MSG("wxTextCtrl::Replace not implemented if wxUSE_CLIPBOARD is 0.");
639#endif
c4955899 640} // end of wxTextCtrl::Replace
0e320a79 641
c4955899
DW
642void wxTextCtrl::Remove(
643 long lFrom
644, long lTo
645)
0e320a79 646{
c4955899
DW
647 HWND hWnd = GetHwnd();
648 long lFromChar = lFrom;
649 long lToChar = lTo;
d90895ac 650
c4955899
DW
651 if (m_bIsMLE)
652 {
653 ::WinSendMsg(hWnd, MLM_SETSEL, MPFROM2SHORT((USHORT)lFrom, (USHORT)lTo), 0);
654 ::WinSendMsg(hWnd, MLM_CUT, 0, 0);
655 }
656 else
657 {
658 ::WinSendMsg(hWnd, EM_SETSEL, MPFROM2SHORT((USHORT)lFrom, (USHORT)lTo), 0);
659 ::WinSendMsg(hWnd, EM_CUT, 0, 0);
660 }
661} // end of wxTextCtrl::Remove
0e320a79 662
c4955899
DW
663void wxTextCtrl::SetSelection(
664 long lFrom
665, long lTo
666)
0e320a79 667{
c4955899
DW
668 HWND hWnd = GetHwnd();
669 long lFromChar = lFrom;
670 long lToChar = lTo;
d90895ac 671
c4955899
DW
672 //
673 // If from and to are both -1, it means (in wxWindows) that all text should
d90895ac 674 // be selected. Translate into Windows convention
c4955899
DW
675 //
676 if ((lFrom == -1L) && (lTo == -1L))
d90895ac 677 {
c4955899
DW
678 lFromChar = 0L;
679 lToChar = -1L;
d90895ac 680 }
c4955899
DW
681 if (m_bIsMLE)
682 ::WinSendMsg(hWnd, MLM_SETSEL, (MPARAM)lFromChar, (MPARAM)lToChar);
683 else
684 ::WinSendMsg(hWnd, EM_SETSEL, MPFROM2SHORT((USHORT)lFromChar, (USHORT)lToChar), (MPARAM)0);
685} // end of wxTextCtrl::SetSelection
d90895ac 686
c4955899
DW
687bool wxTextCtrl::LoadFile(
688 const wxString& rsFile
689)
0e320a79 690{
c4955899 691 if ( wxTextCtrlBase::LoadFile(rsFile) )
d90895ac 692 {
c4955899
DW
693 //
694 // Update the size limit if needed
695 //
d90895ac 696 AdjustSpaceLimit();
d90895ac
DW
697 return TRUE;
698 }
d90895ac 699 return FALSE;
c4955899 700} // end of wxTextCtrl::LoadFile
0e320a79
DW
701
702bool wxTextCtrl::IsModified() const
703{
c4955899
DW
704 bool bRc;
705
706 if (m_bIsMLE)
707 bRc = (bool)LONGFROMMR(::WinSendMsg(GetHwnd(), MLM_QUERYCHANGED, 0, 0));
708 else
709 bRc = (bool)LONGFROMMR(::WinSendMsg(GetHwnd(), EM_QUERYCHANGED, 0, 0));
710 return bRc;
711} // end of wxTextCtrl::IsModified
0e320a79 712
3a9fa0d6
VZ
713void wxTextCtrl::MarkDirty()
714{
b21ec1a7
SN
715 if (m_bIsMLE)
716 ::WinSendMsg(GetHwnd(), MLM_SETCHANGED, MPFROMLONG(TRUE), 0);
717 else
718 // EM controls do not have a SETCHANGED, what can we do??
719 wxFAIL_MSG( _T("not implemented") );
3a9fa0d6
VZ
720}
721
3bd418ca 722//
0e320a79 723// Makes 'unmodified'
3bd418ca 724//
0e320a79
DW
725void wxTextCtrl::DiscardEdits()
726{
3bd418ca
DW
727 if (m_bIsMLE)
728 ::WinSendMsg(GetHwnd(), MLM_SETCHANGED, MPFROMLONG(FALSE), 0);
729 else
730 //
731 // EM controls do not have a SETCHANGED but issuing a query should reset it
732 //
733 ::WinSendMsg(GetHwnd(), EM_QUERYCHANGED, 0, 0);
734} // end of wxTextCtrl::DiscardEdits
0e320a79
DW
735
736int wxTextCtrl::GetNumberOfLines() const
737{
c4955899 738 int nNumLines;
0e320a79 739
c4955899
DW
740 if (m_bIsMLE)
741 nNumLines = (int)::WinSendMsg(GetHwnd(), MLM_QUERYLINECOUNT, 0, 0);
742 else
743 nNumLines = 1;
744 return nNumLines;
745} // end of wxTextCtrl::GetNumberOfLines
d90895ac 746
c4955899
DW
747long wxTextCtrl::XYToPosition(
748 long lX
749, long lY
750) const
751{
752 HWND hWnd = GetHwnd();
753 long lCharIndex = 0L;
754 long lLen;
0e320a79 755
c4955899
DW
756 if (m_bIsMLE)
757 {
758 lLen = (long)::WinSendMsg(GetHwnd(), MLM_QUERYLINELENGTH, 0, 0);
759 lCharIndex = ((lLen * lY) + lX);
760 }
761 else
762 lCharIndex = lX;
763 return lCharIndex;
764} // end of wxTextCtrl::XYToPosition
765
766bool wxTextCtrl::PositionToXY(
767 long lPos
768, long* plX
769, long* plY
770) const
0e320a79 771{
c4955899
DW
772 HWND hWnd = GetHwnd();
773 long nLineNo = -1;
774 long lCharIndex = 0;
d90895ac 775
c4955899
DW
776 if (m_bIsMLE)
777 nLineNo = (long)::WinSendMsg(hWnd, MLM_LINEFROMCHAR, (MPARAM)lPos, 0);
778 else
779 nLineNo = 0;
d90895ac 780
c4955899 781 if (nLineNo == -1)
d90895ac
DW
782 {
783 // no such line
784 return FALSE;
785 }
786
c4955899 787 //
d90895ac 788 // This gets the char index for the _beginning_ of this line
c4955899
DW
789 //
790 long lLineWidth;
791
792 if (m_bIsMLE)
793 {
794 lLineWidth = (long)::WinSendMsg(hWnd, MLM_QUERYLINELENGTH, (MPARAM)0, (MPARAM)0);
795 lCharIndex = (nLineNo + 1) * lLineWidth;
796 }
797 else
798 {
799 WNDPARAMS vParams;
800
801 vParams.fsStatus = WPM_CCHTEXT;
802 if (::WinSendMsg( hWnd
803 ,WM_QUERYWINDOWPARAMS
804 ,&vParams
805 ,0
806 ))
807 {
808 lCharIndex = vParams.cchText;
809 }
810 else
811 lCharIndex = 32;
812 }
813
814 if (lCharIndex == -1)
d90895ac
DW
815 {
816 return FALSE;
817 }
818
c4955899
DW
819 //
820 // The X position must therefore be the difference between pos and charIndex
821 //
822 if (plX)
823 *plX = lPos - lCharIndex;
824 if (plY)
825 *plY = nLineNo;
d90895ac
DW
826
827 return TRUE;
c4955899 828} // end of wxTextCtrl::PositionToXY
0e320a79 829
c4955899
DW
830void wxTextCtrl::ShowPosition(
831 long lPos
832)
0e320a79 833{
c4955899
DW
834 HWND hWnd = GetHwnd();
835 long lCurrentLineLineNo = 0L;
d90895ac
DW
836
837 // To scroll to a position, we pass the number of lines and characters
838 // to scroll *by*. This means that we need to:
839 // (1) Find the line position of the current line.
840 // (2) Find the line position of pos.
841 // (3) Scroll by (pos - current).
842 // For now, ignore the horizontal scrolling.
843
c4955899 844 //
d90895ac
DW
845 // Is this where scrolling is relative to - the line containing the caret?
846 // Or is the first visible line??? Try first visible line.
c4955899
DW
847 //
848 if (m_bIsMLE)
849 {
850 //
851 // In PM this is the actual char position
852 //
853 lCurrentLineLineNo = (long)::WinSendMsg(hWnd, MLM_QUERYFIRSTCHAR, (MPARAM)0, (MPARAM)0);
d90895ac 854
c4955899
DW
855 //
856 // This will cause a scroll to the selected position
857 //
858 ::WinSendMsg(hWnd, MLM_SETSEL, (MPARAM)lCurrentLineLineNo, (MPARAM)lCurrentLineLineNo);
859 }
860} // end of wxTextCtrl::ShowPosition
0e320a79 861
3bd418ca
DW
862int wxTextCtrl::GetLineLength(
863 long lLineNo
864) const
0e320a79 865{
3bd418ca 866 long lLen = 0L;
d90895ac 867
3bd418ca
DW
868 if (m_bIsMLE)
869 lLen = (long)::WinSendMsg(GetHwnd(), MLM_QUERYLINELENGTH, 0, 0);
870 else
871 {
872 WNDPARAMS vParams;
873
874 vParams.fsStatus = WPM_CCHTEXT;
875 if (::WinSendMsg( GetHwnd()
876 ,WM_QUERYWINDOWPARAMS
877 ,&vParams
878 ,0
879 ))
880 {
881 lLen = vParams.cchText;
882 }
883 else
884 lLen = 32;
885 }
886 return lLen;
c4955899 887} // end ofwxTextCtrl::GetLineLength
0e320a79 888
c4955899
DW
889wxString wxTextCtrl::GetLineText(
890 long lLineNo
891) const
0e320a79 892{
c4955899
DW
893 long lLen = (long)GetLineLength((long)lLineNo) + 1;
894 wxString sStr;
895 char* zBuf;
0e320a79 896
c4955899
DW
897 //
898 // There must be at least enough place for the length WORD in the
899 // buffer
900 //
901 lLen += sizeof(WORD);
902 zBuf = new char[lLen];
903 if (m_bIsMLE)
904 {
905 long lIndex;
906 long lBuflen;
907 long lCopied;
0e320a79 908
c4955899
DW
909 lLen = (long)::WinSendMsg(GetHwnd(), MLM_QUERYLINELENGTH, 0, 0);
910 lIndex = lLen * lLineNo;
0e320a79 911
c4955899
DW
912 ::WinSendMsg(GetHwnd(), MLM_SETSEL, (MPARAM)lIndex, (MPARAM)lIndex);
913 ::WinSendMsg(GetHwnd(), MLM_SETIMPORTEXPORT, MPFROMP(zBuf), MPFROMSHORT((USHORT)sizeof(zBuf)));
914 lBuflen = (long)::WinSendMsg(GetHwnd(), MLM_QUERYFORMATTEXTLENGTH, MPFROMLONG(lIndex), MPFROMLONG(-1));
915 lCopied = (long)::WinSendMsg(GetHwnd(), MLM_EXPORT, MPFROMP(&lIndex), MPFROMP(&lBuflen));
916 zBuf[lCopied] = '\0';
917 }
918 else
919 {
920 WNDPARAMS vParams;
921
922 vParams.fsStatus = WPM_CCHTEXT;
923 if (::WinSendMsg( GetHwnd()
924 ,WM_QUERYWINDOWPARAMS
925 ,&vParams
926 ,0
927 ))
928 memcpy(zBuf, vParams.pszText, vParams.cchText);
929 zBuf[vParams.cchText] = '\0';
930 }
931 sStr = zBuf;
932 delete [] zBuf;
933 return sStr;
934} // end of wxTextCtrl::GetLineText
0e320a79 935
d90895ac 936// ----------------------------------------------------------------------------
0e320a79 937// Undo/redo
d90895ac
DW
938// ----------------------------------------------------------------------------
939
0e320a79
DW
940void wxTextCtrl::Undo()
941{
d90895ac
DW
942 if (CanUndo())
943 {
c4955899
DW
944 if (m_bIsMLE)
945 ::WinSendMsg(GetHwnd(), MLM_UNDO, 0, 0);
946 // Simple entryfields cannot be undone
d90895ac 947 }
c4955899 948} // end of wxTextCtrl::Undo
0e320a79
DW
949
950void wxTextCtrl::Redo()
951{
d90895ac
DW
952 if (CanRedo())
953 {
c4955899
DW
954 if (m_bIsMLE)
955 ::WinSendMsg(GetHwnd(), MLM_UNDO, 0, 0);
956 // Simple entryfields cannot be undone
d90895ac 957 }
c4955899 958} // end of wxTextCtrl::Redo
0e320a79
DW
959
960bool wxTextCtrl::CanUndo() const
961{
3bd418ca
DW
962 bool bOk;
963
964 if (m_bIsMLE)
965 bOk = (::WinSendMsg(GetHwnd(), MLM_QUERYUNDO, 0, 0) != 0);
966 else
967 bOk = FALSE; // can't undo regular edit fields in PM
968 return bOk;
969} // end of wxTextCtrl::CanUndo
0e320a79
DW
970
971bool wxTextCtrl::CanRedo() const
972{
3bd418ca
DW
973 bool bOk;
974
975 if (m_bIsMLE)
976 bOk = (::WinSendMsg(GetHwnd(), MLM_QUERYUNDO, 0, 0) != 0);
977 else
978 bOk = FALSE; // can't undo regular edit fields in PM
979 return bOk;
980} // end of wxTextCtrl::CanRedo
0e320a79 981
d90895ac
DW
982// ----------------------------------------------------------------------------
983// implemenation details
984// ----------------------------------------------------------------------------
0e320a79 985
c4955899
DW
986void wxTextCtrl::Command(
987 wxCommandEvent& rEvent
988)
0e320a79 989{
c4955899
DW
990 SetValue(rEvent.GetString());
991 ProcessCommand (rEvent);
992} // end of wxTextCtrl::Command
0e320a79 993
c4955899
DW
994void wxTextCtrl::OnDropFiles(
995 wxDropFilesEvent& rEvent
996)
0e320a79
DW
997{
998 // By default, load the first file into the text window.
c4955899 999 if (rEvent.GetNumberOfFiles() > 0)
0e320a79 1000 {
c4955899 1001 LoadFile(rEvent.GetFiles()[0]);
0e320a79 1002 }
c4955899
DW
1003} // end of wxTextCtrl::OnDropFiles
1004
1005WXHBRUSH wxTextCtrl::OnCtlColor(
1006 WXHDC hWxDC
1007, WXHWND hWnd
1008, WXUINT uCtlColor
1009, WXUINT uMessage
1010, WXWPARAM wParam
1011, WXLPARAM lParam
1012)
d90895ac 1013{
c4955899
DW
1014 HPS hPS = (HPS)hWxDC;
1015 wxBrush* pBrush = NULL;
1016 wxColour vColBack = GetBackgroundColour();
1017 wxColour vColFore = GetForegroundColour();
1018 wxBrush* pBackgroundBrush = wxTheBrushList->FindOrCreateBrush( GetBackgroundColour()
1019 ,wxSOLID
1020 );
1021
1022 if (m_bUseCtl3D)
1023 {
1024 HBRUSH hBrush = NULLHANDLE;
0e320a79 1025
c4955899
DW
1026 return hBrush;
1027 }
1028 if (GetParent()->GetTransparentBackground())
1029 ::GpiSetBackMix(hPS, BM_LEAVEALONE);
1030 else
1031 ::GpiSetBackMix(hPS, BM_OVERPAINT);
1032 if (!IsEnabled() && (GetWindowStyle() & wxTE_MULTILINE) == 0)
a756f210 1033 vColBack = wxSystemSettings::GetColour(wxSYS_COLOUR_3DFACE);
c4955899
DW
1034 ::GpiSetBackColor(hPS, vColBack.GetPixel());
1035 ::GpiSetColor(hPS, vColFore.GetPixel());
1036 return (WXHBRUSH)pBackgroundBrush->GetResourceHandle();
1037} // end of wxTextCtrl::OnCtlColor
1038
54ffa107
DW
1039bool wxTextCtrl::OS2ShouldPreProcessMessage(
1040 WXMSG* pMsg
1041)
1042{
1043 return wxControl::OS2ShouldPreProcessMessage(pMsg);
1044} // end of wxTextCtrl::OS2ShouldPreProcessMessage
1045
c4955899
DW
1046void wxTextCtrl::OnChar(
1047 wxKeyEvent& rEvent
1048)
0e320a79 1049{
c4955899 1050 switch (rEvent.KeyCode())
d90895ac 1051 {
d90895ac
DW
1052 case WXK_RETURN:
1053 if ( !(m_windowStyle & wxTE_MULTILINE) )
1054 {
c4955899
DW
1055 wxCommandEvent vEvent(wxEVT_COMMAND_TEXT_ENTER, m_windowId);
1056
1057 vEvent.SetEventObject(this);
1058 if ( GetEventHandler()->ProcessEvent(vEvent))
d90895ac
DW
1059 return;
1060 }
1061 //else: multiline controls need Enter for themselves
1062
1063 break;
1064
1065 case WXK_TAB:
1066 // always produce navigation event - even if we process TAB
1067 // ourselves the fact that we got here means that the user code
1068 // decided to skip processing of this TAB - probably to let it
1069 // do its default job.
1070 //
1071 // NB: Notice that Ctrl-Tab is handled elsewhere and Alt-Tab is
1072 // handled by Windows
1073 {
c4955899 1074 wxNavigationKeyEvent vEventNav;
d90895ac 1075
c4955899
DW
1076 vEventNav.SetDirection(!rEvent.ShiftDown());
1077 vEventNav.SetWindowChange(FALSE);
1078 vEventNav.SetEventObject(this);
1079
1080 if ( GetEventHandler()->ProcessEvent(vEventNav) )
d90895ac
DW
1081 return;
1082 }
1083 break;
d90895ac 1084 }
c4955899
DW
1085 rEvent.Skip();
1086} // end of wxTextCtrl::OnChar
0e320a79 1087
c4955899
DW
1088bool wxTextCtrl::OS2Command(
1089 WXUINT uParam
1090, WXWORD WXUNUSED(vId)
1091)
0e320a79 1092{
c4955899 1093 switch (uParam)
d90895ac 1094 {
d90895ac
DW
1095 case EN_SETFOCUS:
1096 case EN_KILLFOCUS:
1097 {
c4955899
DW
1098 wxFocusEvent vEvent( uParam == EN_KILLFOCUS ? wxEVT_KILL_FOCUS
1099 : wxEVT_SET_FOCUS
1100 ,m_windowId
1101 );
1102
1103 vEvent.SetEventObject(this);
1104 GetEventHandler()->ProcessEvent(vEvent);
d90895ac
DW
1105 }
1106 break;
1107
1108 case EN_CHANGE:
1109 {
c4955899
DW
1110 wxCommandEvent vEvent( wxEVT_COMMAND_TEXT_UPDATED
1111 ,m_windowId
1112 );
1113
1114 InitCommandEvent(vEvent);
1115 vEvent.SetString((char*)GetValue().c_str());
1116 ProcessCommand(vEvent);
d90895ac
DW
1117 }
1118 break;
1119
c4955899
DW
1120 case EN_OVERFLOW:
1121 //
1122 // The text size limit has been hit - increase it
1123 //
d90895ac
DW
1124 AdjustSpaceLimit();
1125 break;
1126
c4955899
DW
1127 case EN_SCROLL:
1128 case EN_INSERTMODETOGGLE:
1129 case EN_MEMERROR:
1130 return FALSE;
d90895ac
DW
1131 default:
1132 return FALSE;
1133 }
1134
c4955899
DW
1135 //
1136 // Processed
1137 //
d90895ac 1138 return TRUE;
c4955899 1139} // end of wxTextCtrl::OS2Command
0e320a79 1140
d90895ac 1141void wxTextCtrl::AdjustSpaceLimit()
0e320a79 1142{
3bd418ca
DW
1143 unsigned int uLen = 0;
1144 unsigned int uLimit = 0;
d90895ac 1145
3bd418ca
DW
1146 uLen = ::WinQueryWindowTextLength(GetHwnd());
1147 if (m_bIsMLE)
1148 {
1149 uLimit = (unsigned int)::WinSendMsg( GetHwnd()
1150 ,MLM_QUERYTEXTLIMIT
1151 ,0
1152 ,0
1153 );
1154 }
1155 else
1156 {
919312f7 1157 ENTRYFDATA Efd;
3bd418ca
DW
1158 WNDPARAMS vParams;
1159
1160 vParams.fsStatus = WPM_CBCTLDATA;
919312f7 1161 vParams.pCtlData = &Efd;
3bd418ca
DW
1162 vParams.cbCtlData = sizeof(ENTRYFDATA);
1163
1164 if (::WinSendMsg( GetHwnd()
1165 ,WM_QUERYWINDOWPARAMS
1166 ,&vParams
1167 ,0
1168 ))
919312f7 1169 uLimit = (unsigned int)Efd.cchEditLimit;
d90895ac 1170 else
3bd418ca 1171 uLimit = 32; //PM's default
d90895ac 1172 }
3bd418ca
DW
1173 if (uLen >= uLimit)
1174 {
1175 uLimit = uLen + 0x8000; // 32Kb
1176 if (uLimit > 0xffff)
1177 {
1178 uLimit = 0L;
1179 }
1180 if (m_bIsMLE)
1181 ::WinSendMsg(GetHwnd(), MLM_SETTEXTLIMIT, MPFROMLONG(uLimit), 0);
1182 else
919312f7 1183 ::WinSendMsg(GetHwnd(), EM_SETTEXTLIMIT, MPFROMSHORT(uLimit), 0);
3bd418ca
DW
1184 }
1185} // end of wxTextCtrl::AdjustSpaceLimit
0e320a79 1186
d90895ac 1187bool wxTextCtrl::AcceptsFocus() const
0e320a79 1188{
c4955899 1189 //
57ff8a87
DW
1190 // We don't want focus if we can't be edited unless we're a multiline
1191 // control because then it might be still nice to get focus from keyboard
1192 // to be able to scroll it without mouse
c4955899 1193 //
57ff8a87 1194 return (IsEditable() || IsMultiLine()) && wxControl::AcceptsFocus();
c4955899 1195} // end of wxTextCtrl::Command
0e320a79 1196
e78c4d50 1197wxSize wxTextCtrl::DoGetBestSize() const
0e320a79 1198{
c4955899
DW
1199 int nCx;
1200 int nCy;
0e320a79 1201
c4955899 1202 wxGetCharSize(GetHWND(), &nCx, &nCy, (wxFont*)&GetFont());
d90895ac 1203
c4955899 1204 int wText = DEFAULT_ITEM_WIDTH;
987da0d4 1205 int hText = (EDIT_HEIGHT_FROM_CHAR_HEIGHT(nCy) * .8);
c4955899
DW
1206
1207 if (m_windowStyle & wxTE_MULTILINE)
d90895ac 1208 {
239c2e96 1209 hText *= wxMax(GetNumberOfLines(), 5);
d90895ac
DW
1210 }
1211 //else: for single line control everything is ok
d90895ac 1212 return wxSize(wText, hText);
c4955899 1213} // end of wxTextCtrl::DoGetBestSize
0e320a79 1214
d90895ac
DW
1215// ----------------------------------------------------------------------------
1216// standard handlers for standard edit menu events
1217// ----------------------------------------------------------------------------
1218
c4955899
DW
1219void wxTextCtrl::OnCut(
1220 wxCommandEvent& rEvent
1221)
0e320a79
DW
1222{
1223 Cut();
c4955899 1224} // end of wxTextCtrl::OnCut
0e320a79 1225
c4955899
DW
1226void wxTextCtrl::OnCopy(
1227 wxCommandEvent& rEvent
1228)
0e320a79
DW
1229{
1230 Copy();
c4955899 1231} // end of wxTextCtrl::OnCopy
0e320a79 1232
c4955899
DW
1233void wxTextCtrl::OnPaste(
1234 wxCommandEvent& rEvent
1235)
0e320a79
DW
1236{
1237 Paste();
c4955899 1238} // end of wxTextCtrl::OnPaste
0e320a79 1239
c4955899
DW
1240void wxTextCtrl::OnUndo(
1241 wxCommandEvent& rEvent
1242)
0e320a79
DW
1243{
1244 Undo();
c4955899 1245} // end of wxTextCtrl::OnUndo
0e320a79 1246
c4955899
DW
1247void wxTextCtrl::OnRedo(
1248 wxCommandEvent& rEvent
1249)
0e320a79
DW
1250{
1251 Redo();
c4955899 1252} // end of wxTextCtrl::OnRedo
0e320a79 1253
2b5f62a0
VZ
1254void wxTextCtrl::OnDelete(
1255 wxCommandEvent& rEvent
1256)
1257{
1258 long lFrom;
1259 long lTo;
1260
1261 GetSelection( &lFrom
1262 ,&lTo
1263 );
1264 if (lFrom != -1 && lTo != -1)
1265 Remove( lFrom
1266 ,lTo
1267 );
1268} // end of wxTextCtrl::OnDelete
1269
1270void wxTextCtrl::OnSelectAll(
1271 wxCommandEvent& rEvent
1272)
1273{
1274 SetSelection(-1, -1);
1275} // end of wxTextCtrl::OnSelectAll
1276
c4955899
DW
1277void wxTextCtrl::OnUpdateCut(
1278 wxUpdateUIEvent& rEvent
1279)
0e320a79 1280{
c4955899
DW
1281 rEvent.Enable(CanCut());
1282} // end of wxTextCtrl::OnUpdateCut
0e320a79 1283
c4955899
DW
1284void wxTextCtrl::OnUpdateCopy(
1285 wxUpdateUIEvent& rEvent
1286)
0e320a79 1287{
c4955899
DW
1288 rEvent.Enable(CanCopy());
1289} // end of wxTextCtrl::OnUpdateCopy
0e320a79 1290
c4955899
DW
1291void wxTextCtrl::OnUpdatePaste(
1292 wxUpdateUIEvent& rEvent
1293)
0e320a79 1294{
c4955899
DW
1295 rEvent.Enable(CanPaste());
1296} // end of wxTextCtrl::OnUpdatePaste
0e320a79 1297
c4955899
DW
1298void wxTextCtrl::OnUpdateUndo(
1299 wxUpdateUIEvent& rEvent
1300)
0e320a79 1301{
c4955899
DW
1302 rEvent.Enable(CanUndo());
1303} // end of wxTextCtrl::OnUpdateUndo
1304
1305void wxTextCtrl::OnUpdateRedo(
1306 wxUpdateUIEvent& rEvent
1307)
1308{
1309 rEvent.Enable(CanRedo());
1310} // end of wxTextCtrl::OnUpdateRedo
0e320a79 1311
2b5f62a0
VZ
1312void wxTextCtrl::OnUpdateDelete(
1313 wxUpdateUIEvent& rEvent
1314)
1315{
1316 long lFrom;
1317 long lTo;
1318
1319 GetSelection( &lFrom
1320 ,&lTo
1321 );
1322 rEvent.Enable( lFrom != -1L && lTo != -1L && lFrom != lTo && IsEditable()) ;
1323} // end of wxTextCtrl::OnUpdateDelete
1324
1325void wxTextCtrl::OnUpdateSelectAll(
1326 wxUpdateUIEvent& rEvent
1327)
1328{
1329 rEvent.Enable(GetLastPosition() > 0);
1330} // end of wxTextCtrl::OnUpdateSelectAll
1331
c4955899
DW
1332bool wxTextCtrl::SetBackgroundColour(
1333 const wxColour& rColour
1334)
0e320a79 1335{
c4955899
DW
1336 if (m_bIsMLE)
1337 ::WinSendMsg(GetHwnd(), MLM_SETBACKCOLOR, (MPARAM)rColour.GetPixel(), MLE_INDEX);
1338 return TRUE;
1339} // end of wxTextCtrl::SetBackgroundColour
1340
1341bool wxTextCtrl::SetForegroundColour(
1342 const wxColour& rColour
1343)
1344{
1345 if (m_bIsMLE)
1346 ::WinSendMsg(GetHwnd(), MLM_SETTEXTCOLOR, (MPARAM)rColour.GetPixel(), MLE_INDEX);
1347 return TRUE;
1348} // end of wxTextCtrl::SetForegroundColour
d90895ac 1349
39c26ef2
DW
1350bool wxTextCtrl::SetStyle(
1351 long lStart
1352, long lEnd
1353, const wxTextAttr& rStyle
1354)
1355{
1356 HWND hWnd = GetHwnd();
1357
1358 if (lStart > lEnd)
1359 {
1360 long lTmp = lStart;
1361
1362 lStart = lEnd;
1363 lEnd = lTmp;
1364 }
1365
1366 //
1367 // We can only change the format of the selection, so select the range we
1368 // want and restore the old selection later
1369 //
1370 long lStartOld;
1371 long lEndOld;
1372
1373 GetSelection( &lStartOld
1374 ,&lEndOld
1375 );
1376
1377 //
1378 // But do we really have to change the selection?
1379 //
1380 bool bChangeSel = lStart != lStartOld ||
1381 lEnd != lEndOld;
1382
1383 if (bChangeSel)
1384 {
1385 if (m_bIsMLE)
1386 ::WinSendMsg(hWnd, MLM_SETSEL, MPFROM2SHORT((USHORT)lStart, (USHORT)lEnd), 0);
1387 else
1388 ::WinSendMsg(hWnd, EM_SETSEL, MPFROM2SHORT((USHORT)lStart, (USHORT)lEnd), 0);
1389 }
1390
1391 //
1392 // TODO:: finish this part
1393 //
1394 return TRUE;
1395} // end of wxTextCtrl::SetStyle
1396