/**
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