2017-05-27 14:32:32 +03:00
|
|
|
#import <Carbon/Carbon.h>
|
|
|
|
#import "GBTerminalTextFieldCell.h"
|
|
|
|
|
|
|
|
@interface GBTerminalTextView : NSTextView
|
|
|
|
@end
|
|
|
|
|
|
|
|
@implementation GBTerminalTextFieldCell
|
|
|
|
{
|
|
|
|
GBTerminalTextView *field_editor;
|
|
|
|
}
|
|
|
|
|
|
|
|
- (NSTextView *)fieldEditorForView:(NSView *)controlView
|
|
|
|
{
|
|
|
|
if (field_editor) {
|
|
|
|
return field_editor;
|
|
|
|
}
|
|
|
|
field_editor = [[GBTerminalTextView alloc] init];
|
|
|
|
[field_editor setFieldEditor:YES];
|
|
|
|
return field_editor;
|
|
|
|
}
|
|
|
|
|
|
|
|
@end
|
|
|
|
|
|
|
|
@implementation GBTerminalTextView
|
|
|
|
{
|
|
|
|
NSMutableOrderedSet *lines;
|
|
|
|
NSUInteger current_line;
|
2017-05-27 17:15:52 +03:00
|
|
|
bool reverse_search_mode;
|
2017-05-27 14:32:32 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
- (instancetype)init
|
|
|
|
{
|
|
|
|
self = [super init];
|
|
|
|
if (!self) {
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
lines = [[NSMutableOrderedSet alloc] init];
|
|
|
|
return self;
|
|
|
|
}
|
|
|
|
|
|
|
|
- (void)moveUp:(id)sender
|
|
|
|
{
|
2017-05-27 17:15:52 +03:00
|
|
|
reverse_search_mode = false;
|
2017-05-27 14:32:32 +03:00
|
|
|
if (current_line != 0) {
|
|
|
|
current_line--;
|
|
|
|
[self setString:[lines objectAtIndex:current_line]];
|
|
|
|
}
|
2017-05-27 17:15:52 +03:00
|
|
|
else {
|
|
|
|
[self setSelectedRange:NSMakeRange(0, 0)];
|
|
|
|
NSBeep();
|
|
|
|
}
|
2017-05-27 14:32:32 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
- (void)moveDown:(id)sender
|
|
|
|
{
|
2017-05-27 17:15:52 +03:00
|
|
|
reverse_search_mode = false;
|
2017-05-27 14:32:32 +03:00
|
|
|
if (current_line == [lines count]) {
|
2017-05-27 17:15:52 +03:00
|
|
|
[self setString:@""];
|
|
|
|
NSBeep();
|
2017-05-27 14:32:32 +03:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
current_line++;
|
|
|
|
if (current_line == [lines count]) {
|
|
|
|
[self setString:@""];
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
[self setString:[lines objectAtIndex:current_line]];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-05-27 17:15:52 +03:00
|
|
|
-(void)insertNewline:(id)sender
|
2017-05-27 14:32:32 +03:00
|
|
|
{
|
|
|
|
if ([self.string length]) {
|
|
|
|
NSString *string = [self.string copy];
|
|
|
|
[lines removeObject:string];
|
|
|
|
[lines addObject:string];
|
|
|
|
}
|
|
|
|
[super insertNewline:sender];
|
|
|
|
current_line = [lines count];
|
2017-05-27 17:15:52 +03:00
|
|
|
reverse_search_mode = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
- (void)keyDown:(NSEvent *)event
|
|
|
|
{
|
|
|
|
if (event.keyCode == kVK_ANSI_R && (event.modifierFlags & NSEventModifierFlagDeviceIndependentFlagsMask) == NSEventModifierFlagControl) {
|
|
|
|
if ([lines count] == 0) {
|
|
|
|
NSBeep();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (!reverse_search_mode) {
|
|
|
|
[self selectAll:self];
|
|
|
|
current_line = [lines count] - 1;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
if (current_line != 0) {
|
|
|
|
current_line--;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
NSBeep();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (self.string.length) {
|
|
|
|
[self updateReverseSearch];
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
[self setNeedsDisplay:YES];
|
|
|
|
reverse_search_mode = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
[super keyDown:event];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
- (void) updateReverseSearch
|
|
|
|
{
|
|
|
|
NSUInteger old_line = current_line;
|
|
|
|
reverse_search_mode = false;
|
|
|
|
NSString *substring = [self.string substringWithRange:self.selectedRange];
|
|
|
|
do {
|
|
|
|
NSString *line = [lines objectAtIndex:current_line];
|
|
|
|
NSRange range = [line rangeOfString:substring];
|
|
|
|
if (range.location != NSNotFound) {
|
|
|
|
self.string = line;
|
|
|
|
[self setSelectedRange:range];
|
|
|
|
reverse_search_mode = true;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
} while (current_line--);
|
|
|
|
current_line = old_line;
|
|
|
|
reverse_search_mode = true;
|
|
|
|
NSBeep();
|
|
|
|
}
|
|
|
|
|
|
|
|
- (void) insertText:(NSString *)string replacementRange:(NSRange)range
|
|
|
|
{
|
|
|
|
if (reverse_search_mode) {
|
|
|
|
range = self.selectedRange;
|
|
|
|
self.string = [[self.string substringWithRange:range] stringByAppendingString:string];
|
|
|
|
[self selectAll:nil];
|
|
|
|
[self updateReverseSearch];
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
[super insertText:string replacementRange:range];
|
|
|
|
}
|
2017-05-27 14:32:32 +03:00
|
|
|
}
|
|
|
|
|
2017-05-27 17:15:52 +03:00
|
|
|
-(void)deleteBackward:(id)sender
|
|
|
|
{
|
|
|
|
if (reverse_search_mode && self.string.length) {
|
|
|
|
NSRange range = self.selectedRange;
|
|
|
|
range.length--;
|
|
|
|
self.string = [self.string substringWithRange:range];
|
|
|
|
if (range.length) {
|
|
|
|
[self selectAll:nil];
|
|
|
|
[self updateReverseSearch];
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
reverse_search_mode = true;
|
|
|
|
current_line = [lines count] - 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
[super deleteBackward:sender];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
-(void)setSelectedRanges:(NSArray<NSValue *> *)ranges affinity:(NSSelectionAffinity)affinity stillSelecting:(BOOL)stillSelectingFlag
|
|
|
|
{
|
|
|
|
reverse_search_mode = false;
|
|
|
|
[super setSelectedRanges:ranges affinity:affinity stillSelecting:stillSelectingFlag];
|
|
|
|
}
|
|
|
|
|
|
|
|
- (BOOL)resignFirstResponder {
|
|
|
|
reverse_search_mode = false;
|
|
|
|
return [super resignFirstResponder];
|
|
|
|
}
|
|
|
|
|
|
|
|
-(void)drawRect:(NSRect)dirtyRect
|
|
|
|
{
|
|
|
|
[super drawRect:dirtyRect];
|
|
|
|
if (reverse_search_mode && [super string].length == 0) {
|
|
|
|
NSMutableDictionary *attributes = [self.typingAttributes mutableCopy];
|
|
|
|
NSColor *color = [attributes[NSForegroundColorAttributeName] colorWithAlphaComponent:0.5];
|
|
|
|
[attributes setObject:color forKey:NSForegroundColorAttributeName];
|
|
|
|
[[[NSAttributedString alloc] initWithString:@"Reverse search..." attributes:attributes] drawAtPoint:NSMakePoint(2, 0)];
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
2017-05-27 14:32:32 +03:00
|
|
|
@end
|