]> git.saurik.com Git - wxWidgets.git/blame - src/mac/textctrl.cpp
compilation fix
[wxWidgets.git] / src / mac / textctrl.cpp
CommitLineData
e9576ca5
SC
1/////////////////////////////////////////////////////////////////////////////
2// Name: textctrl.cpp
3// Purpose: wxTextCtrl
4// Author: AUTHOR
5// Modified by:
6// Created: ??/??/98
7// RCS-ID: $Id$
8// Copyright: (c) AUTHOR
ed8c2780 9// Licence: wxWindows licence
e9576ca5
SC
10/////////////////////////////////////////////////////////////////////////////
11
12#ifdef __GNUG__
13#pragma implementation "textctrl.h"
14#endif
15
fedad417
GD
16#include "wx/defs.h"
17
18#if wxUSE_TEXTCTRL
19
f5c6eb5c 20#ifdef __DARWIN__
03e11df5
GD
21 #include <sys/types.h>
22 #include <sys/stat.h>
e9576ca5 23#else
03e11df5 24 #include <stat.h>
e9576ca5 25#endif
2b5f62a0
VZ
26
27#if wxUSE_STD_IOSTREAM
28 #if wxUSE_IOSTREAMH
29 #include <fstream.h>
30 #else
31 #include <fstream>
32 #endif
33#endif
e9576ca5 34
03e11df5 35#include "wx/app.h"
5fde6fcc 36#include "wx/dc.h"
03e11df5 37#include "wx/button.h"
422644a3 38#include "wx/toplevel.h"
e9576ca5 39#include "wx/textctrl.h"
90b22aca
SC
40#include "wx/notebook.h"
41#include "wx/tabctrl.h"
e9576ca5
SC
42#include "wx/settings.h"
43#include "wx/filefn.h"
44#include "wx/utils.h"
45
46#if defined(__BORLANDC__) && !defined(__WIN32__)
03e11df5 47 #include <alloc.h>
f5c6eb5c 48#elif !defined(__MWERKS__) && !defined(__GNUWIN32) && !defined(__DARWIN__)
03e11df5 49 #include <malloc.h>
e9576ca5
SC
50#endif
51
66a09d47
SC
52#ifndef __DARWIN__
53#include <Scrap.h>
1b2b1638
SC
54#endif
55#include <MacTextEditor.h>
56#include "ATSUnicode.h"
57#include "TextCommon.h"
58#include "TextEncodingConverter.h"
59#include "wx/mac/uma.h"
72055702 60
29b30405
SC
61#define TE_UNLIMITED_LENGTH 0xFFFFFFFFUL
62
7759d157 63extern wxApp *wxTheApp ;
e600c175 64extern wxControl *wxFindControlFromMacControl(ControlHandle inControl ) ;
72055702 65
7759d157
SC
66// CS:TODO we still have a problem getting properly at the text events of a control because under Carbon
67// the MLTE engine registers itself for the key events thus the normal flow never occurs, the only measure for the
68// moment is to avoid setting the true focus on the control, the proper solution at the end would be to have
69// an alternate path for carbon key events that routes automatically into the same wx flow of events
72055702
SC
70
71#include "MacTextEditor.h"
72
73/* part codes */
74
75/* kmUPTextPart is the part code we return to indicate the user has clicked
ed8c2780 76 in the text area of our control */
72055702
SC
77#define kmUPTextPart 1
78
79/* kmUPScrollPart is the part code we return to indicate the user has clicked
ed8c2780 80 in the scroll bar part of the control. */
72055702
SC
81#define kmUPScrollPart 2
82
83
84/* routines for using existing user pane controls.
ed8c2780
RR
85 These routines are useful for cases where you would like to use an
86 existing user pane control in, say, a dialog window as a scrolling
87 text edit field.*/
88
72055702 89/* mUPOpenControl initializes a user pane control so it will be drawn
ed8c2780
RR
90 and will behave as a scrolling text edit field inside of a window.
91 This routine performs all of the initialization steps necessary,
92 except it does not create the user pane control itself. theControl
93 should refer to a user pane control that you have either created
94 yourself or extracted from a dialog's control heirarchy using
95 the GetDialogItemAsControl routine. */
29b30405 96OSStatus mUPOpenControl(ControlHandle theControl, long wxStyle);
72055702
SC
97
98/* Utility Routines */
99
72055702 100enum {
ed8c2780 101 kShiftKeyCode = 56
72055702
SC
102};
103
104/* kUserClickedToFocusPart is a part code we pass to the SetKeyboardFocus
ed8c2780
RR
105 routine. In our focus switching routine this part code is understood
106 as meaning 'the user has clicked in the control and we need to switch
107 the current focus to ourselves before we can continue'. */
72055702
SC
108#define kUserClickedToFocusPart 100
109
110
111/* kmUPClickScrollDelayTicks is a time measurement in ticks used to
ed8c2780
RR
112 slow the speed of 'auto scrolling' inside of our clickloop routine.
113 This value prevents the text from wizzzzzing by while the mouse
114 is being held down inside of the text area. */
72055702
SC
115#define kmUPClickScrollDelayTicks 3
116
117
118/* STPTextPaneVars is a structure used for storing the the mUP Control's
ed8c2780
RR
119 internal variables and state information. A handle to this record is
120 stored in the pane control's reference value field using the
121 SetControlReference routine. */
72055702
SC
122
123typedef struct {
ed8c2780
RR
124 /* OS records referenced */
125 TXNObject fTXNRec; /* the txn record */
126 TXNFrameID fTXNFrame; /* the txn frame ID */
127 ControlHandle fUserPaneRec; /* handle to the user pane control */
128 WindowPtr fOwner; /* window containing control */
129 GrafPtr fDrawingEnvironment; /* grafport where control is drawn */
130 /* flags */
131 Boolean fInFocus; /* true while the focus rect is drawn around the control */
132 Boolean fIsActive; /* true while the control is drawn in the active state */
133 Boolean fTEActive; /* reflects the activation state of the text edit record */
134 Boolean fInDialogWindow; /* true if displayed in a dialog window */
135 /* calculated locations */
136 Rect fRTextArea; /* area where the text is drawn */
137 Rect fRFocusOutline; /* rectangle used to draw the focus box */
138 Rect fRTextOutline; /* rectangle used to draw the border */
139 RgnHandle fTextBackgroundRgn; /* background region for the text, erased before calling TEUpdate */
140 /* our focus advance override routine */
141 EventHandlerUPP handlerUPP;
142 EventHandlerRef handlerRef;
29e4a190 143 bool fMultiline ;
72055702
SC
144} STPTextPaneVars;
145
146
147
148
149/* Univerals Procedure Pointer variables used by the
ed8c2780
RR
150 mUP Control. These variables are set up
151 the first time that mUPOpenControl is called. */
72055702
SC
152ControlUserPaneDrawUPP gTPDrawProc = NULL;
153ControlUserPaneHitTestUPP gTPHitProc = NULL;
154ControlUserPaneTrackingUPP gTPTrackProc = NULL;
155ControlUserPaneIdleUPP gTPIdleProc = NULL;
156ControlUserPaneKeyDownUPP gTPKeyProc = NULL;
157ControlUserPaneActivateUPP gTPActivateProc = NULL;
158ControlUserPaneFocusUPP gTPFocusProc = NULL;
159
72055702 160/* TPActivatePaneText activates or deactivates the text edit record
ed8c2780
RR
161 according to the value of setActive. The primary purpose of this
162 routine is to ensure each call is only made once. */
72055702 163static void TPActivatePaneText(STPTextPaneVars **tpvars, Boolean setActive) {
ed8c2780
RR
164 STPTextPaneVars *varsp;
165 varsp = *tpvars;
166 if (varsp->fTEActive != setActive) {
167
168 varsp->fTEActive = setActive;
169
170 TXNActivate(varsp->fTXNRec, varsp->fTXNFrame, varsp->fTEActive);
7759d157 171
ed8c2780
RR
172 if (varsp->fInFocus)
173 TXNFocus( varsp->fTXNRec, varsp->fTEActive);
174 }
72055702
SC
175}
176
177
178/* TPFocusPaneText set the focus state for the text record. */
179static void TPFocusPaneText(STPTextPaneVars **tpvars, Boolean setFocus) {
ed8c2780
RR
180 STPTextPaneVars *varsp;
181 varsp = *tpvars;
182 if (varsp->fInFocus != setFocus) {
183 varsp->fInFocus = setFocus;
184 TXNFocus( varsp->fTXNRec, varsp->fInFocus);
185 }
72055702
SC
186}
187
188
189/* TPPaneDrawProc is called to redraw the control and for update events
ed8c2780
RR
190 referring to the control. This routine erases the text area's background,
191 and redraws the text. This routine assumes the scroll bar has been
192 redrawn by a call to DrawControls. */
72055702 193static pascal void TPPaneDrawProc(ControlRef theControl, ControlPartCode thePart) {
ed8c2780
RR
194 STPTextPaneVars **tpvars, *varsp;
195 char state;
196 Rect bounds;
197 /* set up our globals */
fd4393b8 198
ed8c2780
RR
199 tpvars = (STPTextPaneVars **) GetControlReference(theControl);
200 if (tpvars != NULL) {
201 state = HGetState((Handle) tpvars);
202 HLock((Handle) tpvars);
203 varsp = *tpvars;
204
205 /* save the drawing state */
206 SetPort((**tpvars).fDrawingEnvironment);
e600c175 207 /* verify our boundary */
ed8c2780 208 GetControlBounds(theControl, &bounds);
e600c175
SC
209
210 wxMacWindowClipper clipper( wxFindControlFromMacControl(theControl ) ) ;
29e4a190
RR
211 if ( ! EqualRect(&bounds, &varsp->fRFocusOutline) ) {
212 // scrollbar is on the border, we add one
213 Rect oldbounds = varsp->fRFocusOutline ;
214 InsetRect( &oldbounds , -1 , -1 ) ;
215
216 InvalWindowRect( GetControlOwner( theControl ) , &oldbounds ) ;
217 SetRect(&varsp->fRFocusOutline, bounds.left, bounds.top, bounds.right, bounds.bottom);
218 SetRect(&varsp->fRTextOutline, bounds.left, bounds.top, bounds.right, bounds.bottom);
219 SetRect(&varsp->fRTextArea, bounds.left + 2 , bounds.top + (varsp->fMultiline ? 0 : 2) ,
220 bounds.right - (varsp->fMultiline ? 0 : 2), bounds.bottom - (varsp->fMultiline ? 0 : 2));
ed8c2780 221 RectRgn(varsp->fTextBackgroundRgn, &varsp->fRTextOutline);
29e4a190
RR
222 TXNSetFrameBounds( varsp->fTXNRec, varsp->fRTextArea.top, varsp->fRTextArea.left,
223 varsp->fRTextArea.bottom, varsp->fRTextArea.right, varsp->fTXNFrame);
ed8c2780
RR
224 }
225
226 /* update the text region */
29e4a190
RR
227 RGBColor white = { 65535 , 65535 , 65535 } ;
228 RGBBackColor( &white ) ;
ed8c2780
RR
229 EraseRgn(varsp->fTextBackgroundRgn);
230 TXNDraw(varsp->fTXNRec, NULL);
231 /* restore the drawing environment */
232 /* draw the text frame and focus frame (if necessary) */
233 DrawThemeEditTextFrame(&varsp->fRTextOutline, varsp->fIsActive ? kThemeStateActive: kThemeStateInactive);
234 if ((**tpvars).fIsActive && varsp->fInFocus) DrawThemeFocusRect(&varsp->fRFocusOutline, true);
235 /* release our globals */
236 HSetState((Handle) tpvars, state);
e600c175 237
ed8c2780 238 }
72055702
SC
239}
240
241
242/* TPPaneHitTestProc is called when the control manager would
ed8c2780
RR
243 like to determine what part of the control the mouse resides over.
244 We also call this routine from our tracking proc to determine how
245 to handle mouse clicks. */
72055702 246static pascal ControlPartCode TPPaneHitTestProc(ControlHandle theControl, Point where) {
ed8c2780
RR
247 STPTextPaneVars **tpvars;
248 ControlPartCode result;
249 char state;
250 /* set up our locals and lock down our globals*/
251 result = 0;
252 tpvars = (STPTextPaneVars **) GetControlReference(theControl);
253 if (tpvars != NULL) {
254 state = HGetState((Handle) tpvars);
255 HLock((Handle) tpvars);
256 /* find the region where we clicked */
257 if (PtInRect(where, &(**tpvars).fRTextArea)) {
258 result = kmUPTextPart;
259 } else result = 0;
260 /* release oure globals */
261 HSetState((Handle) tpvars, state);
262 }
263 return result;
72055702
SC
264}
265
266
267
268
269
270/* TPPaneTrackingProc is called when the mouse is being held down
ed8c2780
RR
271 over our control. This routine handles clicks in the text area
272 and in the scroll bar. */
72055702 273static pascal ControlPartCode TPPaneTrackingProc(ControlHandle theControl, Point startPt, ControlActionUPP actionProc) {
ed8c2780
RR
274 STPTextPaneVars **tpvars, *varsp;
275 char state;
276 ControlPartCode partCodeResult;
277 /* make sure we have some variables... */
278 partCodeResult = 0;
279 tpvars = (STPTextPaneVars **) GetControlReference(theControl);
280 if (tpvars != NULL) {
281 /* lock 'em down */
282 state = HGetState((Handle) tpvars);
283 HLock((Handle) tpvars);
284 varsp = *tpvars;
285 /* we don't do any of these functions unless we're in focus */
286 if ( ! varsp->fInFocus) {
287 WindowPtr owner;
288 owner = GetControlOwner(theControl);
289 ClearKeyboardFocus(owner);
290 SetKeyboardFocus(owner, theControl, kUserClickedToFocusPart);
291 }
292 /* find the location for the click */
293 switch (TPPaneHitTestProc(theControl, startPt)) {
294
295 /* handle clicks in the text part */
296 case kmUPTextPart:
297 { SetPort((**tpvars).fDrawingEnvironment);
e600c175 298 wxMacWindowClipper clipper( wxFindControlFromMacControl(theControl ) ) ;
1b2b1638 299 TXNClick( varsp->fTXNRec, (const EventRecord*) wxTheApp->MacGetCurrentEvent());
ed8c2780
RR
300 }
301 break;
302
303 }
304
305 HSetState((Handle) tpvars, state);
306 }
307 return partCodeResult;
72055702
SC
308}
309
310
311/* TPPaneIdleProc is our user pane idle routine. When our text field
ed8c2780 312 is active and in focus, we use this routine to set the cursor. */
72055702 313static pascal void TPPaneIdleProc(ControlHandle theControl) {
ed8c2780
RR
314 STPTextPaneVars **tpvars, *varsp;
315 /* set up locals */
316 tpvars = (STPTextPaneVars **) GetControlReference(theControl);
317 if (tpvars != NULL) {
318 /* if we're not active, then we have nothing to say about the cursor */
319 if ((**tpvars).fIsActive) {
320 char state;
321 Rect bounds;
322 Point mousep;
323 /* lock down the globals */
324 state = HGetState((Handle) tpvars);
325 HLock((Handle) tpvars);
326 varsp = *tpvars;
327 /* get the current mouse coordinates (in our window) */
1b2b1638 328 SetPortWindowPort(GetControlOwner(theControl));
e600c175 329 wxMacWindowClipper clipper( wxFindControlFromMacControl(theControl ) ) ;
ed8c2780
RR
330 GetMouse(&mousep);
331 /* there's a 'focus thing' and an 'unfocused thing' */
332 if (varsp->fInFocus) {
333 /* flash the cursor */
334 SetPort((**tpvars).fDrawingEnvironment);
335 TXNIdle(varsp->fTXNRec);
336 /* set the cursor */
337 if (PtInRect(mousep, &varsp->fRTextArea)) {
338 RgnHandle theRgn;
339 RectRgn((theRgn = NewRgn()), &varsp->fRTextArea);
340 TXNAdjustCursor(varsp->fTXNRec, theRgn);
341 DisposeRgn(theRgn);
2b5f62a0
VZ
342 }
343 else
344 {
345 // SetThemeCursor(kThemeArrowCursor);
346 }
ed8c2780
RR
347 } else {
348 /* if it's in our bounds, set the cursor */
349 GetControlBounds(theControl, &bounds);
350 if (PtInRect(mousep, &bounds))
2b5f62a0
VZ
351 {
352 // SetThemeCursor(kThemeArrowCursor);
353 }
ed8c2780
RR
354 }
355
356 HSetState((Handle) tpvars, state);
357 }
358 }
72055702
SC
359}
360
361
362/* TPPaneKeyDownProc is called whenever a keydown event is directed
ed8c2780
RR
363 at our control. Here, we direct the keydown event to the text
364 edit record and redraw the scroll bar and text field as appropriate. */
72055702 365static pascal ControlPartCode TPPaneKeyDownProc(ControlHandle theControl,
ed8c2780
RR
366 SInt16 keyCode, SInt16 charCode, SInt16 modifiers) {
367 STPTextPaneVars **tpvars;
368 tpvars = (STPTextPaneVars **) GetControlReference(theControl);
369 if (tpvars != NULL) {
370 if ((**tpvars).fInFocus) {
371 /* turn autoscrolling on and send the key event to text edit */
372 SetPort((**tpvars).fDrawingEnvironment);
e600c175 373 wxMacWindowClipper clipper( wxFindControlFromMacControl(theControl ) ) ;
8c82361c
SC
374 EventRecord ev ;
375 memset( &ev , 0 , sizeof( ev ) ) ;
376 ev.what = keyDown ;
377 ev.modifiers = modifiers ;
45c8d9e5 378 ev.message = (( keyCode << 8 ) & keyCodeMask ) + ( charCode & charCodeMask ) ;
8c82361c 379 TXNKeyDown( (**tpvars).fTXNRec, &ev);
ed8c2780
RR
380 }
381 }
382 return kControlEntireControl;
72055702
SC
383}
384
385
386/* TPPaneActivateProc is called when the window containing
ed8c2780
RR
387 the user pane control receives activate events. Here, we redraw
388 the control and it's text as necessary for the activation state. */
72055702 389static pascal void TPPaneActivateProc(ControlHandle theControl, Boolean activating) {
ed8c2780
RR
390 Rect bounds;
391 STPTextPaneVars **tpvars, *varsp;
392 char state;
393 /* set up locals */
394 tpvars = (STPTextPaneVars **) GetControlReference(theControl);
395 if (tpvars != NULL) {
396 state = HGetState((Handle) tpvars);
397 HLock((Handle) tpvars);
398 varsp = *tpvars;
399 /* de/activate the text edit record */
400 SetPort((**tpvars).fDrawingEnvironment);
e600c175 401 wxMacWindowClipper clipper( wxFindControlFromMacControl(theControl ) ) ;
ed8c2780
RR
402 GetControlBounds(theControl, &bounds);
403 varsp->fIsActive = activating;
404 TPActivatePaneText(tpvars, varsp->fIsActive && varsp->fInFocus);
405 /* redraw the frame */
406 DrawThemeEditTextFrame(&varsp->fRTextOutline, varsp->fIsActive ? kThemeStateActive: kThemeStateInactive);
407 if (varsp->fInFocus) DrawThemeFocusRect(&varsp->fRFocusOutline, varsp->fIsActive);
408 HSetState((Handle) tpvars, state);
409 }
72055702
SC
410}
411
412
413/* TPPaneFocusProc is called when every the focus changes to or
ed8c2780
RR
414 from our control. Herein, switch the focus appropriately
415 according to the parameters and redraw the control as
416 necessary. */
72055702 417static pascal ControlPartCode TPPaneFocusProc(ControlHandle theControl, ControlFocusPart action) {
ed8c2780
RR
418 ControlPartCode focusResult;
419 STPTextPaneVars **tpvars, *varsp;
420 char state;
421 /* set up locals */
422 focusResult = kControlFocusNoPart;
423 tpvars = (STPTextPaneVars **) GetControlReference(theControl);
424 if (tpvars != NULL) {
425 state = HGetState((Handle) tpvars);
426 HLock((Handle) tpvars);
427 varsp = *tpvars;
428 /* if kControlFocusPrevPart and kControlFocusNextPart are received when the user is
429 tabbing forwards (or shift tabbing backwards) through the items in the dialog,
430 and kControlFocusNextPart will be received. When the user clicks in our field
431 and it is not the current focus, then the constant kUserClickedToFocusPart will
432 be received. The constant kControlFocusNoPart will be received when our control
433 is the current focus and the user clicks in another control. In your focus routine,
434 you should respond to these codes as follows:
435
436 kControlFocusNoPart - turn off focus and return kControlFocusNoPart. redraw
437 the control and the focus rectangle as necessary.
438
439 kControlFocusPrevPart or kControlFocusNextPart - toggle focus on or off
440 depending on its current state. redraw the control and the focus rectangle
441 as appropriate for the new focus state. If the focus state is 'off', return the constant
442 kControlFocusNoPart, otherwise return a non-zero part code.
443 kUserClickedToFocusPart - is a constant defined for this example. You should
444 define your own value for handling click-to-focus type events. */
445 /* save the drawing state */
446 SetPort((**tpvars).fDrawingEnvironment);
e600c175 447 wxMacWindowClipper clipper( wxFindControlFromMacControl(theControl ) ) ;
ed8c2780
RR
448 /* calculate the next highlight state */
449 switch (action) {
450 default:
451 case kControlFocusNoPart:
452 TPFocusPaneText(tpvars, false);
453 focusResult = kControlFocusNoPart;
454 break;
455 case kUserClickedToFocusPart:
456 TPFocusPaneText(tpvars, true);
457 focusResult = 1;
458 break;
459 case kControlFocusPrevPart:
460 case kControlFocusNextPart:
461 TPFocusPaneText(tpvars, ( ! varsp->fInFocus));
462 focusResult = varsp->fInFocus ? 1 : kControlFocusNoPart;
463 break;
464 }
465 TPActivatePaneText(tpvars, varsp->fIsActive && varsp->fInFocus);
466 /* redraw the text fram and focus rectangle to indicate the
467 new focus state */
468 DrawThemeEditTextFrame(&varsp->fRTextOutline, varsp->fIsActive ? kThemeStateActive: kThemeStateInactive);
469 DrawThemeFocusRect(&varsp->fRFocusOutline, varsp->fIsActive && varsp->fInFocus);
470 /* done */
1b2b1638 471 HSetState((Handle) tpvars, state);
ed8c2780 472 }
1b2b1638 473 return focusResult;
72055702
SC
474}
475
72055702 476
1b2b1638 477/* mUPOpenControl initializes a user pane control so it will be drawn
29e4a190
RR
478 and will behave as a scrolling text edit field inside of a window.
479 This routine performs all of the initialization steps necessary,
480 except it does not create the user pane control itself. theControl
481 should refer to a user pane control that you have either created
482 yourself or extracted from a dialog's control heirarchy using
483 the GetDialogItemAsControl routine. */
29b30405 484OSStatus mUPOpenControl(ControlHandle theControl, long wxStyle )
1b2b1638 485{
29e4a190
RR
486 Rect bounds;
487 WindowRef theWindow;
488 STPTextPaneVars **tpvars, *varsp;
29b30405 489 OSStatus err = noErr ;
29e4a190
RR
490 RGBColor rgbWhite = {0xFFFF, 0xFFFF, 0xFFFF};
491 TXNBackground tback;
492
493 /* set up our globals */
494 if (gTPDrawProc == NULL) gTPDrawProc = NewControlUserPaneDrawUPP(TPPaneDrawProc);
495 if (gTPHitProc == NULL) gTPHitProc = NewControlUserPaneHitTestUPP(TPPaneHitTestProc);
496 if (gTPTrackProc == NULL) gTPTrackProc = NewControlUserPaneTrackingUPP(TPPaneTrackingProc);
497 if (gTPIdleProc == NULL) gTPIdleProc = NewControlUserPaneIdleUPP(TPPaneIdleProc);
498 if (gTPKeyProc == NULL) gTPKeyProc = NewControlUserPaneKeyDownUPP(TPPaneKeyDownProc);
499 if (gTPActivateProc == NULL) gTPActivateProc = NewControlUserPaneActivateUPP(TPPaneActivateProc);
500 if (gTPFocusProc == NULL) gTPFocusProc = NewControlUserPaneFocusUPP(TPPaneFocusProc);
501
502 /* allocate our private storage */
503 tpvars = (STPTextPaneVars **) NewHandleClear(sizeof(STPTextPaneVars));
504 SetControlReference(theControl, (long) tpvars);
505 HLock((Handle) tpvars);
506 varsp = *tpvars;
507 /* set the initial settings for our private data */
29b30405 508 varsp->fMultiline = wxStyle & wxTE_MULTILINE ;
29e4a190
RR
509 varsp->fInFocus = false;
510 varsp->fIsActive = true;
511 varsp->fTEActive = true; // in order to get a deactivate
512 varsp->fUserPaneRec = theControl;
513 theWindow = varsp->fOwner = GetControlOwner(theControl);
1b2b1638
SC
514
515 varsp->fDrawingEnvironment = (GrafPtr) GetWindowPort(theWindow);
72055702 516
1b2b1638
SC
517 varsp->fInDialogWindow = ( GetWindowKind(varsp->fOwner) == kDialogWindowKind );
518 /* set up the user pane procedures */
519 SetControlData(theControl, kControlEntireControl, kControlUserPaneDrawProcTag, sizeof(gTPDrawProc), &gTPDrawProc);
520 SetControlData(theControl, kControlEntireControl, kControlUserPaneHitTestProcTag, sizeof(gTPHitProc), &gTPHitProc);
521 SetControlData(theControl, kControlEntireControl, kControlUserPaneTrackingProcTag, sizeof(gTPTrackProc), &gTPTrackProc);
522 SetControlData(theControl, kControlEntireControl, kControlUserPaneIdleProcTag, sizeof(gTPIdleProc), &gTPIdleProc);
523 SetControlData(theControl, kControlEntireControl, kControlUserPaneKeyDownProcTag, sizeof(gTPKeyProc), &gTPKeyProc);
524 SetControlData(theControl, kControlEntireControl, kControlUserPaneActivateProcTag, sizeof(gTPActivateProc), &gTPActivateProc);
525 SetControlData(theControl, kControlEntireControl, kControlUserPaneFocusProcTag, sizeof(gTPFocusProc), &gTPFocusProc);
526 /* calculate the rectangles used by the control */
527 GetControlBounds(theControl, &bounds);
528 SetRect(&varsp->fRFocusOutline, bounds.left, bounds.top, bounds.right, bounds.bottom);
529 SetRect(&varsp->fRTextOutline, bounds.left, bounds.top, bounds.right, bounds.bottom);
530 SetRect(&varsp->fRTextArea, bounds.left + 2 , bounds.top + (varsp->fMultiline ? 0 : 2) ,
531 bounds.right - (varsp->fMultiline ? 0 : 2), bounds.bottom - (varsp->fMultiline ? 0 : 2));
532 /* calculate the background region for the text. In this case, it's kindof
533 and irregular region because we're setting the scroll bar a little ways inside
534 of the text area. */
535 RectRgn((varsp->fTextBackgroundRgn = NewRgn()), &varsp->fRTextOutline);
72055702 536
1b2b1638
SC
537 /* set up the drawing environment */
538 SetPort(varsp->fDrawingEnvironment);
539
540 /* create the new edit field */
29b30405
SC
541
542 TXNFrameOptions frameOptions =
543 kTXNDontDrawCaretWhenInactiveMask ;
544 if ( ! ( wxStyle & wxTE_NOHIDESEL ) )
2b5f62a0
VZ
545 frameOptions |= kTXNDontDrawSelectionWhenInactiveMask ;
546
547 if ( wxStyle & wxTE_MULTILINE )
548 {
549 if ( ! ( wxStyle & wxTE_DONTWRAP ) )
550 frameOptions |= kTXNAlwaysWrapAtViewEdgeMask ;
551 else
552 {
553 frameOptions |= kTXNAlwaysWrapAtViewEdgeMask ;
554 frameOptions |= kTXNWantHScrollBarMask ;
555 }
556
557 if ( !(wxStyle & wxTE_NO_VSCROLL ) )
558 frameOptions |= kTXNWantVScrollBarMask ;
559 }
560 else
561 frameOptions |= kTXNSingleLineOnlyMask ;
562
563 if ( wxStyle & wxTE_READONLY )
564 frameOptions |= kTXNReadOnlyMask ;
29b30405 565
1b2b1638 566 TXNNewObject(NULL, varsp->fOwner, &varsp->fRTextArea,
2b5f62a0 567 frameOptions ,
1b2b1638
SC
568 kTXNTextEditStyleFrameType,
569 kTXNTextensionFile,
570 kTXNSystemDefaultEncoding,
571 &varsp->fTXNRec, &varsp->fTXNFrame, (TXNObjectRefcon) tpvars);
572
29e4a190
RR
573 Str255 fontName ;
574 SInt16 fontSize ;
575 Style fontStyle ;
576
577 GetThemeFont(kThemeSmallSystemFont , GetApplicationScript() , fontName , &fontSize , &fontStyle ) ;
1b2b1638 578
29e4a190
RR
579 TXNTypeAttributes typeAttr[] =
580 {
402679b0 581 { kTXNQDFontNameAttribute , kTXNQDFontNameAttributeSize , { (void*) fontName } } ,
29e4a190
RR
582 { kTXNQDFontSizeAttribute , kTXNFontSizeAttributeSize , { (void*) (fontSize << 16) } } ,
583 { kTXNQDFontStyleAttribute , kTXNQDFontStyleAttributeSize , { (void*) normal } } ,
402679b0 584 } ;
1b2b1638 585
29b30405
SC
586 err = TXNSetTypeAttributes (varsp->fTXNRec, sizeof( typeAttr ) / sizeof(TXNTypeAttributes) , typeAttr,
587 kTXNStartOffset,
588 kTXNEndOffset);
1b2b1638
SC
589 /* set the field's background */
590 tback.bgType = kTXNBackgroundTypeRGB;
591 tback.bg.color = rgbWhite;
592 TXNSetBackground( varsp->fTXNRec, &tback);
29e4a190 593
1b2b1638
SC
594 /* unlock our storage */
595 HUnlock((Handle) tpvars);
596 /* perform final activations and setup for our text field. Here,
597 we assume that the window is going to be the 'active' window. */
598 TPActivatePaneText(tpvars, varsp->fIsActive && varsp->fInFocus);
ed8c2780 599 /* all done */
29b30405 600 return err;
72055702
SC
601}
602
1b2b1638
SC
603
604
605
72055702
SC
606#if !USE_SHARED_LIBRARY
607IMPLEMENT_DYNAMIC_CLASS(wxTextCtrl, wxControl)
608
609BEGIN_EVENT_TABLE(wxTextCtrl, wxControl)
ed8c2780
RR
610 EVT_DROP_FILES(wxTextCtrl::OnDropFiles)
611 EVT_CHAR(wxTextCtrl::OnChar)
72055702
SC
612 EVT_MENU(wxID_CUT, wxTextCtrl::OnCut)
613 EVT_MENU(wxID_COPY, wxTextCtrl::OnCopy)
614 EVT_MENU(wxID_PASTE, wxTextCtrl::OnPaste)
615 EVT_MENU(wxID_UNDO, wxTextCtrl::OnUndo)
616 EVT_MENU(wxID_REDO, wxTextCtrl::OnRedo)
617
618 EVT_UPDATE_UI(wxID_CUT, wxTextCtrl::OnUpdateCut)
619 EVT_UPDATE_UI(wxID_COPY, wxTextCtrl::OnUpdateCopy)
620 EVT_UPDATE_UI(wxID_PASTE, wxTextCtrl::OnUpdatePaste)
621 EVT_UPDATE_UI(wxID_UNDO, wxTextCtrl::OnUpdateUndo)
622 EVT_UPDATE_UI(wxID_REDO, wxTextCtrl::OnUpdateRedo)
623END_EVENT_TABLE()
624#endif
625
626// Text item
627wxTextCtrl::wxTextCtrl()
628{
402679b0
SC
629 m_macTE = NULL ;
630 m_macTXN = NULL ;
631 m_macTXNvars = NULL ;
1b2b1638
SC
632 m_macUsesTXN = false ;
633 m_editable = true ;
29b30405 634 m_maxLength = TE_UNLIMITED_LENGTH ;
1b2b1638
SC
635}
636
637wxTextCtrl::~wxTextCtrl()
638{
639 if ( m_macUsesTXN )
640 {
402679b0 641 SetControlReference((ControlHandle)m_macControl, 0) ;
1b2b1638 642 TXNDeleteObject((TXNObject)m_macTXN);
1b2b1638
SC
643 /* delete our private storage */
644 DisposeHandle((Handle) m_macTXNvars);
645 /* zero the control reference */
1b2b1638 646 }
72055702
SC
647}
648
649const short kVerticalMargin = 2 ;
650const short kHorizontalMargin = 2 ;
651
652bool wxTextCtrl::Create(wxWindow *parent, wxWindowID id,
ed8c2780 653 const wxString& st,
72055702
SC
654 const wxPoint& pos,
655 const wxSize& size, long style,
656 const wxValidator& validator,
657 const wxString& name)
658{
402679b0
SC
659 m_macTE = NULL ;
660 m_macTXN = NULL ;
661 m_macTXNvars = NULL ;
1b2b1638
SC
662 m_macUsesTXN = false ;
663 m_editable = true ;
664
665 m_macUsesTXN = ! (style & wxTE_PASSWORD ) ;
666
667 m_macUsesTXN &= (TXNInitTextension != (void*) kUnresolvedCFragSymbolAddress) ;
668
72055702
SC
669 // base initialization
670 if ( !CreateBase(parent, id, pos, size, style, validator, name) )
671 return FALSE;
672
ed8c2780 673 wxSize mySize = size ;
29e4a190
RR
674 if ( m_macUsesTXN )
675 {
676 m_macHorizontalBorder = 5 ; // additional pixels around the real control
677 m_macVerticalBorder = 3 ;
678 }
679 else
ed8c2780
RR
680 {
681 m_macHorizontalBorder = 5 ; // additional pixels around the real control
682 m_macVerticalBorder = 5 ;
683 }
ed8c2780
RR
684
685
686 Rect bounds ;
687 Str255 title ;
402679b0 688 /*
ed8c2780
RR
689 if ( mySize.y == -1 )
690 {
402679b0
SC
691 mySize.y = 13 ;
692 if ( m_windowStyle & wxTE_MULTILINE )
693 mySize.y *= 5 ;
ed8c2780
RR
694
695 mySize.y += 2 * m_macVerticalBorder ;
696 }
402679b0 697 */
ed8c2780 698 MacPreControlCreate( parent , id , "" , pos , mySize ,style, validator , name , &bounds , title ) ;
72055702
SC
699
700 if ( m_windowStyle & wxTE_MULTILINE )
701 {
702 wxASSERT_MSG( !(m_windowStyle & wxTE_PROCESS_ENTER),
703 wxT("wxTE_PROCESS_ENTER style is ignored for multiline text controls (they always process it)") );
704
705 m_windowStyle |= wxTE_PROCESS_ENTER;
706 }
707
29b30405
SC
708 if ( m_windowStyle & wxTE_READONLY)
709 {
710 m_editable = FALSE ;
711 }
72055702 712
1b2b1638 713 if ( !m_macUsesTXN )
9c641c05 714 {
29e4a190
RR
715 m_macControl = ::NewControl( MAC_WXHWND(parent->MacGetRootWindow()) , &bounds , "\p" , true , 0 , 0 , 1,
716 (style & wxTE_PASSWORD) ? kControlEditTextPasswordProc : kControlEditTextProc , (long) this ) ;
717 long size ;
1b2b1638
SC
718 ::GetControlData((ControlHandle) m_macControl , 0, kControlEditTextTEHandleTag , sizeof( TEHandle ) , (char*) &((TEHandle) m_macTE) , &size ) ;
719
ed8c2780 720 }
29e4a190
RR
721 else
722 {
1b2b1638
SC
723 short featurSet;
724
29b30405 725 featurSet = kControlSupportsEmbedding | kControlSupportsFocus | kControlWantsIdle
1b2b1638
SC
726 | kControlWantsActivate | kControlHandlesTracking | kControlHasSpecialBackground
727 | kControlGetsFocusOnClick | kControlSupportsLiveFeedback;
728 /* create the control */
175a6271 729 m_macControl = NewControl(MAC_WXHWND(parent->MacGetRootWindow()), &bounds, "\p", true, featurSet, 0, featurSet, kControlUserPaneProc, 0);
1b2b1638 730 /* set up the mUP specific features and data */
2b5f62a0 731 mUPOpenControl((ControlHandle) m_macControl, m_windowStyle );
29e4a190
RR
732 if ( parent )
733 {
734 parent->MacGetTopLevelWindow()->MacInstallEventHandler() ;
735 }
736 }
ed8c2780
RR
737 MacPostControlCreate() ;
738
739 wxString value ;
1b2b1638 740
29e4a190
RR
741 if( wxApp::s_macDefaultEncodingIsPC )
742 value = wxMacMakeMacStringFromPC( st ) ;
743 else
744 value = st ;
745
746 if ( !m_macUsesTXN )
747 {
748 ::SetControlData( (ControlHandle) m_macControl, 0, ( m_windowStyle & wxTE_PASSWORD ) ? kControlEditTextPasswordTag : kControlEditTextTextTag , value.Length() , (char*) ((const char*)value) ) ;
1b2b1638 749 }
ed8c2780 750 else
ed8c2780 751 {
1b2b1638 752 STPTextPaneVars **tpvars;
29e4a190 753 /* set up locals */
1b2b1638 754 tpvars = (STPTextPaneVars **) GetControlReference((ControlHandle) m_macControl);
29e4a190 755 /* set the text in the record */
1b2b1638
SC
756 TXNSetData( (**tpvars).fTXNRec, kTXNTextData, (void*)value.c_str(), value.Length(),
757 kTXNStartOffset, kTXNEndOffset);
758 m_macTXN = (**tpvars).fTXNRec ;
759 m_macTXNvars = tpvars ;
760 m_macUsesTXN = true ;
2b5f62a0 761 TXNSetSelection( (TXNObject) m_macTXN, 0, 0);
e600c175 762 TXNShowSelection( (TXNObject) m_macTXN, kTXNShowStart);
1b2b1638 763 }
72055702
SC
764
765 return TRUE;
766}
767
768wxString wxTextCtrl::GetValue() const
769{
ed8c2780 770 Size actualsize;
29e4a190
RR
771
772 if ( !m_macUsesTXN )
773 {
774 ::GetControlData( (ControlHandle) m_macControl, 0,
775 ( m_windowStyle & wxTE_PASSWORD ) ? kControlEditTextPasswordTag : kControlEditTextTextTag,
776 32767 , wxBuffer , &actualsize ) ;
777 }
778 else
779 {
780 Handle theText ;
781 OSStatus err = TXNGetDataEncoded( ((TXNObject) m_macTXN), kTXNStartOffset, kTXNEndOffset, &theText , kTXNTextData );
782 // all done
783 if ( err )
784 {
785 actualsize = 0 ;
786 }
787 else
788 {
789 actualsize = GetHandleSize( theText ) ;
790 if (actualsize != 0)
791 strncpy( wxBuffer , *theText , actualsize ) ;
792 DisposeHandle( theText ) ;
793 }
794 }
795
ed8c2780 796 wxBuffer[actualsize] = 0 ;
29e4a190
RR
797
798 wxString value;
799
ed8c2780 800 if( wxApp::s_macDefaultEncodingIsPC )
2b5f62a0 801 {
29e4a190 802 value = wxMacMakePCStringFromMac( wxBuffer ) ;
2b5f62a0
VZ
803 value.Replace( "\r", "\n" );
804 }
ed8c2780 805 else
29e4a190
RR
806 value = wxBuffer;
807
29e4a190
RR
808
809 return value;
72055702
SC
810}
811
812void wxTextCtrl::GetSelection(long* from, long* to) const
813{
1b2b1638 814 if ( !m_macUsesTXN )
72055702 815 {
1b2b1638
SC
816 *from = (**((TEHandle) m_macTE)).selStart;
817 *to = (**((TEHandle) m_macTE)).selEnd;
72055702
SC
818 }
819 else
820 {
1b2b1638 821 TXNGetSelection( ((TXNObject) m_macTXN) , (TXNOffset*) from , (TXNOffset*) to ) ;
72055702
SC
822 }
823}
824
825void wxTextCtrl::SetValue(const wxString& st)
826{
29e4a190 827 wxString value;
ed8c2780
RR
828
829 if( wxApp::s_macDefaultEncodingIsPC )
2b5f62a0 830 {
ed8c2780 831 value = wxMacMakeMacStringFromPC( st ) ;
2b5f62a0
VZ
832 value.Replace( "\n", "\r" );
833 }
ed8c2780 834 else
29e4a190
RR
835 value = st;
836
29e4a190
RR
837
838 if ( !m_macUsesTXN )
839 {
840 ::SetControlData((ControlHandle) m_macControl, 0, ( m_windowStyle & wxTE_PASSWORD ) ? kControlEditTextPasswordTag : kControlEditTextTextTag , value.Length() , (char*) ((const char*)value) ) ;
841 }
842 else
843 {
e600c175 844 bool formerEditable = IsEditable() ;
2b5f62a0
VZ
845 if ( !formerEditable )
846 SetEditable(true) ;
847 TXNSetData( ((TXNObject) m_macTXN), kTXNTextData, (void*)value.c_str(), value.Length(),
848 kTXNStartOffset, kTXNEndOffset);
849 TXNSetSelection( (TXNObject) m_macTXN, 0, 0);
e600c175 850 TXNShowSelection( (TXNObject) m_macTXN, kTXNShowStart);
2b5f62a0
VZ
851 if ( !formerEditable )
852 SetEditable(formerEditable) ;
29e4a190 853 }
1b2b1638 854 MacRedrawControl() ;
72055702
SC
855}
856
29b30405
SC
857void wxTextCtrl::SetMaxLength(unsigned long len)
858{
859 m_maxLength = len ;
860}
861
862bool wxTextCtrl::SetStyle(long start, long end, const wxTextAttr& style)
863{
864 if ( m_macUsesTXN )
865 {
e600c175 866 bool formerEditable = IsEditable() ;
2b5f62a0
VZ
867 if ( !formerEditable )
868 SetEditable(true) ;
869 TXNTypeAttributes typeAttr[4] ;
870 Str255 fontName = "\pMonaco" ;
871 SInt16 fontSize = 12 ;
872 Style fontStyle = normal ;
873 RGBColor color ;
874 int attrCounter = 0 ;
875 if ( style.HasFont() )
876 {
877 const wxFont &font = style.GetFont() ;
878 CopyCStringToPascal( font.GetFaceName().c_str() , fontName ) ;
879 fontSize = font.GetPointSize() ;
880 if ( font.GetUnderlined() )
881 fontStyle |= underline ;
882 if ( font.GetWeight() == wxBOLD )
883 fontStyle |= bold ;
884 if ( font.GetStyle() == wxITALIC )
885 fontStyle |= italic ;
886
887 typeAttr[attrCounter].tag = kTXNQDFontNameAttribute ;
888 typeAttr[attrCounter].size = kTXNQDFontNameAttributeSize ;
889 typeAttr[attrCounter].data.dataPtr = (void*) fontName ;
890 typeAttr[attrCounter+1].tag = kTXNQDFontSizeAttribute ;
891 typeAttr[attrCounter+1].size = kTXNFontSizeAttributeSize ;
892 typeAttr[attrCounter+1].data.dataValue = (fontSize << 16) ;
893 typeAttr[attrCounter+2].tag = kTXNQDFontStyleAttribute ;
894 typeAttr[attrCounter+2].size = kTXNQDFontStyleAttributeSize ;
895 typeAttr[attrCounter+2].data.dataValue = fontStyle ;
896 attrCounter += 3 ;
897
898 }
899 if ( style.HasTextColour() )
900 {
901 typeAttr[attrCounter].tag = kTXNQDFontColorAttribute ;
902 typeAttr[attrCounter].size = kTXNQDFontColorAttributeSize ;
903 typeAttr[attrCounter].data.dataPtr = (void*) &color ;
904 color = MAC_WXCOLORREF(style.GetTextColour().GetPixel()) ;
905 attrCounter += 1 ;
906 }
29b30405
SC
907
908 if ( attrCounter > 0 )
909 {
910 OSStatus status = TXNSetTypeAttributes ((TXNObject)m_macTXN, attrCounter , typeAttr,
911 start,end);
49b45dcd 912 wxASSERT_MSG( status == noErr , "Couldn't set text attributes" ) ;
29b30405 913 }
2b5f62a0
VZ
914 if ( !formerEditable )
915 SetEditable(formerEditable) ;
29b30405
SC
916 }
917 return TRUE ;
918}
919
920bool wxTextCtrl::SetDefaultStyle(const wxTextAttr& style)
921{
922 wxTextCtrlBase::SetDefaultStyle( style ) ;
923 SetStyle( kTXNUseCurrentSelection , kTXNUseCurrentSelection , GetDefaultStyle() ) ;
924 return TRUE ;
925}
926
72055702
SC
927// Clipboard operations
928void wxTextCtrl::Copy()
929{
930 if (CanCopy())
931 {
1b2b1638 932 if ( !m_macUsesTXN )
72055702 933 {
29e4a190
RR
934 TECopy( ((TEHandle) m_macTE) ) ;
935 ClearCurrentScrap();
936 TEToScrap() ;
937 MacRedrawControl() ;
938 }
939 else
940 {
32b5be3d 941 ClearCurrentScrap();
1b2b1638
SC
942 TXNCopy((TXNObject)m_macTXN);
943 TXNConvertToPublicScrap();
29e4a190 944 }
32b5be3d 945 }
72055702
SC
946}
947
948void wxTextCtrl::Cut()
949{
950 if (CanCut())
951 {
1b2b1638 952 if ( !m_macUsesTXN )
32b5be3d 953 {
29e4a190
RR
954 TECut( ((TEHandle) m_macTE) ) ;
955 ClearCurrentScrap();
956 TEToScrap() ;
957 MacRedrawControl() ;
32b5be3d 958 }
29e4a190
RR
959 else
960 {
1b2b1638
SC
961 ClearCurrentScrap();
962 TXNCut((TXNObject)m_macTXN);
963 TXNConvertToPublicScrap();
29e4a190 964 }
1b2b1638
SC
965 wxCommandEvent event(wxEVT_COMMAND_TEXT_UPDATED, m_windowId);
966 event.SetString( GetValue() ) ;
967 event.SetEventObject( this );
968 GetEventHandler()->ProcessEvent(event);
29e4a190 969 }
72055702
SC
970}
971
972void wxTextCtrl::Paste()
973{
974 if (CanPaste())
975 {
1b2b1638
SC
976 if ( !m_macUsesTXN )
977 {
978 TEFromScrap() ;
979 TEPaste( (TEHandle) m_macTE ) ;
980 MacRedrawControl() ;
981 }
29e4a190
RR
982 else
983 {
1b2b1638
SC
984 TXNConvertFromPublicScrap();
985 TXNPaste((TXNObject)m_macTXN);
e600c175 986 SetStyle( kTXNUseCurrentSelection , kTXNUseCurrentSelection , GetDefaultStyle() ) ;
29e4a190 987 }
1b2b1638
SC
988 wxCommandEvent event(wxEVT_COMMAND_TEXT_UPDATED, m_windowId);
989 event.SetString( GetValue() ) ;
990 event.SetEventObject( this );
991 GetEventHandler()->ProcessEvent(event);
32b5be3d 992 }
72055702
SC
993}
994
995bool wxTextCtrl::CanCopy() const
996{
997 // Can copy if there's a selection
998 long from, to;
999 GetSelection(& from, & to);
1000 return (from != to);
1001}
1002
1003bool wxTextCtrl::CanCut() const
1004{
1b2b1638
SC
1005 if ( !IsEditable() )
1006 {
1007 return false ;
1008 }
72055702
SC
1009 // Can cut if there's a selection
1010 long from, to;
1011 GetSelection(& from, & to);
1012 return (from != to);
1013}
1014
1015bool wxTextCtrl::CanPaste() const
1016{
1017 if (!IsEditable())
1018 return FALSE;
1019
72055702 1020#if TARGET_CARBON
ed8c2780
RR
1021 OSStatus err = noErr;
1022 ScrapRef scrapRef;
1023
1024 err = GetCurrentScrap( &scrapRef );
1025 if ( err != noTypeErr && err != memFullErr )
1026 {
29e4a190
RR
1027 ScrapFlavorFlags flavorFlags;
1028 Size byteCount;
ed8c2780
RR
1029
1030 if (( err = GetScrapFlavorFlags( scrapRef, 'TEXT', &flavorFlags )) == noErr)
1031 {
1032 if (( err = GetScrapFlavorSize( scrapRef, 'TEXT', &byteCount )) == noErr)
1033 {
1034 return TRUE ;
1035 }
1036 }
1037 }
1038 return FALSE;
1039
72055702 1040#else
49b45dcd 1041 long offset ;
ed8c2780
RR
1042 if ( GetScrap( NULL , 'TEXT' , &offset ) > 0 )
1043 {
1044 return TRUE ;
1045 }
72055702 1046#endif
ed8c2780 1047 return FALSE ;
72055702
SC
1048}
1049
1050void wxTextCtrl::SetEditable(bool editable)
1051{
1b2b1638
SC
1052 if ( editable != m_editable )
1053 {
1054 m_editable = editable ;
e600c175
SC
1055 if ( !m_macUsesTXN )
1056 {
1057 if ( editable )
2b5f62a0 1058 UMAActivateControl( (ControlHandle) m_macControl ) ;
e600c175 1059 else
2b5f62a0 1060 UMADeactivateControl((ControlHandle) m_macControl ) ;
e600c175 1061 }
1b2b1638 1062 else
e600c175
SC
1063 {
1064 TXNControlTag tag[] = { kTXNIOPrivilegesTag } ;
49b45dcd 1065 TXNControlData data[] = { { editable ? kTXNReadWrite : kTXNReadOnly } } ;
e600c175
SC
1066 TXNSetTXNObjectControls( (TXNObject) m_macTXN , false , sizeof(tag) / sizeof (TXNControlTag) , tag , data ) ;
1067 }
1b2b1638 1068 }
72055702
SC
1069}
1070
1071void wxTextCtrl::SetInsertionPoint(long pos)
1072{
ed8c2780 1073 SetSelection( pos , pos ) ;
72055702
SC
1074}
1075
1076void wxTextCtrl::SetInsertionPointEnd()
1077{
1078 long pos = GetLastPosition();
1079 SetInsertionPoint(pos);
1080}
1081
1082long wxTextCtrl::GetInsertionPoint() const
1083{
1084 long begin,end ;
1085 GetSelection( &begin , &end ) ;
1086 return begin ;
1087}
1088
1089long wxTextCtrl::GetLastPosition() const
1090{
1b2b1638 1091 if ( !m_macUsesTXN )
72055702 1092 {
1b2b1638 1093 return (**((TEHandle) m_macTE)).teLength ;
72055702 1094 }
9c641c05
SC
1095 else
1096 {
32b5be3d 1097 Handle theText ;
1b2b1638
SC
1098 long actualsize ;
1099 OSErr err = TXNGetDataEncoded( (TXNObject) m_macTXN, kTXNStartOffset, kTXNEndOffset, &theText , kTXNTextData );
32b5be3d
RR
1100 /* all done */
1101 if ( err )
1102 {
1103 actualsize = 0 ;
1104 }
1105 else
1106 {
1107 actualsize = GetHandleSize( theText ) ;
1108 DisposeHandle( theText ) ;
1109 }
1110 return actualsize ;
9c641c05 1111 }
72055702
SC
1112}
1113
1114void wxTextCtrl::Replace(long from, long to, const wxString& value)
1115{
1b2b1638 1116 if ( !m_macUsesTXN )
72055702 1117 {
29e4a190 1118 ControlEditTextSelectionRec selection ;
72055702 1119
29e4a190
RR
1120 selection.selStart = from ;
1121 selection.selEnd = to ;
1122 ::SetControlData((ControlHandle) m_macControl , 0, kControlEditTextSelectionTag , sizeof( selection ) , (char*) &selection ) ;
1123 TESetSelect( from , to , ((TEHandle) m_macTE) ) ;
1124 TEDelete( ((TEHandle) m_macTE) ) ;
1125 TEInsert( value , value.Length() , ((TEHandle) m_macTE) ) ;
1126 }
1127 else
1128 {
e600c175 1129 bool formerEditable = IsEditable() ;
2b5f62a0
VZ
1130 if ( !formerEditable )
1131 SetEditable(true) ;
1132 TXNSetSelection( ((TXNObject) m_macTXN) , from , to ) ;
1133 TXNClear( ((TXNObject) m_macTXN) ) ;
1134 TXNSetData( ((TXNObject) m_macTXN), kTXNTextData, (void*)value.c_str(), value.Length(),
1135 kTXNUseCurrentSelection, kTXNUseCurrentSelection);
1136 if ( !formerEditable )
1137 SetEditable( formerEditable ) ;
29e4a190 1138 }
32b5be3d 1139 Refresh() ;
72055702
SC
1140}
1141
1142void wxTextCtrl::Remove(long from, long to)
1143{
1b2b1638 1144 if ( !m_macUsesTXN )
72055702 1145 {
29e4a190 1146 ControlEditTextSelectionRec selection ;
72055702 1147
29e4a190
RR
1148 selection.selStart = from ;
1149 selection.selEnd = to ;
1150 ::SetControlData( (ControlHandle) m_macControl , 0, kControlEditTextSelectionTag , sizeof( selection ) , (char*) &selection ) ;
1151 TEDelete( ((TEHandle) m_macTE) ) ;
9c641c05
SC
1152 }
1153 else
1154 {
e600c175 1155 bool formerEditable = IsEditable() ;
2b5f62a0
VZ
1156 if ( !formerEditable )
1157 SetEditable(true) ;
1b2b1638
SC
1158 TXNSetSelection( ((TXNObject) m_macTXN) , from , to ) ;
1159 TXNClear( ((TXNObject) m_macTXN) ) ;
2b5f62a0
VZ
1160 if ( !formerEditable )
1161 SetEditable( formerEditable ) ;
72055702 1162 }
ed8c2780 1163 Refresh() ;
72055702
SC
1164}
1165
1166void wxTextCtrl::SetSelection(long from, long to)
1167{
1b2b1638
SC
1168
1169 if ( !m_macUsesTXN )
72055702
SC
1170 {
1171 ControlEditTextSelectionRec selection ;
72055702
SC
1172 selection.selStart = from ;
1173 selection.selEnd = to ;
1174
1b2b1638
SC
1175 TESetSelect( selection.selStart , selection.selEnd , ((TEHandle) m_macTE) ) ;
1176 ::SetControlData((ControlHandle) m_macControl , 0, kControlEditTextSelectionTag , sizeof( selection ) , (char*) &selection ) ;
72055702
SC
1177 }
1178 else
1179 {
29e4a190
RR
1180 STPTextPaneVars **tpvars;
1181 /* set up our locals */
1182 tpvars = (STPTextPaneVars **) GetControlReference((ControlHandle) m_macControl);
1183 /* and our drawing environment as the operation
1184 may force a redraw in the text area. */
1185 SetPort((**tpvars).fDrawingEnvironment);
1186 /* change the selection */
1187 TXNSetSelection( (**tpvars).fTXNRec, from, to);
e600c175 1188 TXNShowSelection( (TXNObject) m_macTXN, kTXNShowStart);
72055702
SC
1189 }
1190}
1191
1192bool wxTextCtrl::LoadFile(const wxString& file)
1193{
1194 if ( wxTextCtrlBase::LoadFile(file) )
1195 {
1196 return TRUE;
1197 }
1198
1199 return FALSE;
1200}
1201
1202void wxTextCtrl::WriteText(const wxString& text)
1203{
1204 wxString value ;
29e4a190 1205 if( wxApp::s_macDefaultEncodingIsPC )
2b5f62a0 1206 {
29e4a190 1207 value = wxMacMakeMacStringFromPC( text ) ;
2b5f62a0
VZ
1208 value.Replace( "\n", "\r" );
1209 }
29e4a190
RR
1210 else
1211 value = text ;
2b5f62a0 1212
1b2b1638 1213 if ( !m_macUsesTXN )
72055702 1214 {
29e4a190
RR
1215 TEInsert( value , value.Length() , ((TEHandle) m_macTE) ) ;
1216 }
1217 else
1218 {
e600c175 1219 bool formerEditable = IsEditable() ;
2b5f62a0
VZ
1220 if ( !formerEditable )
1221 SetEditable(true) ;
1222 long start , end , dummy ;
1223 GetSelection( &start , &dummy ) ;
29e4a190
RR
1224 TXNSetData( ((TXNObject) m_macTXN), kTXNTextData, (void*) (const char*)value, value.Length(),
1225 kTXNUseCurrentSelection, kTXNUseCurrentSelection);
2b5f62a0 1226 GetSelection( &dummy , &end ) ;
29b30405 1227 SetStyle( start , end , GetDefaultStyle() ) ;
2b5f62a0
VZ
1228 if ( !formerEditable )
1229 SetEditable( formerEditable ) ;
29e4a190
RR
1230 }
1231 MacRedrawControl() ;
72055702
SC
1232}
1233
1234void wxTextCtrl::AppendText(const wxString& text)
1235{
1236 SetInsertionPointEnd();
1237 WriteText(text);
1238}
1239
1240void wxTextCtrl::Clear()
1241{
1b2b1638
SC
1242 if ( !IsEditable() )
1243 {
1244 return ;
1245 }
1246 if ( !m_macUsesTXN )
72055702 1247 {
29e4a190 1248 ::SetControlData((ControlHandle) m_macControl, 0, ( m_windowStyle & wxTE_PASSWORD ) ? kControlEditTextPasswordTag : kControlEditTextTextTag , 0 , (char*) ((const char*)NULL) ) ;
72055702 1249 }
9c641c05
SC
1250 else
1251 {
9c34dd9d
SC
1252 TXNSetSelection( (TXNObject)m_macTXN , kTXNStartOffset , kTXNEndOffset ) ;
1253 TXNClear((TXNObject)m_macTXN);
9c641c05 1254 }
32b5be3d 1255 Refresh() ;
72055702
SC
1256}
1257
1258bool wxTextCtrl::IsModified() const
1259{
1260 return TRUE;
1261}
1262
1263bool wxTextCtrl::IsEditable() const
1264{
1b2b1638 1265 return IsEnabled() && m_editable ;
72055702
SC
1266}
1267
1268bool wxTextCtrl::AcceptsFocus() const
1269{
1270 // we don't want focus if we can't be edited
1b2b1638 1271 return /*IsEditable() && */ wxControl::AcceptsFocus();
72055702
SC
1272}
1273
1274wxSize wxTextCtrl::DoGetBestSize() const
1275{
1276 int wText = 100 ;
ed8c2780 1277
402679b0 1278 int hText;
29e4a190
RR
1279 if ( m_macUsesTXN )
1280 {
1281 hText = 17 ;
1282 }
1283 else
1284 {
1285 hText = 13 ;
1286 }
72055702
SC
1287/*
1288 int cx, cy;
1289 wxGetCharSize(GetHWND(), &cx, &cy, &GetFont());
1290
1291 int wText = DEFAULT_ITEM_WIDTH;
1292
1293 int hText = EDIT_HEIGHT_FROM_CHAR_HEIGHT(cy);
1294
1295 return wxSize(wText, hText);
1296*/
1297 if ( m_windowStyle & wxTE_MULTILINE )
1298 {
402679b0 1299 hText *= 5 ;
72055702 1300 }
402679b0
SC
1301 hText += 2 * m_macVerticalBorder ;
1302 wText += 2 * m_macHorizontalBorder ;
72055702
SC
1303 //else: for single line control everything is ok
1304 return wxSize(wText, hText);
1305}
1306
1307// ----------------------------------------------------------------------------
1308// Undo/redo
1309// ----------------------------------------------------------------------------
1310
1311void wxTextCtrl::Undo()
1312{
1313 if (CanUndo())
1314 {
1315 }
1316}
1317
1318void wxTextCtrl::Redo()
1319{
1320 if (CanRedo())
1321 {
1322 }
1323}
1324
1325bool wxTextCtrl::CanUndo() const
1326{
1327 return FALSE ;
1328}
1329
1330bool wxTextCtrl::CanRedo() const
1331{
1332 return FALSE ;
1333}
1334
1335// Makes 'unmodified'
1336void wxTextCtrl::DiscardEdits()
1337{
1338 // TODO
1339}
1340
1341int wxTextCtrl::GetNumberOfLines() const
1342{
9c641c05
SC
1343 // TODO change this if possible to reflect real lines
1344 wxString content = GetValue() ;
32b5be3d
RR
1345
1346 int count = 1;
49b45dcd 1347 for (size_t i = 0; i < content.Length() ; i++)
29e4a190
RR
1348 {
1349 if (content[i] == '\r') count++;
32b5be3d
RR
1350 }
1351
1b2b1638 1352 return count;
72055702
SC
1353}
1354
1355long wxTextCtrl::XYToPosition(long x, long y) const
1356{
1357 // TODO
1358 return 0;
1359}
1360
1361bool wxTextCtrl::PositionToXY(long pos, long *x, long *y) const
1362{
1363 return FALSE ;
1364}
1365
1366void wxTextCtrl::ShowPosition(long pos)
1367{
1368 // TODO
1369}
1370
1371int wxTextCtrl::GetLineLength(long lineNo) const
1372{
9c641c05
SC
1373 // TODO change this if possible to reflect real lines
1374 wxString content = GetValue() ;
29e4a190
RR
1375
1376 // Find line first
1377 int count = 0;
49b45dcd 1378 for (size_t i = 0; i < content.Length() ; i++)
32b5be3d
RR
1379 {
1380 if (count == lineNo)
1381 {
1382 // Count chars in line then
1383 count = 0;
49b45dcd 1384 for (size_t j = i; j < content.Length(); j++)
32b5be3d
RR
1385 {
1386 count++;
29e4a190
RR
1387 if (content[j] == '\r') return count;
1388 }
1389
1390 return count;
1391 }
1392 if (content[i] == '\r') count++;
1393 }
72055702
SC
1394 return 0;
1395}
1396
1397wxString wxTextCtrl::GetLineText(long lineNo) const
1398{
9c641c05
SC
1399 // TODO change this if possible to reflect real lines
1400 wxString content = GetValue() ;
1401
29e4a190
RR
1402 // Find line first
1403 int count = 0;
49b45dcd 1404 for (size_t i = 0; i < content.Length() ; i++)
32b5be3d
RR
1405 {
1406 if (count == lineNo)
1407 {
1408 // Add chars in line then
1409 wxString tmp("");
1410
49b45dcd 1411 for (size_t j = i; j < content.Length(); j++)
29e4a190
RR
1412 {
1413 if (content[j] == '\r')
1414 return tmp;
1415
1416 tmp += content[j];
1417 }
1418
1419 return tmp;
1420 }
1421 if (content[i] == '\r') count++;
1422 }
1423 return "" ;
72055702
SC
1424}
1425
1426/*
1427 * Text item
1428 */
1429
1430void wxTextCtrl::Command(wxCommandEvent & event)
1431{
1432 SetValue (event.GetString());
1433 ProcessCommand (event);
1434}
1435
1436void wxTextCtrl::OnDropFiles(wxDropFilesEvent& event)
1437{
1438 // By default, load the first file into the text window.
1439 if (event.GetNumberOfFiles() > 0)
1440 {
1441 LoadFile(event.GetFiles()[0]);
1442 }
1443}
1444
1445void wxTextCtrl::OnChar(wxKeyEvent& event)
1446{
1b2b1638
SC
1447 int key = event.GetKeyCode() ;
1448 bool eat_key = false ;
1449
e600c175
SC
1450 if ( key == 'c' && event.MetaDown() )
1451 {
1452 if ( CanCopy() )
1453 Copy() ;
1454 return ;
1455 }
1456
1b2b1638
SC
1457 if ( !IsEditable() && key != WXK_LEFT && key != WXK_RIGHT && key != WXK_DOWN && key != WXK_UP && key != WXK_TAB &&
1458 !( key == WXK_RETURN && ( (m_windowStyle & wxPROCESS_ENTER) || (m_windowStyle & wxTE_MULTILINE) ) )
1459/* && key != WXK_PRIOR && key != WXK_NEXT && key != WXK_HOME && key != WXK_END */
1460 )
1461 {
1462 // eat it
1463 return ;
1464 }
e600c175
SC
1465 if ( key == 'v' && event.MetaDown() )
1466 {
1467 if ( CanPaste() )
1468 Paste() ;
1469 return ;
1470 }
1471 if ( key == 'x' && event.MetaDown() )
1472 {
1473 if ( CanCut() )
1474 Cut() ;
1475 return ;
1476 }
1b2b1638 1477 switch ( key )
72055702
SC
1478 {
1479 case WXK_RETURN:
ed8c2780
RR
1480 if (m_windowStyle & wxPROCESS_ENTER)
1481 {
72055702
SC
1482 wxCommandEvent event(wxEVT_COMMAND_TEXT_ENTER, m_windowId);
1483 event.SetEventObject( this );
1b2b1638 1484 event.SetString( GetValue() );
72055702
SC
1485 if ( GetEventHandler()->ProcessEvent(event) )
1486 return;
ed8c2780 1487 }
72055702
SC
1488 if ( !(m_windowStyle & wxTE_MULTILINE) )
1489 {
ed8c2780 1490 wxWindow *parent = GetParent();
9c641c05
SC
1491 while( parent && !parent->IsTopLevel() && parent->GetDefaultItem() == NULL ) {
1492 parent = parent->GetParent() ;
ed8c2780 1493 }
9c641c05 1494 if ( parent && parent->GetDefaultItem() )
ed8c2780 1495 {
9c641c05 1496 wxButton *def = wxDynamicCast(parent->GetDefaultItem(),
72055702 1497 wxButton);
ed8c2780
RR
1498 if ( def && def->IsEnabled() )
1499 {
1500 wxCommandEvent event(wxEVT_COMMAND_BUTTON_CLICKED, def->GetId() );
1501 event.SetEventObject(def);
1502 def->Command(event);
1503 return ;
72055702 1504 }
ed8c2780 1505 }
1b2b1638
SC
1506
1507 // this will make wxWindows eat the ENTER key so that
1508 // we actually prevent line wrapping in a single line
1509 // text control
1510 eat_key = TRUE;
72055702 1511 }
72055702
SC
1512
1513 break;
1514
1515 case WXK_TAB:
1516 // always produce navigation event - even if we process TAB
1517 // ourselves the fact that we got here means that the user code
1518 // decided to skip processing of this TAB - probably to let it
1519 // do its default job.
1520 {
1521 wxNavigationKeyEvent eventNav;
1522 eventNav.SetDirection(!event.ShiftDown());
1523 eventNav.SetWindowChange(event.ControlDown());
1524 eventNav.SetEventObject(this);
1525
1526 if ( GetParent()->GetEventHandler()->ProcessEvent(eventNav) )
1527 return;
1b2b1638 1528
ed8c2780 1529 event.Skip() ;
1b2b1638 1530 return;
72055702
SC
1531 }
1532 break;
1533 }
1b2b1638 1534
1b2b1638 1535 if (!eat_key)
ed8c2780 1536 {
45c8d9e5
SC
1537 // default handling
1538 event.Skip() ;
1b2b1638 1539 }
e600c175 1540 if ( ( key >= 0x20 && key < WXK_START ) ||
45c8d9e5
SC
1541 key == WXK_RETURN ||
1542 key == WXK_DELETE ||
1543 key == WXK_BACK)
1b2b1638
SC
1544 {
1545 wxCommandEvent event1(wxEVT_COMMAND_TEXT_UPDATED, m_windowId);
1546 event1.SetString( GetValue() ) ;
1547 event1.SetEventObject( this );
45c8d9e5 1548 wxPostEvent(GetEventHandler(),event1);
ed8c2780 1549 }
72055702
SC
1550}
1551
fd4393b8
SC
1552void wxTextCtrl::MacSuperShown( bool show )
1553{
1554 bool former = m_macControlIsShown ;
1555 wxControl::MacSuperShown( show ) ;
1556 if ( (former != m_macControlIsShown) && m_macUsesTXN )
1557 {
1558 if ( m_macControlIsShown )
29e4a190
RR
1559 TXNSetFrameBounds( (TXNObject) m_macTXN, (**(STPTextPaneVars **)m_macTXNvars).fRTextArea.top, (**(STPTextPaneVars **)m_macTXNvars).fRTextArea.left,
1560 (**(STPTextPaneVars **)m_macTXNvars).fRTextArea.bottom,(**(STPTextPaneVars **)m_macTXNvars).fRTextArea.right, (**(STPTextPaneVars **)m_macTXNvars).fTXNFrame);
fd4393b8 1561 else
29e4a190
RR
1562 TXNSetFrameBounds( (TXNObject) m_macTXN, (**(STPTextPaneVars **)m_macTXNvars).fRTextArea.top + 30000, (**(STPTextPaneVars **)m_macTXNvars).fRTextArea.left,
1563 (**(STPTextPaneVars **)m_macTXNvars).fRTextArea.bottom + 30000, (**(STPTextPaneVars **)m_macTXNvars).fRTextArea.right, (**(STPTextPaneVars **)m_macTXNvars).fTXNFrame);
fd4393b8
SC
1564 }
1565}
1566
1567bool wxTextCtrl::Show(bool show)
1568{
1569 bool former = m_macControlIsShown ;
1570
1571 bool retval = wxControl::Show( show ) ;
1572
2b5f62a0 1573 if ( former != m_macControlIsShown && show )
fd4393b8
SC
1574 {
1575 if ( m_macControlIsShown )
29e4a190
RR
1576 TXNSetFrameBounds( (TXNObject) m_macTXN, (**(STPTextPaneVars **)m_macTXNvars).fRTextArea.top, (**(STPTextPaneVars **)m_macTXNvars).fRTextArea.left,
1577 (**(STPTextPaneVars **)m_macTXNvars).fRTextArea.bottom,(**(STPTextPaneVars **)m_macTXNvars).fRTextArea.right, (**(STPTextPaneVars **)m_macTXNvars).fTXNFrame);
fd4393b8 1578 else
29e4a190
RR
1579 TXNSetFrameBounds( (TXNObject) m_macTXN, (**(STPTextPaneVars **)m_macTXNvars).fRTextArea.top + 30000, (**(STPTextPaneVars **)m_macTXNvars).fRTextArea.left,
1580 (**(STPTextPaneVars **)m_macTXNvars).fRTextArea.bottom + 30000, (**(STPTextPaneVars **)m_macTXNvars).fRTextArea.right, (**(STPTextPaneVars **)m_macTXNvars).fTXNFrame);
fd4393b8
SC
1581 }
1582
1583 return retval ;
1584}
1585
72055702
SC
1586// ----------------------------------------------------------------------------
1587// standard handlers for standard edit menu events
1588// ----------------------------------------------------------------------------
1589
eb22f2a6 1590void wxTextCtrl::OnCut(wxCommandEvent& WXUNUSED(event))
72055702
SC
1591{
1592 Cut();
1593}
1594
eb22f2a6 1595void wxTextCtrl::OnCopy(wxCommandEvent& WXUNUSED(event))
72055702
SC
1596{
1597 Copy();
1598}
1599
eb22f2a6 1600void wxTextCtrl::OnPaste(wxCommandEvent& WXUNUSED(event))
72055702
SC
1601{
1602 Paste();
1603}
1604
eb22f2a6 1605void wxTextCtrl::OnUndo(wxCommandEvent& WXUNUSED(event))
72055702
SC
1606{
1607 Undo();
1608}
1609
eb22f2a6 1610void wxTextCtrl::OnRedo(wxCommandEvent& WXUNUSED(event))
72055702
SC
1611{
1612 Redo();
1613}
1614
1615void wxTextCtrl::OnUpdateCut(wxUpdateUIEvent& event)
1616{
1617 event.Enable( CanCut() );
1618}
1619
1620void wxTextCtrl::OnUpdateCopy(wxUpdateUIEvent& event)
1621{
1622 event.Enable( CanCopy() );
1623}
1624
1625void wxTextCtrl::OnUpdatePaste(wxUpdateUIEvent& event)
1626{
1627 event.Enable( CanPaste() );
1628}
1629
1630void wxTextCtrl::OnUpdateUndo(wxUpdateUIEvent& event)
1631{
1632 event.Enable( CanUndo() );
1633}
1634
1635void wxTextCtrl::OnUpdateRedo(wxUpdateUIEvent& event)
1636{
1637 event.Enable( CanRedo() );
1638}
1639
2b5f62a0
VZ
1640bool wxTextCtrl::MacSetupCursor( const wxPoint& pt )
1641{
1642 if ( m_macUsesTXN )
1643 return true ;
1644 else
1645 return wxWindow::MacSetupCursor( pt ) ;
1646}
72055702 1647
fedad417
GD
1648#endif
1649 // wxUSE_TEXTCTRL