/** NSTextAttachment Classes to represent text attachments. NSTextAttachment is used to represent text attachments. When inline, text attachments appear as the value of the NSAttachmentAttributeName attached to the special character NSAttachmentCharacter. NSTextAttachment uses an object obeying the NSTextAttachmentCell protocol to get input from the user and to display an image. NSTextAttachmentCell is a simple subclass of NSCell which provides the NSTextAttachment protocol. Copyright (C) 2000 Free Software Foundation, Inc. Author: Fred Kiefer Date: June 2000 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 "AppKit/NSCell.h" #import "AppKit/NSFileWrapper.h" #import "AppKit/NSFileWrapperExtensions.h" #import "AppKit/NSImage.h" #import "AppKit/NSEvent.h" #import "AppKit/NSTextContainer.h" #import "AppKit/NSTextAttachment.h" #import "AppKit/NSTextView.h" @implementation NSTextAttachmentCell - (void)drawWithFrame: (NSRect)cellFrame inView: (NSView *)controlView characterIndex: (NSUInteger)charIndex { // cellFrame.origin.y -= cellFrame.size.height; [self drawWithFrame: cellFrame inView: controlView]; } - (void)drawWithFrame: (NSRect)cellFrame inView: (NSView *)controlView characterIndex: (NSUInteger)charIndex layoutManager: (NSLayoutManager *)layoutManager { [self drawWithFrame: cellFrame inView: controlView characterIndex: charIndex]; } - (NSPoint)cellBaselineOffset { return NSZeroPoint; } - (NSRect)cellFrameForTextContainer: (NSTextContainer *)textContainer proposedLineFragment: (NSRect)lineFrag glyphPosition: (NSPoint)position characterIndex: (NSUInteger)charIndex { NSRect aRect; aRect.origin = [self cellBaselineOffset]; aRect.size = [self cellSize]; return aRect; } - (BOOL)wantsToTrackMouse { return YES; } - (BOOL)wantsToTrackMouseForEvent: (NSEvent *)theEvent inRect: (NSRect)cellFrame ofView: (NSView *)controlView atCharacterIndex: (NSUInteger)charIndex { return [self wantsToTrackMouse]; } - (BOOL)trackMouse: (NSEvent *)theEvent inRect: (NSRect)cellFrame ofView: (NSView *)controlView untilMouseUp: (BOOL)flag { if ([controlView respondsToSelector: @selector(delegate)]) { NSTextView *textView = (NSTextView*)controlView; id delegate = [textView delegate]; NSEventType type = [theEvent type]; if (type == NSLeftMouseUp) { if ([theEvent clickCount] == 2) { if (delegate != nil && [delegate respondsToSelector: @selector(textView:doubleClickedOnCell:inRect:)]) { [delegate textView: textView doubleClickedOnCell: self inRect: cellFrame]; return YES; } } else { if (delegate != nil && [delegate respondsToSelector: @selector(textView:clickedOnCell:inRect:)]) { [delegate textView: textView clickedOnCell: self inRect: cellFrame]; return YES; } } } else if (type == NSLeftMouseDragged) { if (delegate != nil && [delegate respondsToSelector: @selector(textView:draggedCell:inRect:event:)]) { [delegate textView: textView draggedCell: self inRect: cellFrame event: theEvent]; return YES; } } } return [super trackMouse: theEvent inRect: cellFrame ofView: controlView untilMouseUp: flag]; } - (BOOL)trackMouse: (NSEvent *)theEvent inRect: (NSRect)cellFrame ofView: (NSView *)controlView atCharacterIndex: (NSUInteger)charIndex untilMouseUp: (BOOL)flag { if ([controlView respondsToSelector: @selector(delegate)]) { NSTextView *textView = (NSTextView*)controlView; id delegate = [textView delegate]; NSEventType type = [theEvent type]; if (type == NSLeftMouseDown) { if ([theEvent clickCount] == 2) { if (delegate != nil) { if ([delegate respondsToSelector: @selector(textView:doubleClickedOnCell:inRect:atIndex:)]) { [delegate textView: textView doubleClickedOnCell: self inRect: cellFrame atIndex: charIndex]; return YES; } else if ([delegate respondsToSelector: @selector(textView:doubleClickedOnCell:inRect:)]) { [delegate textView: textView doubleClickedOnCell: self inRect: cellFrame]; return YES; } } } else { if (delegate != nil) { if ([delegate respondsToSelector: @selector(textView:clickedOnCell:inRect:atIndex:)]) { [delegate textView: textView clickedOnCell: self inRect: cellFrame atIndex: charIndex]; return YES; } else if ([delegate respondsToSelector: @selector(textView:clickedOnCell:inRect:)]) { [delegate textView: textView clickedOnCell: self inRect: cellFrame]; return YES; } } } } else if (type == NSLeftMouseDragged) { if (delegate != nil && [delegate respondsToSelector: @selector(textView:draggedCell:inRect:event:atIndex:)]) { [delegate textView: textView draggedCell: self inRect: cellFrame event: theEvent atIndex: charIndex]; return YES; } } } return [self trackMouse: theEvent inRect: cellFrame ofView: controlView untilMouseUp: flag]; } - (void)setAttachment: (NSTextAttachment *)anObject { // Do not retain the attachment _attachment = anObject; } - (NSTextAttachment *)attachment { return _attachment; } //FIXME: I had to add those methods to keep the compiler quite. // This are already defined on the super class and should be taken from there. - (NSSize)cellSize { return [super cellSize]; } - (void)highlight: (BOOL)flag withFrame: (NSRect)cellFrame inView: (NSView *)controlView { [super highlight: flag withFrame: cellFrame inView: controlView]; } - (void)drawWithFrame: (NSRect)cellFrame inView: (NSView *)controlView { [super drawWithFrame: cellFrame inView: controlView]; } @end @implementation NSTextAttachment - (id) init { return [self initWithFileWrapper: nil]; } - (void) dealloc { DESTROY(_fileWrapper); DESTROY(_cell); [super dealloc]; } - (id) initWithFileWrapper: (NSFileWrapper *)fileWrapper { self = [super init]; if (self != nil) { _cell = [[NSTextAttachmentCell alloc ] init]; [self setFileWrapper: fileWrapper]; } return self; } - (void)setFileWrapper: (NSFileWrapper *)fileWrapper { ASSIGN(_fileWrapper, fileWrapper); // Reset the cell, so it shows the new attachment if ([_cell respondsToSelector: @selector(setAttachment:)] == YES) { [_cell setAttachment: self]; } if (_taflags.cell_explicitly_set == 0) { if (fileWrapper != nil) { NSImage *icon = nil; NSString *fileName = [fileWrapper filename]; if (fileName != nil) { // Try to set the image to the file wrapper content icon = [[NSImage alloc] initByReferencingFile: fileName]; } if (icon == nil) icon = RETAIN([fileWrapper icon]); [(NSTextAttachmentCell*)_cell setImage: icon]; RELEASE(icon); } } } - (NSFileWrapper *)fileWrapper { return _fileWrapper; } - (id )attachmentCell { return _cell; } - (void)setAttachmentCell: (id )cell { ASSIGN(_cell, cell); _taflags.cell_explicitly_set = 1; if ([_cell respondsToSelector: @selector(setAttachment:)] == YES) { [_cell setAttachment: self]; } } /* * NSCoding protocol */ - (void) encodeWithCoder: (NSCoder*)aCoder { if ([aCoder allowsKeyedCoding]) { [aCoder encodeObject: [self fileWrapper] forKey: @"NSFileWrapper"]; if (_cell != nil) { [aCoder encodeObject: _cell forKey: @"NSCell"]; } } else { [aCoder encodeObject: _fileWrapper]; [aCoder encodeObject: _cell]; } } - (id) initWithCoder: (NSCoder*)aDecoder { if ([aDecoder allowsKeyedCoding]) { [self setFileWrapper: [aDecoder decodeObjectForKey: @"NSFileWrapper"]]; [self setAttachmentCell: [aDecoder decodeObjectForKey: @"NSCell"]]; } else { [aDecoder decodeValueOfObjCType: @encode(id) at: &_fileWrapper]; [aDecoder decodeValueOfObjCType: @encode(id) at: &_cell]; // Reconnect the cell, so the cell does not have to store the attachment if ([_cell respondsToSelector: @selector(setAttachment:)] == YES) { [_cell setAttachment: self]; } } return self; } @end