// // EmulatorView.m // 2Term // // Created by Kelvin Sherlock on 7/3/2010. // Copyright 2010 __MyCompanyName__. All rights reserved. // #import #include #include #import "EmulatorView.h" #import "CharacterGenerator.h" #import "VT52.h" #import "VT100.h" #include "OutputChannel.h" #import "ScanLineFilter.h" @implementation EmulatorView @synthesize fd = _fd; #pragma mark - #pragma mark properties -(void)setFd: (int)fd { _fd = fd; _screen.setFD(fd); } #pragma mark - -(void)awakeFromNib { CIFilter *filter; NSMutableArray *filters; _charWidth = 7; _charHeight = 16; _foregroundColor = [[NSColor greenColor] retain]; _backgroundColor = [[NSColor blackColor] retain]; _boldColor = [[NSColor redColor] retain]; [self setWantsLayer: YES]; filters = [NSMutableArray arrayWithCapacity: 3]; //add the scanlines (which are vertical and must therfore be rotated filter = [[ScanLineFilter new] autorelease]; [filter setValue: [NSNumber numberWithFloat: 0.75] forKey: @"inputOpacity"]; [filters addObject: filter]; //blur it a bit... filter = [CIFilter filterWithName: @"CIGaussianBlur"]; [filter setDefaults]; [filter setValue: [NSNumber numberWithFloat: 0.33] forKey: @"inputRadius"]; [filters addObject: filter]; [self setContentFilters: filters]; _screen.setFD(_fd); _screen.setView(self); _charGen = [[CharacterGenerator generator] retain]; _cursorImg = [[_charGen imageForCharacter: '_'] retain]; _emulator = [VT52 new]; } -(BOOL)isFlipped { return YES; } -(void)viewDidMoveToWindow { [self becomeFirstResponder]; _cursorTimer = [[NSTimer scheduledTimerWithTimeInterval: .5 target: self selector: @selector(cursorTimer:) userInfo: nil repeats: YES] retain]; } -(void)viewDidMoveToSuperview { [self becomeFirstResponder]; } -(BOOL)acceptsFirstResponder { return YES; } -(void)drawRect:(NSRect)dirtyRect { NSRect bounds = [self bounds]; NSRect screenRect = dirtyRect; unsigned x, y; NSColor *currentFront; NSColor *currentBack; BOOL setFront = YES; screenRect.origin.x -= _paddingLeft; screenRect.origin.y -= _paddingTop; if (screenRect.origin.x < 0) { screenRect.size.width -= screenRect.origin.x; screenRect.origin.x = 0; } if (screenRect.origin.y < 0) { screenRect.size.width -= screenRect.origin.y; screenRect.origin.y = 0; } unsigned minX = floor(screenRect.origin.x / _charWidth); unsigned maxX = ceil((screenRect.origin.x + screenRect.size.width) / _charWidth); unsigned minY = floor(screenRect.origin.y / _charHeight); unsigned maxY = ceil((screenRect.origin.y + screenRect.size.height) / _charHeight); // x/y are 0-indexed here. maxY = std::min(_screen.height() - 1, maxY); maxX = std::min(_screen.width() - 1, maxX); [_backgroundColor setFill]; NSRectFill(dirtyRect); [_foregroundColor setFill]; currentFront = _foregroundColor; currentBack = _backgroundColor; _screen.lock(); for (x = minX; x <= maxX; ++x) { for (y = minY; y <= maxY; ++y) { NSRect charRect = NSMakeRect(_paddingLeft + x * _charWidth, _paddingTop + y *_charHeight, _charWidth, _charHeight); NSImage *img; CharInfo ci = _screen.getc(x, y); unsigned flag = ci.flag; // todo -- check flags to determine fg/bg color, etc. // need to draw background individually.... currentBack = _backgroundColor; currentFront = _foregroundColor; if (flag & Screen::FlagBold) currentFront = _boldColor; img = [_charGen imageForCharacter: ci.c]; if (flag & Screen::FlagInverse) { std::swap(currentBack, currentFront); } if (currentBack != _backgroundColor) { [currentBack setFill]; NSRectFill(charRect); setFront = YES; } if (_foregroundColor != currentFront) setFront = YES; /* [img drawAtPoint: NSMakePoint(x * _charWidth, y * _charHeight) fromRect: NSZeroRect operation:NSCompositeCopy fraction: 1.0]; */ if (img) { if (setFront) [currentFront setFill]; [img drawInRect: charRect fromRect: NSZeroRect operation: NSCompositeCopy fraction: 1.0 respectFlipped: YES hints: nil]; } } } // cursor. iPoint cursor = _screen.cursor(); if (_cursorOn && iRect(minX, minY, maxX - minX, maxY - minY).contains(cursor)) { NSRect charRect = NSMakeRect(_paddingLeft + cursor.x * _charWidth, _paddingTop + cursor.y *_charHeight, _charWidth, _charHeight); [_foregroundColor setFill]; [_cursorImg drawInRect: charRect fromRect: NSZeroRect operation: NSCompositeCopy fraction: 1.0 respectFlipped: YES hints: nil]; } _screen.unlock(); //[_scanLine setFill]; //NSRectFillUsingOperation(screenRect, NSCompositeSourceOver); //NSRectFill(screenRect); } -(void)dealloc { close(_fd); [_foregroundColor release]; [_backgroundColor release]; [_readerThread release]; [_emulator release]; [_cursorImg release]; [super dealloc]; } -(void)keyDown:(NSEvent *)theEvent { OutputChannel channel(_fd); iRect updateRect; // should be nil but whatever... _screen.beginUpdate(); [_emulator keyDown: theEvent screen: &_screen output: &channel]; updateRect = _screen.endUpdate(); [self invalidateIRect: updateRect]; } -(void)autoTypeText:(NSString *)text { typedef void (*ProcessCharFX)(id, SEL, uint8_t, Screen *, OutputChannel *); std::vector chars; std::vector::iterator iter; iRect updateRect; // should be nil but whatever... OutputChannel channel(_fd); SEL cmd = @selector(processCharacter: screen: output:); ProcessCharFX fx = (ProcessCharFX)[_emulator methodForSelector: cmd]; unsigned length = [text length]; if (!length) return; chars.resize(length); [text getCharacters: &chars[0] range: NSMakeRange(0, length)]; _screen.beginUpdate(); // this posts as if it was output, need to post as if it was input for (iter = chars.begin(); iter != chars.end(); ++iter) { fx(_emulator,cmd, *iter, &_screen, &channel); } updateRect = _screen.endUpdate(); [self invalidateIRect: updateRect]; } -(void)startBackgroundReader { if (_readerThread) return; _readerThread = [[NSThread alloc] initWithTarget: self selector: @selector(_readerThread) object: nil]; [_readerThread start]; } -(void)_readerThread { // I would prefer to poll(2) but it's broken on os x for ptys. int fd = _fd; for(;;) { int n; fd_set read_set; fd_set error_set; FD_ZERO(&read_set); FD_SET(fd, &read_set); FD_ZERO(&error_set); FD_SET(fd, &error_set); n = select(fd + 1, &read_set, NULL, &error_set, NULL); if (n == 0) continue; if (n < 0) break; if (FD_ISSET(fd, &error_set)) break; if (FD_ISSET(fd, &read_set)) [self dataAvailable]; } } -(void)dataAvailable { typedef void (*ProcessCharFX)(id, SEL, uint8_t, Screen *, OutputChannel *); ProcessCharFX fx; SEL cmd; OutputChannel channel(_fd); cmd = @selector(processCharacter: screen: output:); fx = (ProcessCharFX)[_emulator methodForSelector: cmd]; for(;;) { NSAutoreleasePool *pool; iRect updateRect; uint8_t buffer[512]; ssize_t size; size = read(_fd, buffer, sizeof(buffer)); if (size == 0) break; if (size < 0) { if (errno == EINTR || errno == EAGAIN) continue; perror("[EmulatorView dataAvailable]"); break; } pool = [NSAutoreleasePool new]; _screen.beginUpdate(); for (unsigned i = 0; i < size; ++i) { fx(_emulator,cmd, buffer[i], &_screen, &channel); } updateRect = _screen.endUpdate(); [self invalidateIRect: updateRect]; [pool release]; } } -(void)invalidateIRect: (iRect)updateRect { NSRect rect; if (updateRect.size.width <= 0 || updateRect.size.height <= 0) return; rect.origin.x = updateRect.origin.x; rect.origin.y = updateRect.origin.y; rect.size.width = updateRect.size.width; rect.size.height = updateRect.size.height; rect.origin.x *= _charWidth; rect.origin.y *= _charHeight; rect.size.width *= _charWidth; rect.size.height *= _charHeight; rect.origin.x += _paddingLeft; rect.origin.y += _paddingTop; [self setNeedsDisplayInRect: rect]; } -(void)resizeTo: (iSize)size { NSWindow *window = [self window]; NSRect bounds = [self bounds]; NSSize newSize; NSRect wframe = [window frame]; // TODO -- left/right padding... newSize.width = size.width * _charWidth + _paddingLeft * 2; newSize.height = size.height * _charHeight + _paddingTop * 2; // best case -- no change. if (NSEqualSizes(newSize, bounds.size)) return; // ok, change needed. wframe.origin.y -= (newSize.height - bounds.size.height); wframe.size.height += newSize.height - bounds.size.height; wframe.size.width += newSize.width - bounds.size.width; [window setFrame: wframe display: YES animate: YES]; } -(void)cursorTimer: (NSTimer *)timer { _screen.lock(); _cursorOn = !_cursorOn; iRect r(_screen.cursor(), iSize(1,1)); [self invalidateIRect: r]; _screen.unlock(); } #if 0 -(void)viewWillStartLiveResize { NSLog(@"%s", sel_getName(_cmd)); } -(void)viewDidEndLiveResize { NSLog(@"%s", sel_getName(_cmd)); } #endif -(void)setFrame:(NSRect)frameRect { //NSLog(@"%s", sel_getName(_cmd)); if ([self inLiveResize]) { unsigned width = floor((frameRect.size.width - _paddingLeft) / _charWidth); unsigned height = floor((frameRect.size.height - _paddingTop) / _charHeight); _screen.lock(); _screen.setSize(width, height, false); _screen.unlock(); } [super setFrame: frameRect]; } #pragma mark - #pragma mark IBActions -(IBAction)paste: (id)sender { NSPasteboard *pasteboard = [NSPasteboard generalPasteboard]; NSArray *classArray = [NSArray arrayWithObject:[NSString class]]; NSDictionary *options = [NSDictionary dictionary]; BOOL ok = [pasteboard canReadObjectForClasses:classArray options:options]; if (ok) { NSArray *objectsToPaste = [pasteboard readObjectsForClasses:classArray options:options]; NSString *string = [objectsToPaste objectAtIndex: 0]; NSLog(@"%@", objectsToPaste); [self autoTypeText: string]; } } -(IBAction)copy: (id)sender { } @end void ViewScreen::setSize(unsigned width, unsigned height) { setSize(width, height, true); } void ViewScreen::setSize(unsigned width, unsigned height, bool resizeView) { // struct winsize ws; ws.ws_row = height; ws.ws_col = width; ws.ws_xpixel = 0; ws.ws_ypixel = 0; Screen::setSize(width, height); ioctl(_fd, TIOCSWINSZ, &ws); if (resizeView) { [_view resizeTo: iSize(width, height)]; } }