/**
NSTextView
Categories which add user actions to NSTextView
Copyright (C) 1996, 1998, 2000, 2001, 2002, 2003 Free Software Foundation, Inc.
Originally moved here from NSTextView.m.
Author: Scott Christley
Date: 1996
Author: Felipe A. Rodriguez
Date: July 1998
Author: Daniel Böhringer
Date: August 1998
Author: Fred Kiefer
Date: March 2000, September 2000
Author: Nicola Pero
Date: 2000, 2001, 2002
Author: Pierre-Yves Rivaille
Date: September 2002
Extensive reworking: Alexander Malmberg
Date: December 2002 - February 2003
This file is part of the GNUstep GUI Library.
This library is free software; you can redistribute it and/or
modify it under the terms of the GNU Lesser General Public
License as published by the Free Software Foundation; either
version 2 of the License, or (at your option) any later version.
This library is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
Lesser General Public License for more details.
You should have received a copy of the GNU Lesser General Public
License along with this library; see the file COPYING.LIB.
If not, see or write to the
Free Software Foundation, 51 Franklin Street, Fifth Floor,
Boston, MA 02110-1301, USA.
*/
#import
#import
#import "AppKit/NSAttributedString.h"
#import "AppKit/NSGraphics.h"
#import "AppKit/NSLayoutManager.h"
#import "AppKit/NSPasteboard.h"
#import "AppKit/NSScrollView.h"
#import "AppKit/NSTextStorage.h"
#import "AppKit/NSTextView.h"
#import "AppKit/NSParagraphStyle.h"
/*
These methods are for user actions, ie. they are normally called from
-doCommandBySelector: (which is called by the input manager) in response
to some key press or other user event.
User actions that modify the text must check that a modification is allowed
and make sure all necessary notifications are sent. This is done by sending
-shouldChangeTextInRange:replacementString: before making any changes, and
(if the change is allowed) -didChangeText after the changes have been made.
All actions from NSResponder that make sense for a text view should be
implemented here, but this is _not_ the place to add new actions.
When changing attributes, the range returned by
rangeForUserCharacterAttributeChange or rangeForUserParagraphAttributeChange
should be used. If the location is NSNotFound, nothing should be done (in
particular, the typing attributes should _not_ be changed). Otherwise,
-shouldChangeTextInRange:replacementString: should be called, and if it
returns YES, the attributes of the range and the typing attributes should be
changed, and -didChangeText should be called.
In a non-rich-text text view, the typing attributes _must_always_ hold the
attributes of the text. Thus, the typing attributes must always be changed
in the same way that the attributes of the text are changed.
TODO: can the selected range's location be NSNotFound? when?
Not all user actions are here. Exceptions:
-copy:
-copyFont:
-copyRuler:
-paste:
-pasteFont:
-pasteRuler:
-pasteAsPlainText:
-pasteAsRichText:
-checkSpelling:
-showGuessPanel:
-selectAll: (implemented in NSText)
Not all methods that handle user-induced text modifications are here.
Exceptions:
(TODO)
-insertText:
-changeColor:
-changeFont: (action method?)
drag&drop handling methods
(others?)
All other methods that modify text are for programmatic changes and do not
send -shouldChangeTextInRange:replacementString: or -didChangeText.
*/
/* global kill buffer shared between all text views */
/* Note: I'm not using an attributed string here because Apple apparently is
using a plain string either. Maybe this is because NeXT was using the X11
cut buffer for the kill buffer, which can hold only plain strings? */
static NSString *killBuffer = @"";
/** First some helpers **/
@interface NSTextView (UserActionHelpers)
-(void) _illegalMovement: (int)textMovement;
-(void) _changeAttribute: (NSString *)name
inRange: (NSRange)r
using: (NSNumber*(*)(NSNumber*))func;
@end
@implementation NSTextView (UserActionHelpers)
- (void) _illegalMovement: (int)textMovement
{
/* This is similar to [self resignFirstResponder], with the
difference that in the notification we need to put the
NSTextMovement, which resignFirstResponder does not. Also, if we
are ending editing, we are going to be removed, so it's useless
to update any drawing. Please note that this ends up calling
resignFirstResponder anyway. */
NSNumber *number;
NSDictionary *uiDictionary;
if ((_tf.is_editable)
&& ([_delegate respondsToSelector:
@selector(textShouldEndEditing:)])
&& ([_delegate textShouldEndEditing: self] == NO))
return;
/* TODO: insertion point.
doesn't the -resignFirstResponder take care of that?
*/
number = [NSNumber numberWithInt: textMovement];
uiDictionary = [NSDictionary dictionaryWithObject: number
forKey: @"NSTextMovement"];
[[NSNotificationCenter defaultCenter]
postNotificationName: NSTextDidEndEditingNotification
object: self
userInfo: uiDictionary];
/* The TextField will get the notification, and drop our first responder
* status if it's the case ... in that case, our -resignFirstResponder will
* be called! */
return;
}
- (void) _changeAttribute: (NSString *)name
inRange: (NSRange)r
using: (NSNumber*(*)(NSNumber*))func
{
NSUInteger i;
NSRange e, r2;
id current, new;
if (![self shouldChangeTextInRange: r replacementString: nil])
return;
[_textStorage beginEditing];
for (i = r.location; i < NSMaxRange(r);)
{
current = [_textStorage attribute: name
atIndex: i
effectiveRange: &e];
r2 = NSMakeRange(i, NSMaxRange(e) - i);
r2 = NSIntersectionRange(r2, r);
i = NSMaxRange(e);
new = func(current);
if (new != current)
{
if (!new)
{
[_textStorage removeAttribute: name
range: r2];
}
else
{
[_textStorage addAttribute: name
value: new
range: r2];
}
}
}
[_textStorage endEditing];
current = [_layoutManager->_typingAttributes objectForKey: name];
new = func(current);
if (new != current)
{
if (!new)
{
[_layoutManager->_typingAttributes removeObjectForKey: name];
}
else
{
[_layoutManager->_typingAttributes setObject: new forKey: name];
}
}
[self didChangeText];
}
@end
@implementation NSTextView (UserActions)
/* Helpers used with _changeAttribute:inRange:using:. */
static NSNumber *int_minus_one(NSNumber *cur)
{
int value;
if (cur)
value = [cur intValue] - 1;
else
value = -1;
if (value)
return [NSNumber numberWithInt: value];
else
return nil;
}
static NSNumber *int_plus_one(NSNumber *cur)
{
int value;
if (cur)
value = [cur intValue] + 1;
else
value = 1;
if (value)
return [NSNumber numberWithInt: value];
else
return nil;
}
static NSNumber *float_minus_one(NSNumber *cur)
{
float value;
if (cur)
value = [cur floatValue] - 1;
else
value = -1;
if (value)
return [NSNumber numberWithFloat: value];
else
return nil;
}
static NSNumber *float_plus_one(NSNumber *cur)
{
int value;
if (cur)
value = [cur floatValue] + 1;
else
value = 1;
if (value)
return [NSNumber numberWithFloat: value];
else
return nil;
}
- (void) subscript: (id)sender
{
NSRange r = [self rangeForUserCharacterAttributeChange];
if (r.location == NSNotFound)
return;
[self _changeAttribute: NSSuperscriptAttributeName
inRange: r
using: int_minus_one];
}
- (void) superscript: (id)sender
{
NSRange r = [self rangeForUserCharacterAttributeChange];
if (r.location == NSNotFound)
return;
[self _changeAttribute: NSSuperscriptAttributeName
inRange: r
using: int_plus_one];
}
- (void) lowerBaseline: (id)sender
{
NSRange r = [self rangeForUserCharacterAttributeChange];
if (r.location == NSNotFound)
return;
[self _changeAttribute: NSBaselineOffsetAttributeName
inRange: r
using: float_plus_one];
}
- (void) raiseBaseline: (id)sender
{
NSRange r = [self rangeForUserCharacterAttributeChange];
if (r.location == NSNotFound)
return;
[self _changeAttribute: NSBaselineOffsetAttributeName
inRange: r
using: float_minus_one];
}
- (void) unscript: (id)sender
{
NSRange aRange = [self rangeForUserCharacterAttributeChange];
if (aRange.location == NSNotFound)
return;
if (![self shouldChangeTextInRange: aRange
replacementString: nil])
return;
if (aRange.length)
{
[_textStorage beginEditing];
[_textStorage removeAttribute: NSSuperscriptAttributeName
range: aRange];
[_textStorage removeAttribute: NSBaselineOffsetAttributeName
range: aRange];
[_textStorage endEditing];
}
[_layoutManager->_typingAttributes removeObjectForKey: NSSuperscriptAttributeName];
[_layoutManager->_typingAttributes removeObjectForKey: NSBaselineOffsetAttributeName];
[self didChangeText];
}
- (void) underline: (id)sender
{
BOOL doUnderline = YES;
NSRange aRange = [self rangeForUserCharacterAttributeChange];
if (aRange.location == NSNotFound)
return;
if ([[_textStorage attribute: NSUnderlineStyleAttributeName
atIndex: aRange.location
effectiveRange: NULL] intValue])
doUnderline = NO;
if (aRange.length)
{
if (![self shouldChangeTextInRange: aRange
replacementString: nil])
return;
[_textStorage beginEditing];
[_textStorage addAttribute: NSUnderlineStyleAttributeName
value: [NSNumber numberWithInt: doUnderline]
range: aRange];
[_textStorage endEditing];
[self didChangeText];
}
[_layoutManager->_typingAttributes
setObject: [NSNumber numberWithInt: doUnderline]
forKey: NSUnderlineStyleAttributeName];
}
- (void) useStandardKerning: (id)sender
{
NSRange aRange = [self rangeForUserCharacterAttributeChange];
if (aRange.location == NSNotFound)
return;
if (![self shouldChangeTextInRange: aRange
replacementString: nil])
return;
[_textStorage removeAttribute: NSKernAttributeName
range: aRange];
[_layoutManager->_typingAttributes removeObjectForKey: NSKernAttributeName];
[self didChangeText];
}
- (void) turnOffKerning: (id)sender
{
NSRange aRange = [self rangeForUserCharacterAttributeChange];
if (aRange.location == NSNotFound)
return;
if (![self shouldChangeTextInRange: aRange
replacementString: nil])
return;
[_textStorage addAttribute: NSKernAttributeName
value: [NSNumber numberWithFloat: 0.0]
range: aRange];
[_layoutManager->_typingAttributes setObject: [NSNumber numberWithFloat: 0.0]
forKey: NSKernAttributeName];
[self didChangeText];
}
- (void) loosenKerning: (id)sender
{
NSRange r = [self rangeForUserCharacterAttributeChange];
if (r.location == NSNotFound)
return;
[self _changeAttribute: NSKernAttributeName
inRange: r
using: float_plus_one];
}
- (void) tightenKerning: (id)sender
{
NSRange r = [self rangeForUserCharacterAttributeChange];
if (r.location == NSNotFound)
return;
[self _changeAttribute: NSKernAttributeName
inRange: r
using: float_minus_one];
}
- (void) turnOffLigatures: (id)sender
{
NSRange aRange = [self rangeForUserCharacterAttributeChange];
if (aRange.location == NSNotFound)
return;
if (![self shouldChangeTextInRange: aRange
replacementString: nil])
return;
[_textStorage addAttribute: NSLigatureAttributeName
value: [NSNumber numberWithInt: 0]
range: aRange];
[_layoutManager->_typingAttributes setObject: [NSNumber numberWithInt: 0]
forKey: NSLigatureAttributeName];
[self didChangeText];
}
- (void) useStandardLigatures: (id)sender
{
NSRange aRange = [self rangeForUserCharacterAttributeChange];
if (aRange.location == NSNotFound)
return;
if (![self shouldChangeTextInRange: aRange
replacementString: nil])
return;
[_textStorage removeAttribute: NSLigatureAttributeName
range: aRange];
[_layoutManager->_typingAttributes removeObjectForKey: NSLigatureAttributeName];
[self didChangeText];
}
- (void) useAllLigatures: (id)sender
{
NSRange aRange = [self rangeForUserCharacterAttributeChange];
if (aRange.location == NSNotFound)
return;
if (![self shouldChangeTextInRange: aRange
replacementString: nil])
return;
[_textStorage addAttribute: NSLigatureAttributeName
value: [NSNumber numberWithInt: 2]
range: aRange];
[_layoutManager->_typingAttributes setObject: [NSNumber numberWithInt: 2]
forKey: NSLigatureAttributeName];
[self didChangeText];
}
- (void) toggleTraditionalCharacterShape: (id)sender
{
// TODO
NSLog(@"Method %s is not implemented for class %s",
"toggleTraditionalCharacterShape:", "NSTextView");
}
- (void) insertNewline: (id)sender
{
if (_tf.is_field_editor)
{
[self _illegalMovement: NSReturnTextMovement];
return;
}
[self insertText: @"\n"];
}
- (void) insertTab: (id)sender
{
if (_tf.is_field_editor)
{
[self _illegalMovement: NSTabTextMovement];
return;
}
[self insertText: @"\t"];
}
- (void) insertBacktab: (id)sender
{
if (_tf.is_field_editor)
{
[self _illegalMovement: NSBacktabTextMovement];
return;
}
/* TODO */
//[self insertText: @"\t"];
}
- (void) insertNewlineIgnoringFieldEditor: (id)sender
{
[self insertText: @"\n"];
}
- (void) insertTabIgnoringFieldEditor: (id)sender
{
[self insertText: @"\t"];
}
- (void) insertContainerBreak: (id)sender
{
unichar ch = NSFormFeedCharacter;
[self insertText: [NSString stringWithCharacters: &ch length: 1]];
}
- (void) insertLineBreak: (id)sender
{
unichar ch = NSLineSeparatorCharacter;
[self insertText: [NSString stringWithCharacters: &ch length: 1]];
}
- (void) deleteForward: (id)sender
{
NSRange range = [self rangeForUserTextChange];
NSDictionary *attributes;
if (range.location == NSNotFound)
{
return;
}
/* Manage case of insertion point - implicitly means to delete following
character */
if (range.length == 0)
{
if (range.location != [_textStorage length])
{
/* Not at the end of text -- delete following character */
range.length = 1;
}
else
{
/* At the end of text - TODO: Make beeping or not beeping
configurable vie User Defaults */
NSBeep ();
return;
}
}
else if ([self smartInsertDeleteEnabled] &&
[self selectionGranularity] == NSSelectByWord)
{
range = [self smartDeleteRangeForProposedRange: range];
}
if (![self shouldChangeTextInRange: range replacementString: @""])
{
return;
}
attributes = RETAIN([_textStorage attributesAtIndex: range.location
effectiveRange: NULL]);
[_textStorage beginEditing];
[_textStorage deleteCharactersInRange: range];
[_textStorage endEditing];
[self setTypingAttributes: attributes];
RELEASE(attributes);
[self didChangeText];
}
- (void) deleteBackward: (id)sender
{
NSRange range = [self rangeForUserTextChange];
NSDictionary *attributes;
if (range.location == NSNotFound)
{
return;
}
/* Manage case of insertion point - implicitly means to delete
previous character */
if (range.length == 0)
{
if (range.location != 0)
{
/* Not at the beginning of text -- delete previous character */
range.location -= 1;
range.length = 1;
}
else
{
/* At the beginning of text - TODO: Make beeping or not
beeping configurable via User Defaults */
NSBeep ();
return;
}
}
else if ([self smartInsertDeleteEnabled] &&
[self selectionGranularity] == NSSelectByWord)
{
range = [self smartDeleteRangeForProposedRange: range];
}
if (![self shouldChangeTextInRange: range replacementString: @""])
{
return;
}
attributes = RETAIN([_textStorage attributesAtIndex: range.location
effectiveRange: NULL]);
[_textStorage beginEditing];
[_textStorage deleteCharactersInRange: range];
[_textStorage endEditing];
[self setTypingAttributes: attributes];
RELEASE(attributes);
[self didChangeText];
}
- (void) deleteToEndOfLine: (id)sender
{
NSRange range = [self rangeForUserTextChange];
NSDictionary *attributes;
if (range.location == NSNotFound)
{
return;
}
/* If the selection is not empty delete it, otherwise delete up to the
next line end from the insertion point or the delete the line end
itself when the insertion point is already at the end of the line. */
if (range.length == 0)
{
NSUInteger start, end, contentsEnd;
[[_textStorage string] getLineStart: &start
end: &end
contentsEnd: &contentsEnd
forRange: range];
if (range.location == contentsEnd)
{
range = NSMakeRange(contentsEnd, end - contentsEnd);
}
else
{
range.length = contentsEnd - range.location;
}
if (range.length == 0)
{
return;
}
}
if (![self shouldChangeTextInRange: range replacementString: @""])
{
return;
}
ASSIGN(killBuffer, [[_textStorage string] substringWithRange: range]);
attributes = RETAIN([_textStorage attributesAtIndex: range.location
effectiveRange: NULL]);
[_textStorage beginEditing];
[_textStorage deleteCharactersInRange: range];
[_textStorage endEditing];
[self setTypingAttributes: attributes];
RELEASE(attributes);
[self didChangeText];
}
- (void) deleteToEndOfParagraph: (id)sender
{
NSRange range = [self rangeForUserTextChange];
NSDictionary *attributes;
if (range.location == NSNotFound)
{
return;
}
/* If the selection is not empty delete it, otherwise delete up to
the next paragraph end from the insertion point or the delete the
paragraph end itself when the insertion point is already at the
end of the paragraph. */
if (range.length == 0)
{
NSUInteger start, end, contentsEnd;
[[_textStorage string] getParagraphStart: &start
end: &end
contentsEnd: &contentsEnd
forRange: range];
if (range.location == contentsEnd)
{
range = NSMakeRange(contentsEnd, end - contentsEnd);
}
else
{
range.length = contentsEnd - range.location;
}
if (range.length == 0)
{
return;
}
}
if (![self shouldChangeTextInRange: range replacementString: @""])
{
return;
}
ASSIGN(killBuffer, [[_textStorage string] substringWithRange: range]);
attributes = RETAIN([_textStorage attributesAtIndex: range.location
effectiveRange: NULL]);
[_textStorage beginEditing];
[_textStorage deleteCharactersInRange: range];
[_textStorage endEditing];
[self setTypingAttributes: attributes];
RELEASE(attributes);
[self didChangeText];
}
- (void) yank: (id)sender
{
if ([killBuffer length] > 0)
{
[self insertText: killBuffer];
}
}
/*
TODO: find out what affinity is supposed to mean
My current assumption:
Affinity deals with which direction we are selecting in, ie. which end of
the selected range is the moving end, and which is the anchor.
NSSelectionAffinityUpstream means that the minimum index of the selected
range is moving (ie. _selected_range.location).
NSSelectionAffinityDownstream means that the maximum index of the selected
range is moving (ie. _selected_range.location+_selected_range.length).
Thus, when moving and selecting, we use the affinity to find out which end
of the selected range to move, and after moving, we compare the character
index we moved to with the anchor and set the range and affinity.
The affinity is important when making keyboard selection have sensible
behavior. Example:
If, in the string "abcd", the insertion point is between the "c" and the "d"
(selected range is (3,0)), and the user hits shift-left twice, we select
the "c" and "b" (1,2) and set the affinity to NSSelectionAffinityUpstream.
If the user hits shift-right, only the "c" will be selected (2,1).
If the insertion point is between the "a" and the "b" (1,0) and the user hits
shift-right twice, we again select the "b" and "c" (1,2), but the affinity
is NSSelectionAffinityDownstream. If the user hits shift-right, the "d" is
added to the selection (1,3).
*/
- (unsigned int) _movementOrigin
{
NSRange range = [self selectedRange];
if ([self selectionAffinity] == NSSelectionAffinityUpstream)
return range.location;
else
return NSMaxRange(range);
}
- (NSUInteger) _movementEnd
{
NSRange range = [self selectedRange];
if ([self selectionAffinity] == NSSelectionAffinityDownstream)
return range.location;
else
return NSMaxRange(range);
}
- (void) _moveTo: (NSUInteger)cindex
select: (BOOL)select
{
if (select)
{
NSUInteger anchor = [self _movementEnd];
if (anchor < cindex)
{
[self setSelectedRange: NSMakeRange(anchor, cindex - anchor)
affinity: NSSelectionAffinityDownstream
stillSelecting: NO];
}
else
{
[self setSelectedRange: NSMakeRange(cindex, anchor - cindex)
affinity: NSSelectionAffinityUpstream
stillSelecting: NO];
}
}
else
{
[self setSelectedRange: NSMakeRange(cindex, 0)];
}
[self scrollRangeToVisible: NSMakeRange(cindex, 0)];
}
- (void) _moveFrom: (NSUInteger)cindex
direction: (GSInsertionPointMovementDirection)direction
distance: (CGFloat)distance
select: (BOOL)select
{
int new_direction;
if (direction == GSInsertionPointMoveUp ||
direction == GSInsertionPointMoveDown)
{
new_direction = 2;
}
else if (direction == GSInsertionPointMoveLeft ||
direction == GSInsertionPointMoveRight)
{
new_direction = 1;
}
else
{
new_direction = 0;
}
if (new_direction != _currentInsertionPointMovementDirection ||
!new_direction)
{
_originalInsertionPointCharacterIndex = cindex;
}
cindex = [_layoutManager characterIndexMoving: direction
fromCharacterIndex: cindex
originalCharacterIndex: _originalInsertionPointCharacterIndex
distance: distance];
[self _moveTo: cindex
select: select];
/* Setting the selected range will clear out the current direction, but
not the index. Thus, we always set the direction here. */
_currentInsertionPointMovementDirection = new_direction;
}
- (void) _move: (GSInsertionPointMovementDirection)direction
distance: (CGFloat)distance
select: (BOOL)select
{
[self _moveFrom: [self _movementOrigin]
direction: direction
distance: distance
select: select];
}
/*
* returns the character index for the left or right side of the selected text
* based upon the writing direction of the paragraph style.
* it should only be used when moving a literal direction such as left right
* up or down, not directions like forward, backward, beginning or end
*/
- (NSUInteger) _characterIndexForSelectedRange: (NSRange)range
direction: (GSInsertionPointMovementDirection)direction
{
NSUInteger cIndex;
NSParagraphStyle *parStyle;
NSWritingDirection writingDirection;
parStyle = [[self typingAttributes]
objectForKey: NSParagraphStyleAttributeName];
writingDirection = [parStyle baseWritingDirection];
switch (writingDirection)
{
case NSWritingDirectionLeftToRight:
cIndex = (direction == GSInsertionPointMoveLeft
|| direction == GSInsertionPointMoveUp)
? range.location
: NSMaxRange(range);
break;
case NSWritingDirectionRightToLeft:
cIndex = (direction == GSInsertionPointMoveLeft
|| direction == GSInsertionPointMoveUp)
? NSMaxRange(range)
: range.location;
break;
case NSWritingDirectionNaturalDirection:
// not sure if we should see this as it should resolve to either
// LeftToRight or RightToLeft in NSParagraphStyle
// for the users language.
//
// currently falls back to default..
default:
/* default to LeftToRight */
cIndex = (direction == GSInsertionPointMoveLeft
|| direction == GSInsertionPointMoveUp)
? range.location
: NSMaxRange(range);
break;
}
return cIndex;
}
/*
Insertion point movement actions.
TODO: some of these used to do nothing if self is a field editor. should
check if there was a reason for that.
*/
- (void) moveUp: (id)sender
{
NSRange range = [self selectedRange];
NSUInteger cIndex = [self _characterIndexForSelectedRange:range
direction:GSInsertionPointMoveUp];
[self _moveFrom: cIndex
direction: GSInsertionPointMoveUp
distance: 0.0
select: NO];
}
- (void) moveUpAndModifySelection: (id)sender
{
[self _move: GSInsertionPointMoveUp
distance: 0.0
select: YES];
}
- (void) moveDown: (id)sender
{
NSRange range = [self selectedRange];
NSUInteger cIndex = [self _characterIndexForSelectedRange: range
direction: GSInsertionPointMoveDown];
[self _moveFrom: cIndex
direction: GSInsertionPointMoveDown
distance: 0.0
select: NO];
}
- (void) moveDownAndModifySelection: (id)sender
{
[self _move: GSInsertionPointMoveDown
distance: 0.0
select: YES];
}
- (void) moveLeft: (id)sender
{
NSRange range = [self selectedRange];
if (range.length)
{
NSUInteger cIndex;
cIndex = [self _characterIndexForSelectedRange: range
direction:GSInsertionPointMoveLeft];
[self _moveTo: cIndex select: NO];
}
else
{
[self _move: GSInsertionPointMoveLeft
distance: 0.0
select: NO];
}
}
- (void) moveLeftAndModifySelection: (id)sender
{
NSParagraphStyle *parStyle;
NSWritingDirection writingDirection;
parStyle = [[self typingAttributes]
objectForKey: NSParagraphStyleAttributeName];
writingDirection = [parStyle baseWritingDirection];
if (writingDirection == NSWritingDirectionRightToLeft)
{
[self moveForwardAndModifySelection: sender];
}
else
{
[self moveBackwardAndModifySelection: sender];
}
}
- (void) moveRight: (id)sender
{
NSRange range = [self selectedRange];
if (range.length)
{
NSUInteger cIndex;
cIndex = [self _characterIndexForSelectedRange: range
direction: GSInsertionPointMoveRight];
[self _moveTo: cIndex select: NO];
}
else
{
[self _move: GSInsertionPointMoveRight
distance: 0.0
select: NO];
}
}
- (void) moveRightAndModifySelection: (id)sender
{
NSParagraphStyle *parStyle;
NSWritingDirection writingDirection;
parStyle = [[self typingAttributes]
objectForKey: NSParagraphStyleAttributeName];
writingDirection = [parStyle baseWritingDirection];
if (writingDirection == NSWritingDirectionRightToLeft)
{
[self moveBackwardAndModifySelection: sender];
}
else
{
[self moveForwardAndModifySelection: sender];
}
}
- (void) moveBackward: (id)sender
{
NSRange range = [self selectedRange];
NSUInteger to = range.location;
if (range.length == 0 && to)
{
to--;
}
[self _moveTo: to
select: NO];
}
- (void) moveBackwardAndModifySelection: (id)sender
{
NSUInteger to = [self _movementOrigin];
if (to == 0)
return;
to--;
[self _moveTo: to
select: YES];
}
- (void) moveForward: (id)sender
{
NSRange range = [self selectedRange];
NSUInteger to = NSMaxRange(range);
if (range.length == 0 && to != [_textStorage length])
{
to++;
}
[self _moveTo: to
select: NO];
}
- (void) moveForwardAndModifySelection: (id)sender
{
NSUInteger to = [self _movementOrigin];
if (to == [_textStorage length])
return;
to++;
[self _moveTo: to
select: YES];
}
- (void) moveWordBackward: (id)sender
{
NSRange range = [self selectedRange];
NSUInteger newLocation;
NSUInteger cIndex = range.location;
newLocation = [_textStorage nextWordFromIndex: cIndex
forward: NO];
[self _moveTo: newLocation
select: NO];
}
- (void) moveWordBackwardAndModifySelection: (id)sender
{
NSUInteger newLocation;
newLocation = [_textStorage nextWordFromIndex: [self _movementOrigin]
forward: NO];
[self _moveTo: newLocation
select: YES];
}
- (void) moveWordForward: (id)sender
{
NSUInteger newLocation;
NSUInteger cIndex = NSMaxRange([self selectedRange]);
newLocation = [_textStorage nextWordFromIndex: cIndex
forward: YES];
[self _moveTo: newLocation
select: NO];
}
- (void) moveWordForwardAndModifySelection: (id)sender
{
NSUInteger newLocation;
newLocation = [_textStorage nextWordFromIndex: [self _movementOrigin]
forward: YES];
[self _moveTo: newLocation
select: YES];
}
- (void) moveWordLeft: (id)sender
{
NSParagraphStyle *parStyle;
NSWritingDirection writingDirection;
parStyle = [[self typingAttributes]
objectForKey: NSParagraphStyleAttributeName];
writingDirection = [parStyle baseWritingDirection];
if (writingDirection == NSWritingDirectionRightToLeft)
{
[self moveWordForward: sender];
}
else
{
[self moveWordBackward: sender];
}
}
- (void) moveWordLeftAndModifySelection: (id)sender
{
NSParagraphStyle *parStyle;
NSWritingDirection writingDirection;
parStyle = [[self typingAttributes]
objectForKey: NSParagraphStyleAttributeName];
writingDirection = [parStyle baseWritingDirection];
if (writingDirection == NSWritingDirectionRightToLeft)
{
[self moveWordForwardAndModifySelection: sender];
}
else
{
[self moveWordBackwardAndModifySelection: sender];
}
}
- (void) moveWordRight: (id)sender
{
NSParagraphStyle *parStyle;
NSWritingDirection writingDirection;
parStyle = [[self typingAttributes]
objectForKey: NSParagraphStyleAttributeName];
writingDirection = [parStyle baseWritingDirection];
if (writingDirection == NSWritingDirectionRightToLeft)
{
[self moveWordBackward: sender];
}
else
{
[self moveWordForward: sender];
}
}
- (void) moveWordRightAndModifySelection: (id)sender
{
NSParagraphStyle *parStyle;
NSWritingDirection writingDirection;
parStyle = [[self typingAttributes]
objectForKey: NSParagraphStyleAttributeName];
writingDirection = [parStyle baseWritingDirection];
if (writingDirection == NSWritingDirectionRightToLeft)
{
[self moveWordBackwardAndModifySelection: sender];
}
else
{
[self moveWordForwardAndModifySelection: sender];
}
}
- (void) moveToBeginningOfDocument: (id)sender
{
[self _moveTo: 0
select: NO];
}
- (void) moveToBeginningOfDocumentAndModifySelection: (id)sender
{
[self _moveTo: 0
select:YES];
}
- (void) moveToEndOfDocument: (id)sender
{
[self _moveTo: [_textStorage length]
select: NO];
}
- (void) moveToEndOfDocumentAndModifySelection: (id)sender
{
[self _moveTo: [_textStorage length]
select:YES];
}
- (void) moveToBeginningOfParagraph: (id)sender
{
NSRange aRange = [self selectedRange];
aRange = [[_textStorage string] lineRangeForRange:
NSMakeRange(aRange.location, 0)];
[self _moveTo: aRange.location
select: NO];
}
- (void) moveToBeginningOfParagraphAndModifySelection: (id)sender
{
NSRange aRange;
aRange = [[_textStorage string] lineRangeForRange:
NSMakeRange([self _movementOrigin], 0)];
[self _moveTo: aRange.location
select: YES];
}
- (void) _moveToEndOfParagraph: (id)sender modify:(BOOL)flag
{
NSRange aRange;
NSUInteger newLocation;
NSUInteger maxRange;
NSUInteger cIndex;
if (flag)
{
cIndex = [self _movementOrigin];
}
else
{
cIndex = NSMaxRange([self selectedRange]);
}
aRange = [[_textStorage string] lineRangeForRange:
NSMakeRange(cIndex, 0)];
maxRange = NSMaxRange (aRange);
if (maxRange == 0)
{
/* Beginning of text is special only for technical reasons -
since maxRange is an unsigned, we can't safely subtract 1
from it if it is 0. */
newLocation = maxRange;
}
else if (maxRange == [_textStorage length])
{
/* End of text is special - we want the insertion point to
appear *after* the last character, which means as if before
the next (virtual) character after the end of text ... unless
the last character is a newline, and we are trying to go to
the end of the line which is displayed as the
one-before-the-last. Please note (maxRange - 1) is a valid
char since the maxRange == 0 case has already been
eliminated. */
unichar u = [[_textStorage string] characterAtIndex: (maxRange - 1)];
if (u == '\n' || u == '\r')
{
newLocation = maxRange - 1;
}
else
{
newLocation = maxRange;
}
}
else
{
/* Else, we want the insertion point to appear before the last
character in the paragraph range. Normally the last
character in the paragraph range is a newline. */
newLocation = maxRange - 1;
}
if (newLocation < aRange.location)
{
newLocation = aRange.location;
}
[self _moveTo: newLocation
select: flag];
}
- (void) moveToEndOfParagraph: (id)sender
{
[self _moveToEndOfParagraph:sender modify:NO];
}
- (void) moveToEndOfParagraphAndModifySelection: (id)sender
{
[self _moveToEndOfParagraph:sender modify:YES];
}
/* TODO: this is only the beginning and end of lines if lines are horizontal
and layout is left-to-right */
- (void) moveToBeginningOfLine: (id)sender
{
NSRange range = [self selectedRange];
NSUInteger cIndex = range.location;
[self _moveFrom: cIndex
direction: GSInsertionPointMoveLeft
distance: 1e8
select: NO];
}
- (void) moveToBeginningOfLineAndModifySelection: (id)sender
{
[self _move: GSInsertionPointMoveLeft
distance: 1e8
select: YES];
}
- (void) moveToEndOfLine: (id)sender
{
NSUInteger cIndex = NSMaxRange([self selectedRange]);
[self _moveFrom: cIndex
direction: GSInsertionPointMoveRight
distance: 1e8
select: NO];
}
- (void) moveToEndOfLineAndModifySelection: (id)sender
{
[self _move: GSInsertionPointMoveRight
distance: 1e8
select: YES];
}
/**
* Tries to move the selection/insertion point down one page of the
* visible rect in the receiver while trying to maintain the
* horizontal position of the last vertical movement.
* If the receiver is a field editor, this method returns immediatly.
*/
- (void) _pageDown: (id)sender modify: (BOOL)flag
{
CGFloat scrollDelta;
CGFloat oldOriginY;
CGFloat newOriginY;
NSUInteger cIndex;
if (flag)
{
cIndex = [self _movementOrigin];
}
else
{
cIndex = [self _characterIndexForSelectedRange: [self selectedRange]
direction: GSInsertionPointMoveDown];
}
/*
* Scroll; also determine how far to move the insertion point.
*/
oldOriginY = NSMinY([self visibleRect]);
[[self enclosingScrollView] pageDown: sender];
newOriginY = NSMinY([self visibleRect]);
scrollDelta = newOriginY - oldOriginY;
if (scrollDelta == 0)
{
[self _moveTo:[_textStorage length] select:flag];
return;
}
[self _moveFrom: cIndex
direction: GSInsertionPointMoveDown
distance: scrollDelta
select: flag];
}
- (void) pageDown:(id)sender
{
[self _pageDown:sender modify:NO];
}
- (void) pageDownAndModifySelection:(id)sender
{
[self _pageDown:sender modify:YES];
}
/**
* Tries to move the selection/insertion point up one page of the
* visible rect in the receiver while trying to maintain the
* horizontal position of the last vertical movement.
* If the receiver is a field editor, this method returns immediatly.
*/
- (void) _pageUp: (id)sender modify:(BOOL)flag
{
CGFloat scrollDelta;
CGFloat oldOriginY;
CGFloat newOriginY;
NSUInteger cIndex;
if (flag)
{
cIndex = [self _movementOrigin];
}
else
{
cIndex = [self _characterIndexForSelectedRange:[self selectedRange]
direction: GSInsertionPointMoveUp];
}
/*
* Scroll; also determine how far to move the insertion point.
*/
oldOriginY = NSMinY([self visibleRect]);
[[self enclosingScrollView] pageUp: sender];
newOriginY = NSMinY([self visibleRect]);
scrollDelta = newOriginY - oldOriginY;
if (scrollDelta == 0)
{
[self _moveTo:0 select:flag];
return;
}
[self _moveFrom: cIndex
direction: GSInsertionPointMoveUp
distance: -scrollDelta
select: flag];
}
- (void) pageUp:(id)sender
{
[self _pageUp:sender modify:NO];
}
- (void) pageUpAndModifySelection:(id)sender
{
[self _pageUp:sender modify:YES];
}
- (void) scrollLineDown: (id)sender
{
[[self enclosingScrollView] scrollLineDown: sender];
}
- (void) scrollLineUp: (id)sender
{
[[self enclosingScrollView] scrollLineUp: sender];
}
- (void) scrollPageDown: (id)sender
{
[[self enclosingScrollView] scrollPageDown: sender];
}
- (void) scrollPageUp: (id)sender
{
[[self enclosingScrollView] scrollPageUp: sender];
}
- (void) scrollToBeginningOfDocument: (id)sender
{
[[self enclosingScrollView] scrollToBeginningOfDocument: sender];
}
- (void) scrollToEndOfDocument: (id)sender
{
[[self enclosingScrollView] scrollToEndOfDocument: sender];
}
- (void) centerSelectionInVisibleArea: (id)sender
{
NSRange range;
NSPoint new;
NSRect rect, vRect;
vRect = [self visibleRect];
range = [self selectedRange];
if (range.length == 0)
{
rect =
[_layoutManager insertionPointRectForCharacterIndex: range.location
inTextContainer: _textContainer];
}
else
{
range = [_layoutManager glyphRangeForCharacterRange: range
actualCharacterRange: NULL];
rect = [_layoutManager boundingRectForGlyphRange: range
inTextContainer: _textContainer];
}
if (NSWidth(_bounds) <= NSWidth(vRect))
new.x = 0;
else if (NSWidth(rect) > NSWidth(vRect))
new.x = NSMinX(rect);
else
new.x = NSMinX(rect) - (NSWidth(vRect) - NSWidth(rect)) / 2;
if (NSHeight(_bounds) <= NSHeight(vRect))
new.y = 0;
else if (NSHeight(rect) > NSHeight(vRect))
new.y = NSMinY(rect);
else
new.y = NSMinY(rect) - (NSHeight(vRect) - NSHeight(rect)) / 2;
[self scrollPoint: new];
}
/* -selectAll: inherited from NSText */
- (void) selectLine: (id)sender
{
NSUInteger start, end, cindex;
cindex = [self _movementOrigin];
start = [_layoutManager characterIndexMoving: GSInsertionPointMoveLeft
fromCharacterIndex: cindex
originalCharacterIndex: cindex
distance: 1e8];
end = [_layoutManager characterIndexMoving: GSInsertionPointMoveRight
fromCharacterIndex: cindex
originalCharacterIndex: cindex
distance: 1e8];
[self setSelectedRange: NSMakeRange(start, end - start)];
}
/* The following method is bound to 'Control-t', and works exactly like
* pressing 'Control-t' inside Emacs, i.e., in general it swaps the
* character immediately before and after the insertion point and moves
* the insertion point forward by one character. If, however, the
* insertion point is at the end of a line, it swaps the two characters
* before the insertion point and does not move the insertion point.
* Note that Mac OS X does not implement the special case at the end
* of a line, but I consider Emacs' behavior more useful.
*/
- (void) transpose: (id)sender
{
NSRange range = [self selectedRange];
NSString *string;
NSString *replacementString;
unichar chars[2];
/* Do nothing if the selection is not empty or if we are at the
* beginning of text. */
if (range.length > 0 || range.location < 1)
{
return;
}
range = NSMakeRange(range.location - 1, 2);
/* Eventually adjust the range if we are at the end of a line. */
string = [_textStorage string];
if (range.location + 1 == [string length]
|| [string characterAtIndex: range.location + 1] == '\n')
{
if (range.location == 0)
return;
range.location -= 1;
}
/* Get the two chars and swap them. */
chars[1] = [string characterAtIndex: range.location];
chars[0] = [string characterAtIndex: (range.location + 1)];
/* Replace the original chars with the swapped ones. */
replacementString = [NSString stringWithCharacters: chars length: 2];
if ([self shouldChangeTextInRange: range
replacementString: replacementString])
{
[self replaceCharactersInRange: range
withString: replacementString];
[self setSelectedRange: NSMakeRange(range.location + 2, 0)];
[self didChangeText];
}
}
- (void) delete: (id)sender
{
[self deleteForward: sender];
}
/* Helper for -align*: */
- (void) _alignUser: (NSTextAlignment)alignment
{
NSRange r = [self rangeForUserParagraphAttributeChange];
if (r.location == NSNotFound)
return;
if (![self shouldChangeTextInRange: r
replacementString: nil])
return;
[self setAlignment: alignment
range: r];
[self didChangeText];
}
- (void) alignCenter: (id)sender
{
[self _alignUser: NSCenterTextAlignment];
}
- (void) alignLeft: (id)sender
{
[self _alignUser: NSLeftTextAlignment];
}
- (void) alignRight: (id)sender
{
[self _alignUser: NSRightTextAlignment];
}
- (void) alignJustified: (id)sender
{
[self _alignUser: NSJustifiedTextAlignment];
}
- (void) toggleContinuousSpellChecking: (id)sender
{
[self setContinuousSpellCheckingEnabled:
![self isContinuousSpellCheckingEnabled]];
}
- (void) toggleRuler: (id)sender
{
[self setRulerVisible: !_tf.is_ruler_visible];
}
- (void) outline: (id)sender
{
// FIXME
}
- (void) setBaseWritingDirection: (NSWritingDirection)direction
range: (NSRange)range
{
if (!_tf.is_rich_text)
return;
[_textStorage setBaseWritingDirection: direction range: range];
}
- (void) toggleBaseWritingDirection: (id)sender
{
// FIXME
}
@end