]> git.saurik.com Git - wxWidgets.git/blame - src/os2/textctrl.cpp
added arrow cursor for standard controls
[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
3bd418ca
DW
89bool wxTextCtrl::Create(
90 wxWindow* pParent
91, wxWindowID vId
92, const wxString& rsValue
93, const wxPoint& rPos
94, const wxSize& rSize
95, long lStyle
5d4b632b 96#if wxUSE_VALIDATORS
3bd418ca 97, const wxValidator& rValidator
5d4b632b 98#endif
3bd418ca
DW
99, const wxString& rsName
100)
101{
39c26ef2
DW
102 HWND hParent;
103 int nTempy;
cfcebdb1 104
3bd418ca
DW
105 //
106 // Base initialization
107 //
108 if ( !CreateBase( pParent
109 ,vId
110 ,rPos
111 ,rSize
112 ,lStyle
113#if wxUSE_VALIDATORS
114 ,rValidator
115#endif
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
510 SetInsertionPoint(lPos);
511} // end of wxTextCtrl::SetInsertionPointEnd
0e320a79
DW
512
513long wxTextCtrl::GetInsertionPoint() const
514{
3bd418ca
DW
515 WXDWORD dwPos = 0L;
516
517 if (m_bIsMLE)
518 dwPos = (WXDWORD)::WinSendMsg(GetHwnd(), MLM_QUERYSEL, (MPARAM)MLFQS_MINSEL, 0);
519 else
520 {
521 dwPos = (WXDWORD)::WinSendMsg(GetHwnd(), EM_QUERYSEL, 0, 0);
522 dwPos = SHORT1FROMMP((MPARAM)dwPos); // the first 16 bit value is the min pos
523 }
524 return (dwPos & 0xFFFF);
525} // end of wxTextCtrl::GetInsertionPoint
0e320a79
DW
526
527long wxTextCtrl::GetLastPosition() const
528{
3bd418ca
DW
529 HWND hWnd = GetHwnd();
530 long lCharIndex;
531 long lLineLength;
0e320a79 532
3bd418ca
DW
533 if (m_bIsMLE)
534 {
535 lCharIndex = 0;
d90895ac 536
3bd418ca
DW
537 //
538 // This just gets the total text length. The last will be this value
539 //
540 lLineLength = (long)::WinSendMsg(hWnd, MLM_QUERYTEXTLENGTH, 0, 0);
541 }
542 else
543 {
544 WNDPARAMS vParams;
545
546 lCharIndex = 0;
547 vParams.fsStatus = WPM_CCHTEXT;
548 if (::WinSendMsg( GetHwnd()
549 ,WM_QUERYWINDOWPARAMS
550 ,&vParams
551 ,0
552 ))
553 {
554 lLineLength = (long)vParams.cchText;
555 }
556 else
557 lLineLength = 0;
558 }
559 return(lCharIndex + lLineLength);
560} // end of wxTextCtrl::GetLastPosition
0e320a79 561
d90895ac
DW
562// If the return values from and to are the same, there is no
563// selection.
c4955899
DW
564void wxTextCtrl::GetSelection(
565 long* plFrom
566, long* plTo
567) const
0e320a79 568{
c4955899 569 WXDWORD dwPos;
0e320a79 570
c4955899
DW
571 if (m_bIsMLE)
572 dwPos = (WXDWORD)::WinSendMsg(GetHwnd(), MLM_QUERYSEL, (MPARAM)MLFQS_MINSEL, 0);
573 else
574 {
575 dwPos = (WXDWORD)::WinSendMsg(GetHwnd(), EM_QUERYSEL, 0, 0);
576 }
577 *plFrom = SHORT1FROMMP((MPARAM)dwPos); // the first 16 bit value is the min pos
578 *plTo = SHORT2FROMMP((MPARAM)dwPos); // the first 16 bit value is the min pos
579} // end of wxTextCtrl::GetSelection
0e320a79 580
d90895ac
DW
581bool wxTextCtrl::IsEditable() const
582{
c4955899
DW
583 if (m_bIsMLE)
584 return((bool)LONGFROMMR(::WinSendMsg(GetHwnd(), MLM_QUERYREADONLY, 0, 0)));
585 else
586 return((bool)LONGFROMMR(::WinSendMsg(GetHwnd(), EM_QUERYREADONLY, 0, 0)));
587} // end of wxTextCtrl::IsEditable
0e320a79 588
d90895ac
DW
589// ----------------------------------------------------------------------------
590// Editing
591// ----------------------------------------------------------------------------
592
c4955899
DW
593void wxTextCtrl::Replace(
594 long lFrom
595, long lTo
596, const wxString& rsValue
597)
0e320a79 598{
d90895ac 599#if wxUSE_CLIPBOARD
c4955899
DW
600 HWND hWnd = GetHwnd();
601 long lFromChar = lFrom;
602 long lToChar = lTo;
0e320a79 603
c4955899 604 //
d90895ac 605 // Set selection and remove it
c4955899
DW
606 //
607 if (m_bIsMLE)
608 {
609 ::WinSendMsg(hWnd, MLM_SETSEL, MPFROM2SHORT((USHORT)lFrom, (USHORT)lTo), 0);
610 ::WinSendMsg(hWnd, MLM_CUT, 0, 0);
611 }
612 else
613 {
614 ::WinSendMsg(hWnd, EM_SETSEL, MPFROM2SHORT((USHORT)lFrom, (USHORT)lTo), 0);
615 ::WinSendMsg(hWnd, EM_CUT, 0, 0);
616 }
0e320a79 617
c4955899 618 //
d90895ac 619 // Now replace with 'value', by pasting.
c4955899
DW
620 //
621 wxSetClipboardData(wxDF_TEXT, (wxObject *) (const wxChar *)rsValue, 0, 0);
0e320a79 622
d90895ac 623 // Paste into edit control
c4955899
DW
624 if (m_bIsMLE)
625 ::WinSendMsg(hWnd, MLM_PASTE, (MPARAM)0, (MPARAM)0);
626 else
627 ::WinSendMsg(hWnd, EM_PASTE, (MPARAM)0, (MPARAM)0);
d90895ac
DW
628#else
629 wxFAIL_MSG("wxTextCtrl::Replace not implemented if wxUSE_CLIPBOARD is 0.");
630#endif
c4955899 631} // end of wxTextCtrl::Replace
0e320a79 632
c4955899
DW
633void wxTextCtrl::Remove(
634 long lFrom
635, long lTo
636)
0e320a79 637{
c4955899
DW
638 HWND hWnd = GetHwnd();
639 long lFromChar = lFrom;
640 long lToChar = lTo;
d90895ac 641
c4955899
DW
642 if (m_bIsMLE)
643 {
644 ::WinSendMsg(hWnd, MLM_SETSEL, MPFROM2SHORT((USHORT)lFrom, (USHORT)lTo), 0);
645 ::WinSendMsg(hWnd, MLM_CUT, 0, 0);
646 }
647 else
648 {
649 ::WinSendMsg(hWnd, EM_SETSEL, MPFROM2SHORT((USHORT)lFrom, (USHORT)lTo), 0);
650 ::WinSendMsg(hWnd, EM_CUT, 0, 0);
651 }
652} // end of wxTextCtrl::Remove
0e320a79 653
c4955899
DW
654void wxTextCtrl::SetSelection(
655 long lFrom
656, long lTo
657)
0e320a79 658{
c4955899
DW
659 HWND hWnd = GetHwnd();
660 long lFromChar = lFrom;
661 long lToChar = lTo;
d90895ac 662
c4955899
DW
663 //
664 // If from and to are both -1, it means (in wxWindows) that all text should
d90895ac 665 // be selected. Translate into Windows convention
c4955899
DW
666 //
667 if ((lFrom == -1L) && (lTo == -1L))
d90895ac 668 {
c4955899
DW
669 lFromChar = 0L;
670 lToChar = -1L;
d90895ac 671 }
c4955899
DW
672 if (m_bIsMLE)
673 ::WinSendMsg(hWnd, MLM_SETSEL, (MPARAM)lFromChar, (MPARAM)lToChar);
674 else
675 ::WinSendMsg(hWnd, EM_SETSEL, MPFROM2SHORT((USHORT)lFromChar, (USHORT)lToChar), (MPARAM)0);
676} // end of wxTextCtrl::SetSelection
d90895ac 677
c4955899
DW
678bool wxTextCtrl::LoadFile(
679 const wxString& rsFile
680)
0e320a79 681{
c4955899 682 if ( wxTextCtrlBase::LoadFile(rsFile) )
d90895ac 683 {
c4955899
DW
684 //
685 // Update the size limit if needed
686 //
d90895ac 687 AdjustSpaceLimit();
d90895ac
DW
688 return TRUE;
689 }
d90895ac 690 return FALSE;
c4955899 691} // end of wxTextCtrl::LoadFile
0e320a79
DW
692
693bool wxTextCtrl::IsModified() const
694{
c4955899
DW
695 bool bRc;
696
697 if (m_bIsMLE)
698 bRc = (bool)LONGFROMMR(::WinSendMsg(GetHwnd(), MLM_QUERYCHANGED, 0, 0));
699 else
700 bRc = (bool)LONGFROMMR(::WinSendMsg(GetHwnd(), EM_QUERYCHANGED, 0, 0));
701 return bRc;
702} // end of wxTextCtrl::IsModified
0e320a79 703
3bd418ca 704//
0e320a79 705// Makes 'unmodified'
3bd418ca 706//
0e320a79
DW
707void wxTextCtrl::DiscardEdits()
708{
3bd418ca
DW
709 if (m_bIsMLE)
710 ::WinSendMsg(GetHwnd(), MLM_SETCHANGED, MPFROMLONG(FALSE), 0);
711 else
712 //
713 // EM controls do not have a SETCHANGED but issuing a query should reset it
714 //
715 ::WinSendMsg(GetHwnd(), EM_QUERYCHANGED, 0, 0);
716} // end of wxTextCtrl::DiscardEdits
0e320a79
DW
717
718int wxTextCtrl::GetNumberOfLines() const
719{
c4955899 720 int nNumLines;
0e320a79 721
c4955899
DW
722 if (m_bIsMLE)
723 nNumLines = (int)::WinSendMsg(GetHwnd(), MLM_QUERYLINECOUNT, 0, 0);
724 else
725 nNumLines = 1;
726 return nNumLines;
727} // end of wxTextCtrl::GetNumberOfLines
d90895ac 728
c4955899
DW
729long wxTextCtrl::XYToPosition(
730 long lX
731, long lY
732) const
733{
734 HWND hWnd = GetHwnd();
735 long lCharIndex = 0L;
736 long lLen;
0e320a79 737
c4955899
DW
738 if (m_bIsMLE)
739 {
740 lLen = (long)::WinSendMsg(GetHwnd(), MLM_QUERYLINELENGTH, 0, 0);
741 lCharIndex = ((lLen * lY) + lX);
742 }
743 else
744 lCharIndex = lX;
745 return lCharIndex;
746} // end of wxTextCtrl::XYToPosition
747
748bool wxTextCtrl::PositionToXY(
749 long lPos
750, long* plX
751, long* plY
752) const
0e320a79 753{
c4955899
DW
754 HWND hWnd = GetHwnd();
755 long nLineNo = -1;
756 long lCharIndex = 0;
d90895ac 757
c4955899
DW
758 if (m_bIsMLE)
759 nLineNo = (long)::WinSendMsg(hWnd, MLM_LINEFROMCHAR, (MPARAM)lPos, 0);
760 else
761 nLineNo = 0;
d90895ac 762
c4955899 763 if (nLineNo == -1)
d90895ac
DW
764 {
765 // no such line
766 return FALSE;
767 }
768
c4955899 769 //
d90895ac 770 // This gets the char index for the _beginning_ of this line
c4955899
DW
771 //
772 long lLineWidth;
773
774 if (m_bIsMLE)
775 {
776 lLineWidth = (long)::WinSendMsg(hWnd, MLM_QUERYLINELENGTH, (MPARAM)0, (MPARAM)0);
777 lCharIndex = (nLineNo + 1) * lLineWidth;
778 }
779 else
780 {
781 WNDPARAMS vParams;
782
783 vParams.fsStatus = WPM_CCHTEXT;
784 if (::WinSendMsg( hWnd
785 ,WM_QUERYWINDOWPARAMS
786 ,&vParams
787 ,0
788 ))
789 {
790 lCharIndex = vParams.cchText;
791 }
792 else
793 lCharIndex = 32;
794 }
795
796 if (lCharIndex == -1)
d90895ac
DW
797 {
798 return FALSE;
799 }
800
c4955899
DW
801 //
802 // The X position must therefore be the difference between pos and charIndex
803 //
804 if (plX)
805 *plX = lPos - lCharIndex;
806 if (plY)
807 *plY = nLineNo;
d90895ac
DW
808
809 return TRUE;
c4955899 810} // end of wxTextCtrl::PositionToXY
0e320a79 811
c4955899
DW
812void wxTextCtrl::ShowPosition(
813 long lPos
814)
0e320a79 815{
c4955899
DW
816 HWND hWnd = GetHwnd();
817 long lCurrentLineLineNo = 0L;
d90895ac
DW
818
819 // To scroll to a position, we pass the number of lines and characters
820 // to scroll *by*. This means that we need to:
821 // (1) Find the line position of the current line.
822 // (2) Find the line position of pos.
823 // (3) Scroll by (pos - current).
824 // For now, ignore the horizontal scrolling.
825
c4955899 826 //
d90895ac
DW
827 // Is this where scrolling is relative to - the line containing the caret?
828 // Or is the first visible line??? Try first visible line.
c4955899
DW
829 //
830 if (m_bIsMLE)
831 {
832 //
833 // In PM this is the actual char position
834 //
835 lCurrentLineLineNo = (long)::WinSendMsg(hWnd, MLM_QUERYFIRSTCHAR, (MPARAM)0, (MPARAM)0);
d90895ac 836
c4955899
DW
837 //
838 // This will cause a scroll to the selected position
839 //
840 ::WinSendMsg(hWnd, MLM_SETSEL, (MPARAM)lCurrentLineLineNo, (MPARAM)lCurrentLineLineNo);
841 }
842} // end of wxTextCtrl::ShowPosition
0e320a79 843
3bd418ca
DW
844int wxTextCtrl::GetLineLength(
845 long lLineNo
846) const
0e320a79 847{
3bd418ca 848 long lLen = 0L;
d90895ac 849
3bd418ca
DW
850 if (m_bIsMLE)
851 lLen = (long)::WinSendMsg(GetHwnd(), MLM_QUERYLINELENGTH, 0, 0);
852 else
853 {
854 WNDPARAMS vParams;
855
856 vParams.fsStatus = WPM_CCHTEXT;
857 if (::WinSendMsg( GetHwnd()
858 ,WM_QUERYWINDOWPARAMS
859 ,&vParams
860 ,0
861 ))
862 {
863 lLen = vParams.cchText;
864 }
865 else
866 lLen = 32;
867 }
868 return lLen;
c4955899 869} // end ofwxTextCtrl::GetLineLength
0e320a79 870
c4955899
DW
871wxString wxTextCtrl::GetLineText(
872 long lLineNo
873) const
0e320a79 874{
c4955899
DW
875 long lLen = (long)GetLineLength((long)lLineNo) + 1;
876 wxString sStr;
877 char* zBuf;
0e320a79 878
c4955899
DW
879 //
880 // There must be at least enough place for the length WORD in the
881 // buffer
882 //
883 lLen += sizeof(WORD);
884 zBuf = new char[lLen];
885 if (m_bIsMLE)
886 {
887 long lIndex;
888 long lBuflen;
889 long lCopied;
0e320a79 890
c4955899
DW
891 lLen = (long)::WinSendMsg(GetHwnd(), MLM_QUERYLINELENGTH, 0, 0);
892 lIndex = lLen * lLineNo;
0e320a79 893
c4955899
DW
894 ::WinSendMsg(GetHwnd(), MLM_SETSEL, (MPARAM)lIndex, (MPARAM)lIndex);
895 ::WinSendMsg(GetHwnd(), MLM_SETIMPORTEXPORT, MPFROMP(zBuf), MPFROMSHORT((USHORT)sizeof(zBuf)));
896 lBuflen = (long)::WinSendMsg(GetHwnd(), MLM_QUERYFORMATTEXTLENGTH, MPFROMLONG(lIndex), MPFROMLONG(-1));
897 lCopied = (long)::WinSendMsg(GetHwnd(), MLM_EXPORT, MPFROMP(&lIndex), MPFROMP(&lBuflen));
898 zBuf[lCopied] = '\0';
899 }
900 else
901 {
902 WNDPARAMS vParams;
903
904 vParams.fsStatus = WPM_CCHTEXT;
905 if (::WinSendMsg( GetHwnd()
906 ,WM_QUERYWINDOWPARAMS
907 ,&vParams
908 ,0
909 ))
910 memcpy(zBuf, vParams.pszText, vParams.cchText);
911 zBuf[vParams.cchText] = '\0';
912 }
913 sStr = zBuf;
914 delete [] zBuf;
915 return sStr;
916} // end of wxTextCtrl::GetLineText
0e320a79 917
d90895ac 918// ----------------------------------------------------------------------------
0e320a79 919// Undo/redo
d90895ac
DW
920// ----------------------------------------------------------------------------
921
0e320a79
DW
922void wxTextCtrl::Undo()
923{
d90895ac
DW
924 if (CanUndo())
925 {
c4955899
DW
926 if (m_bIsMLE)
927 ::WinSendMsg(GetHwnd(), MLM_UNDO, 0, 0);
928 // Simple entryfields cannot be undone
d90895ac 929 }
c4955899 930} // end of wxTextCtrl::Undo
0e320a79
DW
931
932void wxTextCtrl::Redo()
933{
d90895ac
DW
934 if (CanRedo())
935 {
c4955899
DW
936 if (m_bIsMLE)
937 ::WinSendMsg(GetHwnd(), MLM_UNDO, 0, 0);
938 // Simple entryfields cannot be undone
d90895ac 939 }
c4955899 940} // end of wxTextCtrl::Redo
0e320a79
DW
941
942bool wxTextCtrl::CanUndo() const
943{
3bd418ca
DW
944 bool bOk;
945
946 if (m_bIsMLE)
947 bOk = (::WinSendMsg(GetHwnd(), MLM_QUERYUNDO, 0, 0) != 0);
948 else
949 bOk = FALSE; // can't undo regular edit fields in PM
950 return bOk;
951} // end of wxTextCtrl::CanUndo
0e320a79
DW
952
953bool wxTextCtrl::CanRedo() const
954{
3bd418ca
DW
955 bool bOk;
956
957 if (m_bIsMLE)
958 bOk = (::WinSendMsg(GetHwnd(), MLM_QUERYUNDO, 0, 0) != 0);
959 else
960 bOk = FALSE; // can't undo regular edit fields in PM
961 return bOk;
962} // end of wxTextCtrl::CanRedo
0e320a79 963
d90895ac
DW
964// ----------------------------------------------------------------------------
965// implemenation details
966// ----------------------------------------------------------------------------
0e320a79 967
c4955899
DW
968void wxTextCtrl::Command(
969 wxCommandEvent& rEvent
970)
0e320a79 971{
c4955899
DW
972 SetValue(rEvent.GetString());
973 ProcessCommand (rEvent);
974} // end of wxTextCtrl::Command
0e320a79 975
c4955899
DW
976void wxTextCtrl::OnDropFiles(
977 wxDropFilesEvent& rEvent
978)
0e320a79
DW
979{
980 // By default, load the first file into the text window.
c4955899 981 if (rEvent.GetNumberOfFiles() > 0)
0e320a79 982 {
c4955899 983 LoadFile(rEvent.GetFiles()[0]);
0e320a79 984 }
c4955899
DW
985} // end of wxTextCtrl::OnDropFiles
986
987WXHBRUSH wxTextCtrl::OnCtlColor(
988 WXHDC hWxDC
989, WXHWND hWnd
990, WXUINT uCtlColor
991, WXUINT uMessage
992, WXWPARAM wParam
993, WXLPARAM lParam
994)
d90895ac 995{
c4955899
DW
996 HPS hPS = (HPS)hWxDC;
997 wxBrush* pBrush = NULL;
998 wxColour vColBack = GetBackgroundColour();
999 wxColour vColFore = GetForegroundColour();
1000 wxBrush* pBackgroundBrush = wxTheBrushList->FindOrCreateBrush( GetBackgroundColour()
1001 ,wxSOLID
1002 );
1003
1004 if (m_bUseCtl3D)
1005 {
1006 HBRUSH hBrush = NULLHANDLE;
0e320a79 1007
c4955899
DW
1008 return hBrush;
1009 }
1010 if (GetParent()->GetTransparentBackground())
1011 ::GpiSetBackMix(hPS, BM_LEAVEALONE);
1012 else
1013 ::GpiSetBackMix(hPS, BM_OVERPAINT);
1014 if (!IsEnabled() && (GetWindowStyle() & wxTE_MULTILINE) == 0)
a756f210 1015 vColBack = wxSystemSettings::GetColour(wxSYS_COLOUR_3DFACE);
c4955899
DW
1016 ::GpiSetBackColor(hPS, vColBack.GetPixel());
1017 ::GpiSetColor(hPS, vColFore.GetPixel());
1018 return (WXHBRUSH)pBackgroundBrush->GetResourceHandle();
1019} // end of wxTextCtrl::OnCtlColor
1020
54ffa107
DW
1021bool wxTextCtrl::OS2ShouldPreProcessMessage(
1022 WXMSG* pMsg
1023)
1024{
1025 return wxControl::OS2ShouldPreProcessMessage(pMsg);
1026} // end of wxTextCtrl::OS2ShouldPreProcessMessage
1027
c4955899
DW
1028void wxTextCtrl::OnChar(
1029 wxKeyEvent& rEvent
1030)
0e320a79 1031{
c4955899 1032 switch (rEvent.KeyCode())
d90895ac 1033 {
d90895ac
DW
1034 case WXK_RETURN:
1035 if ( !(m_windowStyle & wxTE_MULTILINE) )
1036 {
c4955899
DW
1037 wxCommandEvent vEvent(wxEVT_COMMAND_TEXT_ENTER, m_windowId);
1038
1039 vEvent.SetEventObject(this);
1040 if ( GetEventHandler()->ProcessEvent(vEvent))
d90895ac
DW
1041 return;
1042 }
1043 //else: multiline controls need Enter for themselves
1044
1045 break;
1046
1047 case WXK_TAB:
1048 // always produce navigation event - even if we process TAB
1049 // ourselves the fact that we got here means that the user code
1050 // decided to skip processing of this TAB - probably to let it
1051 // do its default job.
1052 //
1053 // NB: Notice that Ctrl-Tab is handled elsewhere and Alt-Tab is
1054 // handled by Windows
1055 {
c4955899 1056 wxNavigationKeyEvent vEventNav;
d90895ac 1057
c4955899
DW
1058 vEventNav.SetDirection(!rEvent.ShiftDown());
1059 vEventNav.SetWindowChange(FALSE);
1060 vEventNav.SetEventObject(this);
1061
1062 if ( GetEventHandler()->ProcessEvent(vEventNav) )
d90895ac
DW
1063 return;
1064 }
1065 break;
d90895ac 1066 }
c4955899
DW
1067 rEvent.Skip();
1068} // end of wxTextCtrl::OnChar
0e320a79 1069
c4955899
DW
1070bool wxTextCtrl::OS2Command(
1071 WXUINT uParam
1072, WXWORD WXUNUSED(vId)
1073)
0e320a79 1074{
c4955899 1075 switch (uParam)
d90895ac 1076 {
d90895ac
DW
1077 case EN_SETFOCUS:
1078 case EN_KILLFOCUS:
1079 {
c4955899
DW
1080 wxFocusEvent vEvent( uParam == EN_KILLFOCUS ? wxEVT_KILL_FOCUS
1081 : wxEVT_SET_FOCUS
1082 ,m_windowId
1083 );
1084
1085 vEvent.SetEventObject(this);
1086 GetEventHandler()->ProcessEvent(vEvent);
d90895ac
DW
1087 }
1088 break;
1089
1090 case EN_CHANGE:
1091 {
c4955899
DW
1092 wxCommandEvent vEvent( wxEVT_COMMAND_TEXT_UPDATED
1093 ,m_windowId
1094 );
1095
1096 InitCommandEvent(vEvent);
1097 vEvent.SetString((char*)GetValue().c_str());
1098 ProcessCommand(vEvent);
d90895ac
DW
1099 }
1100 break;
1101
c4955899
DW
1102 case EN_OVERFLOW:
1103 //
1104 // The text size limit has been hit - increase it
1105 //
d90895ac
DW
1106 AdjustSpaceLimit();
1107 break;
1108
c4955899
DW
1109 case EN_SCROLL:
1110 case EN_INSERTMODETOGGLE:
1111 case EN_MEMERROR:
1112 return FALSE;
d90895ac
DW
1113 default:
1114 return FALSE;
1115 }
1116
c4955899
DW
1117 //
1118 // Processed
1119 //
d90895ac 1120 return TRUE;
c4955899 1121} // end of wxTextCtrl::OS2Command
0e320a79 1122
d90895ac 1123void wxTextCtrl::AdjustSpaceLimit()
0e320a79 1124{
3bd418ca
DW
1125 unsigned int uLen = 0;
1126 unsigned int uLimit = 0;
d90895ac 1127
3bd418ca
DW
1128 uLen = ::WinQueryWindowTextLength(GetHwnd());
1129 if (m_bIsMLE)
1130 {
1131 uLimit = (unsigned int)::WinSendMsg( GetHwnd()
1132 ,MLM_QUERYTEXTLIMIT
1133 ,0
1134 ,0
1135 );
1136 }
1137 else
1138 {
1139 ENTRYFDATA* pEfd;
1140 WNDPARAMS vParams;
1141
1142 vParams.fsStatus = WPM_CBCTLDATA;
1143 vParams.cbCtlData = sizeof(ENTRYFDATA);
1144
1145 if (::WinSendMsg( GetHwnd()
1146 ,WM_QUERYWINDOWPARAMS
1147 ,&vParams
1148 ,0
1149 ))
1150 {
1151 pEfd = (ENTRYFDATA*)vParams.pCtlData;
1152 uLimit = (unsigned int)pEfd->cchEditLimit;
1153 }
d90895ac 1154 else
3bd418ca 1155 uLimit = 32; //PM's default
d90895ac 1156 }
3bd418ca
DW
1157 if (uLen >= uLimit)
1158 {
1159 uLimit = uLen + 0x8000; // 32Kb
1160 if (uLimit > 0xffff)
1161 {
1162 uLimit = 0L;
1163 }
1164 if (m_bIsMLE)
1165 ::WinSendMsg(GetHwnd(), MLM_SETTEXTLIMIT, MPFROMLONG(uLimit), 0);
1166 else
1167 ::WinSendMsg(GetHwnd(), EM_SETTEXTLIMIT, MPFROMLONG(uLimit), 0);
1168 }
1169} // end of wxTextCtrl::AdjustSpaceLimit
0e320a79 1170
d90895ac 1171bool wxTextCtrl::AcceptsFocus() const
0e320a79 1172{
c4955899
DW
1173 //
1174 // We don't want focus if we can't be edited
1175 //
d90895ac 1176 return IsEditable() && wxControl::AcceptsFocus();
c4955899 1177} // end of wxTextCtrl::Command
0e320a79 1178
e78c4d50 1179wxSize wxTextCtrl::DoGetBestSize() const
0e320a79 1180{
c4955899
DW
1181 int nCx;
1182 int nCy;
0e320a79 1183
c4955899 1184 wxGetCharSize(GetHWND(), &nCx, &nCy, (wxFont*)&GetFont());
d90895ac 1185
c4955899 1186 int wText = DEFAULT_ITEM_WIDTH;
987da0d4 1187 int hText = (EDIT_HEIGHT_FROM_CHAR_HEIGHT(nCy) * .8);
c4955899
DW
1188
1189 if (m_windowStyle & wxTE_MULTILINE)
d90895ac 1190 {
239c2e96 1191 hText *= wxMax(GetNumberOfLines(), 5);
d90895ac
DW
1192 }
1193 //else: for single line control everything is ok
d90895ac 1194 return wxSize(wText, hText);
c4955899 1195} // end of wxTextCtrl::DoGetBestSize
0e320a79 1196
d90895ac
DW
1197// ----------------------------------------------------------------------------
1198// standard handlers for standard edit menu events
1199// ----------------------------------------------------------------------------
1200
c4955899
DW
1201void wxTextCtrl::OnCut(
1202 wxCommandEvent& rEvent
1203)
0e320a79
DW
1204{
1205 Cut();
c4955899 1206} // end of wxTextCtrl::OnCut
0e320a79 1207
c4955899
DW
1208void wxTextCtrl::OnCopy(
1209 wxCommandEvent& rEvent
1210)
0e320a79
DW
1211{
1212 Copy();
c4955899 1213} // end of wxTextCtrl::OnCopy
0e320a79 1214
c4955899
DW
1215void wxTextCtrl::OnPaste(
1216 wxCommandEvent& rEvent
1217)
0e320a79
DW
1218{
1219 Paste();
c4955899 1220} // end of wxTextCtrl::OnPaste
0e320a79 1221
c4955899
DW
1222void wxTextCtrl::OnUndo(
1223 wxCommandEvent& rEvent
1224)
0e320a79
DW
1225{
1226 Undo();
c4955899 1227} // end of wxTextCtrl::OnUndo
0e320a79 1228
c4955899
DW
1229void wxTextCtrl::OnRedo(
1230 wxCommandEvent& rEvent
1231)
0e320a79
DW
1232{
1233 Redo();
c4955899 1234} // end of wxTextCtrl::OnRedo
0e320a79 1235
c4955899
DW
1236void wxTextCtrl::OnUpdateCut(
1237 wxUpdateUIEvent& rEvent
1238)
0e320a79 1239{
c4955899
DW
1240 rEvent.Enable(CanCut());
1241} // end of wxTextCtrl::OnUpdateCut
0e320a79 1242
c4955899
DW
1243void wxTextCtrl::OnUpdateCopy(
1244 wxUpdateUIEvent& rEvent
1245)
0e320a79 1246{
c4955899
DW
1247 rEvent.Enable(CanCopy());
1248} // end of wxTextCtrl::OnUpdateCopy
0e320a79 1249
c4955899
DW
1250void wxTextCtrl::OnUpdatePaste(
1251 wxUpdateUIEvent& rEvent
1252)
0e320a79 1253{
c4955899
DW
1254 rEvent.Enable(CanPaste());
1255} // end of wxTextCtrl::OnUpdatePaste
0e320a79 1256
c4955899
DW
1257void wxTextCtrl::OnUpdateUndo(
1258 wxUpdateUIEvent& rEvent
1259)
0e320a79 1260{
c4955899
DW
1261 rEvent.Enable(CanUndo());
1262} // end of wxTextCtrl::OnUpdateUndo
1263
1264void wxTextCtrl::OnUpdateRedo(
1265 wxUpdateUIEvent& rEvent
1266)
1267{
1268 rEvent.Enable(CanRedo());
1269} // end of wxTextCtrl::OnUpdateRedo
0e320a79 1270
c4955899
DW
1271bool wxTextCtrl::SetBackgroundColour(
1272 const wxColour& rColour
1273)
0e320a79 1274{
c4955899
DW
1275 if (m_bIsMLE)
1276 ::WinSendMsg(GetHwnd(), MLM_SETBACKCOLOR, (MPARAM)rColour.GetPixel(), MLE_INDEX);
1277 return TRUE;
1278} // end of wxTextCtrl::SetBackgroundColour
1279
1280bool wxTextCtrl::SetForegroundColour(
1281 const wxColour& rColour
1282)
1283{
1284 if (m_bIsMLE)
1285 ::WinSendMsg(GetHwnd(), MLM_SETTEXTCOLOR, (MPARAM)rColour.GetPixel(), MLE_INDEX);
1286 return TRUE;
1287} // end of wxTextCtrl::SetForegroundColour
d90895ac 1288
39c26ef2
DW
1289bool wxTextCtrl::SetStyle(
1290 long lStart
1291, long lEnd
1292, const wxTextAttr& rStyle
1293)
1294{
1295 HWND hWnd = GetHwnd();
1296
1297 if (lStart > lEnd)
1298 {
1299 long lTmp = lStart;
1300
1301 lStart = lEnd;
1302 lEnd = lTmp;
1303 }
1304
1305 //
1306 // We can only change the format of the selection, so select the range we
1307 // want and restore the old selection later
1308 //
1309 long lStartOld;
1310 long lEndOld;
1311
1312 GetSelection( &lStartOld
1313 ,&lEndOld
1314 );
1315
1316 //
1317 // But do we really have to change the selection?
1318 //
1319 bool bChangeSel = lStart != lStartOld ||
1320 lEnd != lEndOld;
1321
1322 if (bChangeSel)
1323 {
1324 if (m_bIsMLE)
1325 ::WinSendMsg(hWnd, MLM_SETSEL, MPFROM2SHORT((USHORT)lStart, (USHORT)lEnd), 0);
1326 else
1327 ::WinSendMsg(hWnd, EM_SETSEL, MPFROM2SHORT((USHORT)lStart, (USHORT)lEnd), 0);
1328 }
1329
1330 //
1331 // TODO:: finish this part
1332 //
1333 return TRUE;
1334} // end of wxTextCtrl::SetStyle
1335