/** NSCell The abstract cell class Copyright (C) 1996 Free Software Foundation, Inc. Author: Scott Christley Date: 1996 Modifications: Felipe A. Rodriguez Date: August 1998 Rewrite: Multiple authors Date: 1999 Editing, formatters: Nicola Pero Date: 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 3 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. */ #include "config.h" #include #include #include #include #include #include #include #include #include #include #include "AppKit/AppKitExceptions.h" #include "AppKit/NSAttributedString.h" #include "AppKit/NSApplication.h" #include "AppKit/NSControl.h" #include "AppKit/NSCell.h" #include "AppKit/NSClipView.h" #include "AppKit/NSColor.h" #include "AppKit/NSCursor.h" #include "AppKit/NSEvent.h" #include "AppKit/NSFont.h" #include "AppKit/NSGraphics.h" #include "AppKit/NSImage.h" #include "AppKit/NSMenu.h" #include "AppKit/NSParagraphStyle.h" #include "AppKit/NSTextView.h" #include "AppKit/NSTextContainer.h" #include "AppKit/NSView.h" #include "AppKit/NSWindow.h" #include "GNUstepGUI/GSTheme.h" static Class colorClass; static Class cellClass; static Class fontClass; static Class imageClass; static NSColor *txtCol; static NSColor *dtxtCol; static NSColor *shadowCol; @interface NSCell (PrivateColor) + (void) _systemColorsChanged: (NSNotification*)n; @end @implementation NSCell (PrivateColor) + (void) _systemColorsChanged: (NSNotification*)n { ASSIGN (txtCol, [colorClass controlTextColor]); ASSIGN (dtxtCol, [colorClass disabledControlTextColor]); ASSIGN (shadowCol, [colorClass controlDarkShadowColor]); } @end /** *

TODO Desctiption

*/ @implementation NSCell /* * Class methods */ + (void) initialize { if (self == [NSCell class]) { [self setVersion: 3]; colorClass = [NSColor class]; cellClass = [NSCell class]; fontClass = [NSFont class]; imageClass = [NSImage class]; /* * Watch for changes to system colors, and simulate an initial change * in order to set up our defaults. */ [[NSNotificationCenter defaultCenter] addObserver: self selector: @selector(_systemColorsChanged:) name: NSSystemColorsDidChangeNotification object: nil]; [self _systemColorsChanged: nil]; } } + (NSMenu*) defaultMenu { return nil; } + (NSFocusRingType) defaultFocusRingType { return NSFocusRingTypeDefault; } /**

This class method returns NO. This method should be overrided by subclasses.

*/ + (BOOL) prefersTrackingUntilMouseUp { return NO; } /* * Instance methods */ - (id) init { return [self initTextCell: @""]; } /**

Initializes and returns a new NSCell with a NSImage anImage. This method sets the image position to NSImageOnly and the cell's type to NSImageCellType.

See Also: -initTextCell:

*/ - (id) initImageCell: (NSImage*)anImage { _cell.type = NSImageCellType; _cell_image = RETAIN (anImage); _cell.image_position = NSImageOnly; _font = RETAIN ([fontClass systemFontOfSize: 0]); // Implicitly set by allocation: // //_font = nil; //_cell.contents_is_attributed_string = NO; //_cell.is_highlighted = NO; //_cell.is_disabled = NO; //_cell.is_editable = NO; //_cell.is_rich_text = NO; //_cell.imports_graphics = NO; //_cell.shows_first_responder = NO; //_cell.refuses_first_responder = NO; //_cell.sends_action_on_end_editing = NO; //_cell.is_bordered = NO; //_cell.is_bezeled = NO; //_cell.is_scrollable = NO; //_cell.is_selectable = NO; //_cell.state = 0; _action_mask = NSLeftMouseUpMask; _menu = [isa defaultMenu]; [self setFocusRingType: [isa defaultFocusRingType]]; return self; } /**

Initializes and returns a new NSCell with a NSString aString. This method sets the cell's type to NSTextCellType.

See Also: -initImageCell:

*/ - (id) initTextCell: (NSString*)aString { _cell.type = NSTextCellType; _contents = RETAIN (aString); _font = RETAIN ([fontClass systemFontOfSize: 0]); // Implicitly set by allocation: // //_cell.contents_is_attributed_string = NO; //_cell_image = nil; //_cell.image_position = NSNoImage; //_cell.is_disabled = NO; //_cell.state = 0; //_cell.is_highlighted = NO; //_cell.is_editable = NO; //_cell.is_bordered = NO; //_cell.is_bezeled = NO; //_cell.is_scrollable = NO; //_cell.is_selectable = NO; _action_mask = NSLeftMouseUpMask; _menu = [isa defaultMenu]; [self setFocusRingType: [isa defaultFocusRingType]]; return self; } - (void) dealloc { TEST_RELEASE (_contents); TEST_RELEASE (_cell_image); TEST_RELEASE (_font); TEST_RELEASE (_represented_object); TEST_RELEASE (_object_value); TEST_RELEASE (_formatter); TEST_RELEASE (_menu); [super dealloc]; } /* * Setting the NSCell's Value */ - (id) objectValue { if (_cell.has_valid_object_value) { return _object_value; } else { return nil; } } - (BOOL) hasValidObjectValue { return _cell.has_valid_object_value; } /**

Returns the NSCell's value as a double.

*

See Also: -setDoubleValue:

*/ - (double) doubleValue { if ((_cell.has_valid_object_value == YES) && ([_object_value respondsToSelector: @selector(doubleValue)])) { return [_object_value doubleValue]; } else { return [[self stringValue] doubleValue]; } } /**

Returns the cell's value as a float.

*

See Also: -setFloatValue:

*/ - (float) floatValue { if ((_cell.has_valid_object_value == YES) && ([_object_value respondsToSelector: @selector(floatValue)])) { return [_object_value floatValue]; } else { return [[self stringValue] floatValue]; } } /**

Returns the cell's value as an int.

*

See Also: -setIntValue:

*/ - (int) intValue { if ((_cell.has_valid_object_value == YES) && ([_object_value respondsToSelector: @selector(intValue)])) { return [_object_value intValue]; } else { return [[self stringValue] intValue]; } } /**

Returns the cell's value as a NSString.

*

See Also: -setStringValue:

*/ - (NSString*) stringValue { if (_cell.contents_is_attributed_string == NO) { // If we have a formatter this is also the string of the _object_value return _contents; } else { return [(NSAttributedString *)_contents string]; } } - (void) setObjectValue: (id)object { id newContents; ASSIGN (_object_value, object); if (_formatter == nil) { if ([object isKindOfClass: [NSString class]] == YES) { newContents = object; _cell.contents_is_attributed_string = NO; _cell.has_valid_object_value = YES; } if ([object isKindOfClass: [NSAttributedString class]] == YES) { newContents = object; _cell.contents_is_attributed_string = YES; _cell.has_valid_object_value = YES; } else { newContents = [_object_value description]; _cell.contents_is_attributed_string = NO; _cell.has_valid_object_value = YES; } } else { newContents = [_formatter stringForObjectValue: _object_value]; _cell.contents_is_attributed_string = NO; if (newContents != nil) { _cell.has_valid_object_value = YES; } else { _cell.has_valid_object_value = NO; } } ASSIGN (_contents, newContents); } /**

Sets the NSCell's value to aDouble.

*

See Also: -doubleValue

*/ - (void) setDoubleValue: (double)aDouble { NSNumber *number; // NB: GNUstep can set a double value for an image cell number = [NSNumber numberWithDouble: aDouble]; [self setObjectValue: number]; } /** *

Sets the NSCell's value to a aFloat. This used for example in NSSliderCell

*

See Also: -floatValue

*/ - (void) setFloatValue: (float)aFloat { NSNumber *number; // NB: GNUstep can set a float value for an image cell. // NSSliderCell is an example of it! number = [NSNumber numberWithFloat: aFloat]; [self setObjectValue: number]; } /** *

Sets the NSCell's value to anInt.

*

See Also: -intValue

*/ - (void) setIntValue: (int)anInt { NSNumber *number; // NB: GNUstep can set an int value for an image cell. number = [NSNumber numberWithInt: anInt]; [self setObjectValue: number]; } /**

Sets the cell's value to a NSString. The NSCell's type is set to NSTextCellType if needed

See Also: -stringValue

*/ - (void) setStringValue: (NSString*)aString { NSString *string = aString; /* We warn about nil for compatibiliy with MacOS X, which refuses nil. */ if (string == nil) { NSDebugMLLog (@"MacOSXCompatibility", @"Attempt to use nil as string value"); } if (_cell.type != NSTextCellType) { [self setType: NSTextCellType]; } _cell.contents_is_attributed_string = NO; if (_formatter == nil) { ASSIGN (_contents, string); ASSIGN (_object_value, string); _cell.has_valid_object_value = YES; } else { id newObjectValue; if ([_formatter getObjectValue: &newObjectValue forString: string errorDescription: NULL]) { [self setObjectValue: newObjectValue]; } else { _cell.has_valid_object_value = NO; ASSIGN (_contents, string); } } } /**

Returns some NSCell's attributes for the specified NSCellAttribute

See Also: -setCellAttribute:to:

*/ - (int) cellAttribute: (NSCellAttribute)aParameter { switch (aParameter) { case NSCellDisabled: return _cell.is_disabled; case NSCellState: return _cell.state; case NSCellEditable: return _cell.is_editable; case NSCellHighlighted: return _cell.is_highlighted; case NSCellIsBordered: return _cell.is_bordered; case NSCellAllowsMixedState: return _cell.allows_mixed_state; /* case NSPushInCell: return 0; case NSChangeGrayCell: return 0; case NSCellLightsByContents: return 0; case NSCellLightsByGray: return 0; case NSChangeBackgroundCell: return 0; case NSCellLightsByBackground: return 0; case NSCellChangesContents: return 0; case NSCellIsInsetButton: return 0; */ case NSCellHasOverlappingImage: { return _cell.image_position == NSImageOverlaps; } case NSCellHasImageHorizontal: { return (_cell.image_position == NSImageRight) || (_cell.image_position == NSImageLeft); } case NSCellHasImageOnLeftOrBottom: { return (_cell.image_position == NSImageBelow) || (_cell.image_position == NSImageLeft); } default: { NSWarnLog (@"cell attribute %d not supported", aParameter); break; } } return 0; } /**

TODO

*

See Also: -cellAttribute:

*/ - (void) setCellAttribute: (NSCellAttribute)aParameter to: (int)value { switch (aParameter) { case NSCellDisabled: { _cell.is_disabled = value; break; } case NSCellState: { _cell.state = value; break; } case NSCellEditable: { _cell.is_editable = value; break; } case NSCellHighlighted: { _cell.is_highlighted = value; break; } case NSCellHasOverlappingImage: { if (value) { _cell.image_position = NSImageOverlaps; } else { if (_cell.image_position == NSImageOverlaps) { _cell.image_position = NSImageLeft; } } break; } case NSCellHasImageHorizontal: { if (value) { if (_cell.image_position != NSImageLeft && _cell.image_position != NSImageRight) { _cell.image_position = NSImageLeft; } } else { if (_cell.image_position == NSImageLeft) { _cell.image_position = NSImageAbove; } else if (_cell.image_position == NSImageRight) { _cell.image_position = NSImageBelow; } } break; } case NSCellHasImageOnLeftOrBottom: { if (value) { if (_cell.image_position == NSImageAbove) { _cell.image_position = NSImageBelow; } else { _cell.image_position = NSImageLeft; } } else { if (_cell.image_position == NSImageBelow) { _cell.image_position = NSImageAbove; } else { _cell.image_position = NSImageRight; } } break; } /* case NSCellChangesContents: _cell. = value; break; case NSCellIsInsetButton: _cell. = value; break; */ case NSCellIsBordered: { _cell.is_bordered = value; break; } case NSCellAllowsMixedState: { _cell.allows_mixed_state = value; break; } default: { NSWarnLog (@"cell attribute %d not supported", aParameter); break; } } } /**

Sets the NSCell's type. See NSCellType .If the cell is set to NSTextCellType, the cell is given a default title and is reset to the default system font.

See Also: -type

*/ - (void) setType: (NSCellType)aType { if (_cell.type == aType) { return; } _cell.type = aType; switch (_cell.type) { case NSTextCellType: { ASSIGN (_contents, @"title"); _cell.contents_is_attributed_string = NO; /* Doc says we have to reset the font too. */ ASSIGN (_font, [fontClass systemFontOfSize: 0]); break; } case NSImageCellType: { TEST_RELEASE (_cell_image); _cell_image = nil; break; } } } /**

Returns the cell's type. Returns NSNullCellType if the cell's type flag is set to NSImageCellType and if the cell's image is nil. See NSCellType for more information.

See Also -setType:

*/ - (NSCellType) type { if (_cell.type == NSImageCellType && _cell_image == nil) return NSNullCellType; return _cell.type; } /**

Returns whether the NSCell can respond to mouse events.

*

See Also: -setEnabled:

*/ - (BOOL) isEnabled { return !_cell.is_disabled; } /**

Sets whether the NSCell can respond to mouse events

See Also: -isEnabled

*/ - (void) setEnabled: (BOOL)flag { _cell.is_disabled = !flag; } /**

Returns whether the NSCell has a bezeled border. By default a NSCell has no bezeled border

See Also: -setBezeled:

*/ - (BOOL) isBezeled { return _cell.is_bezeled; } /**

Returns whether the NSCell has a border. By default a NSCell has border

See Also: -setBordered: -setBezeled: -isBezeled

*/ - (BOOL) isBordered { return _cell.is_bordered; } /**

Returns whether the cell is opaque. Return NO by default

*/ - (BOOL) isOpaque { return NO; } /**

Sets whether the cell has a bezeled border. If this method is called, the bordered flag is turn off. By default a NSCell has no bezeled border

See Also: -isBezeled -setBordered: -isBordered

*/ - (void) setBezeled: (BOOL)flag { _cell.is_bezeled = flag; _cell.is_bordered = NO; } /**

Sets whether the cell has a border. If this method is called, the bezeled flag is turn off. By default a NSCell has no border

See Also: -isBordered -setBezeled: -isBezeled

*/ - (void) setBordered: (BOOL)flag { _cell.is_bordered = flag; _cell.is_bezeled = NO; } - (NSFocusRingType) focusRingType { return _cell.focus_ring_type; } - (void) setFocusRingType: (NSFocusRingType)type { _cell.focus_ring_type = type; } /**

Sets the NSCell's state. Please use always symbolic constants when calling this method. The integer values could be changed in the this implementation. (Currently they match the Cocoa values but they are quite strange)

See Also: -state

*/ - (void) setState: (int)value { /* We do exactly as in macosx when value is not NSOnState, * NSOffState, NSMixedState, even if their behaviour (value < 0 ==> * NSMixedState) is a bit strange. We could decide to do * differently in the future, so please use always symbolic * constants when calling this method, this way your code won't be * broken by changes. */ if (value > 0 || (value < 0 && _cell.allows_mixed_state == NO)) { _cell.state = NSOnState; } else if (value == 0) { _cell.state = NSOffState; } else { _cell.state = NSMixedState; } } /**

Returns the NSCell's state

See Also: -setState:

*/ - (int) state { return _cell.state; } - (BOOL) allowsMixedState { return _cell.allows_mixed_state; } - (void) setAllowsMixedState: (BOOL)flag { _cell.allows_mixed_state = flag; if (!flag && _cell.state == NSMixedState) { [self setNextState]; } } - (int) nextState { switch (_cell.state) { case NSOnState: { return NSOffState; } case NSOffState: { if (_cell.allows_mixed_state) { return NSMixedState; } else { return NSOnState; } } case NSMixedState: default: { return NSOnState; } } } - (void) setNextState { [self setState: [self nextState]]; } /**

Returns the alignment of the text used in the NSCell. See NSTextAlignment for more informations. By default the text alignment is NSJustifiedTextAlignment

See Also: -setAlignment:

*/ - (NSTextAlignment) alignment { return _cell.text_align; } /**

Returns the font of the text used in the NSCell

See Also: -setFont:

*/ - (NSFont*) font { return _font; } /**

Returns whether the cell is editable.By default a NSCell is not editable.

See Also: -setEditable:

*/ - (BOOL) isEditable { return _cell.is_editable; } /**

Returns whether the cell is selectable. This method returns YES if the cell is selectable or editable. NO otherwise

See Also: -setSelectable: -isEditable -setEditable:

*/ - (BOOL) isSelectable { return _cell.is_selectable || _cell.is_editable; } /**

Returns whether the NSCell is scrollable. By default a NSCell is not scrollable

See Also: -setScrollable:

*/ - (BOOL) isScrollable { return _cell.is_scrollable; } /**

Sets the alignment of the text. See NSTextAlignment.

See Also: -alignment

*/ - (void) setAlignment: (NSTextAlignment)mode { // This does not have any influence on attributed strings _cell.text_align = mode; } /**

Sets whether the NSCell's text is editable.

See Also: -isEditable -setSelectable: -isSelectable

*/ - (void) setEditable: (BOOL)flag { /* * The cell_editable flag is also checked to see if the cell is * selectable so turning edit on also turns selectability on (until * edit is turned off again). */ _cell.is_editable = flag; } /**

Sets the text font. The NSCell's type is set to NSTextCellType if needed

See Also: -font -setType: -type

*/ - (void) setFont: (NSFont*)fontObject { if (_cell.type != NSTextCellType) { [self setType: NSTextCellType]; } // This does not have any influence on attributed strings ASSIGN (_font, fontObject); } /**

Sets whether the cell selectable. Making a cell unselectable also * makes it uneditable until a -setEditable: re-enables it.

*

See Also: -isSelectable -setEditable: -isEditable

*/ - (void) setSelectable: (BOOL)flag { _cell.is_selectable = flag; if (!flag) _cell.is_editable = NO; } /**

Sets whether the NCell is scrollable. By default a NSCell is not scrollable

See Also: -isSelectable

*/ - (void) setScrollable: (BOOL)flag { _cell.is_scrollable = flag; if (flag) { [self setWraps: NO]; } } - (void) setWraps: (BOOL)flag { _cell.wraps = flag; if (flag) { _cell.is_scrollable = NO; [self setLineBreakMode: NSLineBreakByWordWrapping]; } else { [self setLineBreakMode: NSLineBreakByClipping]; } } - (BOOL) wraps { return _cell.wraps; } - (void) setAttributedStringValue: (NSAttributedString*)attribStr { /* Hmm. FIXME. Not sure what to do here. */ if (_formatter != nil) { id newObjectValue; if ([_formatter getObjectValue: &newObjectValue forString: [attribStr string] errorDescription: NULL] == YES) { [self setObjectValue: newObjectValue]; /* What about the attributed string ? We are loosing it. */ return; } } /* In all other cases */ ASSIGN (_contents, attribStr); _cell.has_valid_object_value = NO; _cell.contents_is_attributed_string = YES; } - (NSAttributedString*) attributedStringValue { if (_formatter != nil) { NSDictionary *attributes; NSAttributedString *attrStr; attributes = [self _nonAutoreleasedTypingAttributes]; attrStr = [_formatter attributedStringForObjectValue: _object_value withDefaultAttributes: attributes]; RELEASE(attributes); if (attrStr != nil) { return attrStr; } } /* In all other cases */ if (_cell.contents_is_attributed_string) { return (NSAttributedString *)_contents; } else { NSDictionary *dict; NSAttributedString *attrStr; dict = [self _nonAutoreleasedTypingAttributes]; attrStr = [[NSAttributedString alloc] initWithString: _contents attributes: dict]; RELEASE(dict); return AUTORELEASE(attrStr); } } - (void) setAllowsEditingTextAttributes: (BOOL)flag { _cell.is_rich_text = flag; if (!flag) _cell.imports_graphics = NO; } - (BOOL) allowsEditingTextAttributes { return _cell.is_rich_text; } - (void) setImportsGraphics: (BOOL)flag { _cell.imports_graphics = flag; if (flag) _cell.is_rich_text = YES; } - (BOOL) importsGraphics { return _cell.imports_graphics; } - (NSString*) title { return [self stringValue]; } - (void) setTitle: (NSString*)aString { [self setStringValue: aString]; } - (NSLineBreakMode) lineBreakMode { return _cell.line_break_mode; } - (void) setLineBreakMode: (NSLineBreakMode)mode { _cell.line_break_mode = mode; } - (NSWritingDirection) baseWritingDirection { return _cell.base_writing_direction; } - (void) setBaseWritingDirection: (NSWritingDirection)direction { _cell.base_writing_direction = direction; } /**

Implemented by subclasses to return the action method. The NSCell implementaiton returns NULL.

See Also: -setAction: -setTarget: -target

*/ - (SEL) action { return NULL; } /**

Implemented by subclasses to set the action method. The NSCell implementation raises a NSInternalInconsistencyException

See Also: -action -setTarget: -target

*/ - (void) setAction: (SEL)aSelector { [NSException raise: NSInternalInconsistencyException format: @"attempt to set an action in an NSCell"]; } /**

Implemented by subclasses to set the target object. The NSCell implementation raises a NSInternalInconsistencyException

See Also: -target -setAction: -action

*/ - (void) setTarget: (id)anObject { [NSException raise: NSInternalInconsistencyException format: @"attempt to set a target in an NSCell"]; } /**

Implemented by subclass to return the target object. The NSCell implementation returns nil

See Also: -setTarget: -setAction: -action

*/ - (id) target { return nil; } /**

Returns whether the cell can continuously send its action messages.

See Also: -setContinuous:

*/ - (BOOL) isContinuous { // Some subclasses should redefine this with NSLeftMouseDraggedMask return (_action_mask & NSPeriodicMask) != 0; } /**

Sets whether the cell can continuously send its action messages.

*

See Also: -isContinuous

*/ - (void) setContinuous: (BOOL)flag { // Some subclasses should redefine this with NSLeftMouseDraggedMask if (flag) { _action_mask |= NSPeriodicMask; } else { _action_mask &= ~NSPeriodicMask; } } /**

TODO Explain

*/ - (int) sendActionOn: (int)mask { unsigned int previousMask = _action_mask; _action_mask = mask; return previousMask; } /**

Returns the NSCell's image if the NSCell's type is NSImageCellType, returns nil otherwise.

See Also: -setImage: -setType: -type

*/ - (NSImage*) image { if (_cell.type == NSImageCellType) { return _cell_image; } else return nil; } /**

Sets the NSCell's image to anImage. This method sets the cell's type to NSImageCellType if needed. Raises an NSInvalidArgumentException if the anImage is not an NSImage (sub)class. The new image is retained and the old one is released

See Also: -image

*/ - (void) setImage: (NSImage*)anImage { if (anImage) { NSAssert ([anImage isKindOfClass: imageClass], NSInvalidArgumentException); } if (_cell.type != NSImageCellType) { [self setType: NSImageCellType]; } ASSIGN (_cell_image, anImage); } /**

Implemented by sublclasses to assigns the tag anInt. The NSCell implementation raises an NSInvalidArgumentException.

See Also: -tag

*/ - (void) setTag: (int)anInt { [NSException raise: NSInternalInconsistencyException format: @"attempt to set a tag in an NSCell"]; } /**

Implemented by subclasses to Return the tag. The NSCell implementation returns -1

See Also: -setTag:

*/ - (int) tag { return -1; } /* * Formatting Data */ - (void) setFloatingPointFormat: (BOOL)autoRange left: (unsigned int)leftDigits right: (unsigned int)rightDigits { NSNumberFormatter *formatter = [[NSNumberFormatter alloc] init]; NSMutableString *format = [[NSMutableString alloc] init]; if (autoRange) { unsigned fieldWidth = leftDigits + rightDigits + 1; // FIXME: this does not fully match the documentation. while (fieldWidth--) { [format appendString: @"#"]; } } else { while (leftDigits--) { [format appendString: @"#"]; } [format appendString: @"."]; while (rightDigits--) { [format appendString: @"0"]; } } [formatter setFormat: format]; RELEASE(format); [self setFormatter: formatter]; RELEASE(formatter); } - (void) setFormatter: (NSFormatter*)newFormatter { ASSIGN(_formatter, newFormatter); } - (id) formatter { return _formatter; } /**

TODO

*/ - (int) entryType { return _cell.entry_type; } /**

TODO

*/ - (void) setEntryType: (int)aType { [self setType: NSTextCellType]; // TODO: This should select a suitable formatter _cell.entry_type = aType; } - (BOOL) isEntryAcceptable: (NSString*)aString { if ((_formatter != nil) && ![aString isEqualToString: @""]) { id newObjectValue; return [_formatter getObjectValue: &newObjectValue forString: aString errorDescription: NULL]; } else { return YES; } } /* * Menu */ - (void) setMenu: (NSMenu*)aMenu { ASSIGN (_menu, aMenu); } - (NSMenu*) menu { return _menu; } - (NSMenu*) menuForEvent: (NSEvent*)anEvent inRect: (NSRect)cellFrame ofView: (NSView*)aView { return [self menu]; } /** * Compares the reciever to another to another NSCell. * The argument must be an NSCell sublclass and have * the NSCellType NSTextCellType. Returns the result * of the comparison of each cell's stringValue. */ - (NSComparisonResult) compare: (id)otherCell { if ([otherCell isKindOfClass: cellClass] == NO) { [NSException raise: NSBadComparisonException format: @"NSCell comparison with non-NSCell"]; } if (_cell.type != NSTextCellType || ((NSCell*)otherCell)->_cell.type != NSTextCellType) { [NSException raise: NSBadComparisonException format: @"Comparison between non-text cells"]; } /* We shouldn't access instance variables directly as subclasses may override stringValue to retrieve the value from somewhere else. */ return [[self stringValue] compare: [(NSCell*)otherCell stringValue]]; } /* * Should this cell respond to keyboard input? */ - (BOOL) acceptsFirstResponder { return _cell.is_disabled == NO && _cell.refuses_first_responder == NO; } - (void) setShowsFirstResponder: (BOOL)flag { _cell.shows_first_responder = flag; } - (BOOL) showsFirstResponder { return _cell.shows_first_responder; } - (void) setTitleWithMnemonic: (NSString*)aString { unsigned int location = [aString rangeOfString: @"&"].location; [self setTitle: [aString stringByReplacingString: @"&" withString: @""]]; // TODO: We should underline this character [self setMnemonicLocation: location]; } - (NSString*) mnemonic { unsigned int location = [self mnemonicLocation]; NSString *c = [self title]; if ((location == NSNotFound) || location >= [c length]) return @""; return [c substringWithRange: NSMakeRange (location, 1)]; } - (void) setMnemonicLocation: (unsigned int)location { _cell.mnemonic_location = location; } - (unsigned int) mnemonicLocation { return _cell.mnemonic_location; } - (BOOL) refusesFirstResponder { return _cell.refuses_first_responder; } - (void) setRefusesFirstResponder: (BOOL)flag { _cell.refuses_first_responder = flag; } /** * Simulates a single click in the cell (only works with controls which have * no more than one cell). This method is deprecated, * performClickWithFrame:inView: is the right method to use now. */ - (void) performClick: (id)sender { NSView *cv = [self controlView]; if (cv != nil) [self performClickWithFrame: [cv bounds] inView: cv]; } /** * Simulates a single click in the cell. * The display of the cell with this event * occurs in the area delimited by cellFrame within * controlView. */ - (void) performClickWithFrame: (NSRect)cellFrame inView: (NSView *)controlView { SEL action = [self action]; if (_cell.is_disabled == YES) { return; } if (controlView != nil) { NSWindow *cvWin = [controlView window]; NSDate *limit = [NSDate dateWithTimeIntervalSinceNow: 0.1]; [controlView lockFocus]; [self setNextState]; [self highlight: YES withFrame: cellFrame inView: controlView]; [cvWin flushWindow]; // Wait approx 1/10 seconds [[NSRunLoop currentRunLoop] runUntilDate: limit]; [self highlight: NO withFrame: cellFrame inView: controlView]; [cvWin flushWindow]; [controlView unlockFocus]; if (action) { NS_DURING { [(NSControl*)controlView sendAction: action to: [self target]]; } NS_HANDLER { [localException raise]; } NS_ENDHANDLER } } else // We have no control view. The best we can do is the following. { if (action) { [self setNextState]; NS_DURING { [[NSApplication sharedApplication] sendAction: action to: [self target] from: self]; } NS_HANDLER { [localException raise]; } NS_ENDHANDLER } } } /* * Deriving values from other objects (not necessarily cells) */ - (void) takeObjectValueFrom: (id)sender { [self setObjectValue: [sender objectValue]]; } /**

Sets the NSCell's double value to sender's double value

See Also: -setDoubleValue:

*/ - (void) takeDoubleValueFrom: (id)sender { [self setDoubleValue: [sender doubleValue]]; } /**

Sets the NSCell's float value to sender's float value

See Also: -setFloatValue:

*/ - (void) takeFloatValueFrom: (id)sender { [self setFloatValue: [sender floatValue]]; } /**

Sets the NSCell's int value to sender's int value

See Also: -setIntValue:

*/ - (void) takeIntValueFrom: (id)sender { [self setIntValue: [sender intValue]]; } /**

Sets the NSCell's NSString value to sender's NSSting value

See Also: -setStringValue:

*/ - (void) takeStringValueFrom: (id)sender { [self setStringValue: [sender stringValue]]; } /**

Returns the NSCell's represented object

See Also: -setRepresentedObject:

*/ - (id) representedObject { return _represented_object; } /**

Sets the NSCell's represented object to anObject. anObject will be retain.

See Also: -representedObject

*/ - (void) setRepresentedObject: (id)anObject { /* Ahm - not nice - the RETAIN here could cause retain cycles - anyway. */ ASSIGN (_represented_object, anObject); } /**

Returns YES. Subclasses should overrided this method if you want stop tracking the mouse. This method is call in the -trackMouse:inRect:ofView:untilMouseUp: main loop.

See Also: -trackMouse:inRect:ofView:untilMouseUp:

*/ - (BOOL) continueTracking: (NSPoint)lastPoint at: (NSPoint)currentPoint inView: (NSView*)controlView { return YES; } /**

Returns the mouse flags. This flags are usally sets in the -trackMouse:inRect:ofView:untilMouseUp: method

*/ - (int) mouseDownFlags { return _mouse_down_flags; } /**

Gets the NSCell's delay and the interval parameters used when NSCell sends continouly action messages. The NSCell implementation sets both delay and interval to 0.1.

See Also: -trackMouse:inRect:ofView:untilMouseUp:

*/ - (void) getPeriodicDelay: (float*)delay interval: (float*)interval { *delay = 0.1; *interval = 0.1; } /**

Returns whether tracking starts. The NSCell implementation returns YES when the startPoint is into the control view retangle, NO otherwise. This method is call at the early stage of -trackMouse:inRect:ofView:untilMouseUp:

See Also: [NSView-mouse:inRect:] -trackMouse:inRect:ofView:untilMouseUp:

*/ - (BOOL) startTrackingAt: (NSPoint)startPoint inView: (NSView*)controlView { // If the point is in the view then yes start tracking if ([controlView mouse: startPoint inRect: [controlView bounds]]) return YES; else return NO; } /**

TODO

*/ - (void) stopTracking: (NSPoint)lastPoint at: (NSPoint)stopPoint inView: (NSView*)controlView mouseIsUp: (BOOL)flag { } - (BOOL) trackMouse: (NSEvent*)theEvent inRect: (NSRect)cellFrame ofView: (NSView*)controlView untilMouseUp: (BOOL)flag { NSApplication *theApp = [NSApplication sharedApplication]; unsigned event_mask = NSLeftMouseDownMask | NSLeftMouseUpMask | NSMouseMovedMask | NSLeftMouseDraggedMask | NSOtherMouseDraggedMask | NSRightMouseDraggedMask; NSPoint location = [theEvent locationInWindow]; NSPoint point = [controlView convertPoint: location fromView: nil]; float delay; float interval; id target = [self target]; SEL action = [self action]; NSPoint last_point = point; BOOL done; BOOL mouseWentUp; unsigned periodCount = 0; NSDebugLLog(@"NSCell", @"cell start tracking in rect %@ initial point %f %f", NSStringFromRect(cellFrame), point.x, point.y); _mouse_down_flags = [theEvent modifierFlags]; if (![self startTrackingAt: point inView: controlView]) return NO; if (![controlView mouse: point inRect: cellFrame]) return NO; // point is not in cell if ((_action_mask & NSLeftMouseDownMask) && [theEvent type] == NSLeftMouseDown) [(NSControl*)controlView sendAction: action to: target]; if (_action_mask & NSPeriodicMask) { [self getPeriodicDelay: &delay interval: &interval]; [NSEvent startPeriodicEventsAfterDelay: delay withPeriod: interval]; event_mask |= NSPeriodicMask; } NSDebugLLog(@"NSCell", @"cell get mouse events\n"); mouseWentUp = NO; done = NO; if (theEvent != [NSApp currentEvent]) theEvent = [NSApp currentEvent]; else theEvent = [theApp nextEventMatchingMask: event_mask untilDate: nil inMode: NSEventTrackingRunLoopMode dequeue: YES]; while (!done) { NSEventType eventType; BOOL pointIsInCell; eventType = [theEvent type]; if (eventType != NSPeriodic || periodCount == 4) { last_point = point; if (eventType == NSPeriodic) { NSWindow *w = [controlView window]; /* * Too many periodic events in succession - * update the mouse location and reset the counter. */ location = [w mouseLocationOutsideOfEventStream]; periodCount = 0; } else { location = [theEvent locationInWindow]; } point = [controlView convertPoint: location fromView: nil]; NSDebugLLog(@"NSCell", @"location %f %f\n", location.x, location.y); NSDebugLLog(@"NSCell", @"point %f %f\n", point.x, point.y); } else { periodCount++; NSDebugLLog (@"NSCell", @"cell got a periodic event"); } if (![controlView mouse: point inRect: cellFrame]) { NSDebugLLog(@"NSCell", @"point not in cell frame\n"); pointIsInCell = NO; if (flag == NO) { NSDebugLLog(@"NSCell", @"cell return immediately\n"); done = YES; } } else { pointIsInCell = YES; } if (!done && ![self continueTracking: last_point // should continue at: point // tracking? inView: controlView]) { NSDebugLLog(@"NSCell", @"cell stop tracking\n"); done = YES; } // Did the mouse go up? if (eventType == NSLeftMouseUp) { NSDebugLLog(@"NSCell", @"cell mouse went up\n"); mouseWentUp = YES; done = YES; } else { if (pointIsInCell && ((eventType == NSLeftMouseDragged && (_action_mask & NSLeftMouseDraggedMask)) || ((eventType == NSPeriodic) && (_action_mask & NSPeriodicMask)))) [(NSControl*)controlView sendAction: action to: target]; } if (!done) theEvent = [theApp nextEventMatchingMask: event_mask untilDate: nil inMode: NSEventTrackingRunLoopMode dequeue: YES]; } // Hook called when stop tracking [self stopTracking: last_point at: point inView: controlView mouseIsUp: mouseWentUp]; if (_action_mask & NSPeriodicMask) [NSEvent stopPeriodicEvents]; if (mouseWentUp) { [self setNextState]; if ((_action_mask & NSLeftMouseUpMask)) [(NSControl*)controlView sendAction: action to: target]; } // Return YES only if the mouse went up within the cell if (mouseWentUp && (flag || [controlView mouse: point inRect: cellFrame])) { NSDebugLLog(@"NSCell", @"mouse went up in cell\n"); return YES; } NSDebugLLog(@"NSCell", @"mouse did not go up in cell\n"); return NO; // Otherwise return NO } /**

TODO

*/ - (void) resetCursorRect: (NSRect)cellFrame inView: (NSView*)controlView { if (_cell.type == NSTextCellType && _cell.is_disabled == NO && (_cell.is_selectable == YES || _cell.is_editable == YES)) { static NSCursor *cursor = nil; NSRect rect; if (cursor== nil) { cursor = RETAIN([NSCursor IBeamCursor]); } rect = NSIntersectionRect(cellFrame, [controlView visibleRect]); /* * Here we depend on an undocumented feature of NSCursor which may or * may not exist in OPENSTEP or MacOS-X ... * If we add a cursor rect to a view and don't set it to be set on * either entry to or exit from the view, we push it on entry and * pop it from the cursor stack on exit. */ [controlView addCursorRect: rect cursor: cursor]; } } /**

Implemented by subclasses to returns the key equivalent. The NSCell implementation returns an empty NSString.

*/ - (NSString*) keyEquivalent { return @""; } /**

Does nothing. This method is used by subclasses to recalculate sizes

It is usally called from a NSControl object

See Also: [NSControl-calcSize]

*/ - (void) calcDrawInfo: (NSRect)aRect { } /**Returns the minimun size needed to display the NSCell. This size is calculate by adding : the borders (plain or bezeled) size the spacing between the border and inside the cell the TODO ... if the cell is type of NSTextCellType or the image size if the cell has a NSImageCellType type.

This method returns NSZeroSize if the cell has a NSNullCellType type (Cocoa returns a very big size instead).

*/ - (NSSize) cellSize { NSSize borderSize, s; // Get border size if (_cell.is_bordered) borderSize = _sizeForBorderType (NSLineBorder); else if (_cell.is_bezeled) borderSize = _sizeForBorderType (NSBezelBorder); else borderSize = NSZeroSize; // Add spacing between border and inside if (_cell.is_bordered || _cell.is_bezeled) { borderSize.height += 1; borderSize.width += 3; } // Get Content Size switch (_cell.type) { case NSTextCellType: { NSAttributedString *attrStr; attrStr = [self attributedStringValue]; if ([attrStr length] != 0) { s = [attrStr size]; } else { s = [self _sizeText: @"A"]; } } break; case NSImageCellType: if (_cell_image == nil) { s = NSZeroSize; } else { s = [_cell_image size]; } break; default: case NSNullCellType: // macosx instead returns a 'very big size' here; we return NSZeroSize s = NSZeroSize; break; } // Add in border size s.width += 2 * borderSize.width; s.height += 2 * borderSize.height; return s; } /**

TODO. Currently the GNUstep implementation returns -cellSize

See Also: -cellSize

*/ - (NSSize) cellSizeForBounds: (NSRect)aRect { if (_cell.type == NSTextCellType) { // TODO: Resize the text to fit } return [self cellSize]; } /**

TODO

*/ - (NSRect) drawingRectForBounds: (NSRect)theRect { NSSize borderSize; // Get border size if (_cell.is_bordered) borderSize = _sizeForBorderType (NSLineBorder); else if (_cell.is_bezeled) borderSize = _sizeForBorderType (NSBezelBorder); else borderSize = NSZeroSize; return NSInsetRect(theRect, borderSize.width, borderSize.height); } /**

The GNUstep implementation returns -drawingRectForBounds:

*/ - (NSRect) imageRectForBounds: (NSRect)theRect { return [self drawingRectForBounds: theRect]; } /**

TODO

*/ - (NSRect) titleRectForBounds: (NSRect)theRect { if (_cell.type == NSTextCellType) { NSRect frame = [self drawingRectForBounds: theRect]; // Add spacing between border and inside if (_cell.is_bordered || _cell.is_bezeled) { frame.origin.x += 3; frame.size.width -= 6; frame.origin.y += 1; frame.size.height -= 2; } return frame; } else { return theRect; } } - (void) setControlSize: (NSControlSize)controlSize { _cell.control_size = controlSize; } - (NSControlSize) controlSize { return _cell.control_size; } - (void) setControlTint: (NSControlTint)controlTint { _cell.control_tint = controlTint; } - (NSControlTint) controlTint { return _cell.control_tint; } /**

This method is used by subclasses to get the control view. This method returns nil.

*/ - (NSView*) controlView { return nil; } /**

This method is used by subclasses to specify the control view.

*/ - (void) setControlView: (NSView*)view { // Do nothing } /**

This drawing is minimal and with no background, * to make it easier for subclass to customize drawing.

*/ - (void) drawInteriorWithFrame: (NSRect)cellFrame inView: (NSView*)controlView { cellFrame = [self drawingRectForBounds: cellFrame]; //FIXME: Check if this is also neccessary for images, // Add spacing between border and inside if (_cell.is_bordered || _cell.is_bezeled) { cellFrame.origin.x += 3; cellFrame.size.width -= 6; cellFrame.origin.y += 1; cellFrame.size.height -= 2; } switch (_cell.type) { case NSTextCellType: [self _drawAttributedText: [self _drawAttributedString] inFrame: cellFrame]; break; case NSImageCellType: if (_cell_image) { NSSize size; NSPoint position; size = [_cell_image size]; position.x = MAX(NSMidX(cellFrame) - (size.width/2.),0.); position.y = MAX(NSMidY(cellFrame) - (size.height/2.),0.); /* * Images are always drawn with their bottom-left corner * at the origin so we must adjust the position to take * account of a flipped view. */ if ([controlView isFlipped]) position.y += size.height; [_cell_image compositeToPoint: position operation: NSCompositeSourceOver]; } break; case NSNullCellType: break; } // NB: We don't do any highlighting to make it easier for subclasses // to reuse this code while doing their own custom highlighting and // prettyfying } /**

Draws the cell in controlView

*/ - (void) drawWithFrame: (NSRect)cellFrame inView: (NSView*)controlView { // do nothing if cell's frame rect is zero if (NSIsEmptyRect(cellFrame)) return; // draw the border if needed [self _drawBorderAndBackgroundWithFrame: cellFrame inView: controlView]; // draw interior [self drawInteriorWithFrame: cellFrame inView: controlView]; // Draw first responder [self _drawFocusRingWithFrame: cellFrame inView: controlView]; } /**

Sets whether the NSCell is highlighted.

See Also: -isHighlighted

*/ - (void) setHighlighted: (BOOL) flag { _cell.is_highlighted = flag; } /**

Returns whether the cell is highlighted. By default NO

See Also: -setHighlighted:

*/ - (BOOL) isHighlighted { return _cell.is_highlighted; } /** *

TODO explain

*/ - (void) highlight: (BOOL)lit withFrame: (NSRect)cellFrame inView: (NSView*)controlView { if (_cell.is_highlighted != lit) { _cell.is_highlighted = lit; /* * NB: This has a visible effect only if subclasses override * drawWithFrame:inView: to draw something special when the * cell is highlighted. * NSCell simply draws border+text/image and makes no highlighting, * for easier subclassing. */ if ([self isOpaque] == NO) { /* FIXME - This looks like potentially generating an * infinite loop! The control asking the cell to draw * itself in the rect, the cell asking the control to draw * the rect, the control asking the cell to draw itself in * the rect, the cell ... * * I think we should remove it. The control is responsible * for using the cell to draw, not vice versa. */ [controlView displayRect: cellFrame]; } [self drawWithFrame: cellFrame inView: controlView]; } } - (NSColor*) highlightColorWithFrame: (NSRect)cellFrame inView: (NSView *)controlView { return [NSColor selectedControlColor]; } - (NSText*) setUpFieldEditorAttributes: (NSText*)textObject { // Reset the string to have a well defined state. The real string gets set later on. [textObject setString: @""]; [textObject setTextColor: [self textColor]]; if ([self isBezeled]) { [textObject setBackgroundColor: [NSColor textBackgroundColor]]; [textObject setDrawsBackground: YES]; } else { [textObject setDrawsBackground: NO]; } [textObject setFont: [self font]]; [textObject setAlignment: [self alignment]]; // FIXME: Add base writing direction [textObject setEditable: [self isEditable]]; [textObject setSelectable: [self isSelectable]]; [textObject setRichText: [self allowsEditingTextAttributes]]; [textObject setImportsGraphics: [self importsGraphics]]; [(NSTextView*)textObject setAllowsUndo: [self allowsUndo]]; return textObject; } - (void) _setupTextWithFrame: (NSRect)aRect inView: (NSView*)controlView editor: (NSText*)textObject delegate: (id)anObject range: (NSRange)selection { NSRect titleRect = [self titleRectForBounds: aRect]; NSClipView *cv = [[NSClipView alloc] initWithFrame: titleRect]; NSTextContainer *ct = [(NSTextView*)textObject textContainer]; NSRect maxRect; // A clip view should is only created for scrollable text if ([self isScrollable]) { /* See comments in NSStringDrawing.m about the choice of maximum size. */ maxRect = NSMakeRect(0, 0, 1e6, titleRect.size.height); } else { maxRect = NSMakeRect(0, 0, titleRect.size.width, titleRect.size.height); } [controlView addSubview: cv]; RELEASE(cv); [cv setAutoresizesSubviews: NO]; [cv setDocumentView: textObject]; [ct setContainerSize: maxRect.size]; [ct setHeightTracksTextView: NO]; [ct setWidthTracksTextView: NO]; [textObject setFrame: maxRect]; [textObject setHorizontallyResizable: NO]; [textObject setVerticallyResizable: NO]; [textObject setMaxSize: maxRect.size]; [textObject setMinSize: titleRect.size]; if (_formatter != nil) { NSString *contents; contents = [_formatter editingStringForObjectValue: _object_value]; if (contents == nil) { contents = _contents; } [textObject setText: contents]; } else { if (_cell.contents_is_attributed_string == NO) { [textObject setText: _contents]; } else { // The curent text has size 0, so this replaces the whole text. [textObject replaceCharactersInRange: NSMakeRange(0, 0) withAttributedString: (NSAttributedString *)_contents]; } } [textObject sizeToFit]; [textObject setSelectedRange: selection]; [textObject scrollRangeToVisible: selection]; [textObject setDelegate: anObject]; [[controlView window] makeFirstResponder: textObject]; _cell.in_editing = YES; } /**

Ends any text editing. This method sets the text object's delegate to nil, and remove the NSClipView and the text object used for editing

See Also: -editWithFrame:inView:editor:delegate:event:

*/ - (void) endEditing: (NSText*)textObject { NSClipView *clipView; _cell.in_editing = NO; [textObject setString: @""]; [textObject setDelegate: nil]; clipView = (NSClipView*)[textObject superview]; [textObject removeFromSuperview]; [clipView removeFromSuperview]; } /* * Editing Text */ /**

.This method does nothing if a the controlView is nil, if text object does not exist or if the cell's type is not NSTextCellType

*/ - (void) editWithFrame: (NSRect)aRect inView: (NSView*)controlView editor: (NSText*)textObject delegate: (id)anObject event: (NSEvent*)theEvent { if (!controlView || !textObject || (_cell.type != NSTextCellType)) return; [self _setupTextWithFrame: aRect inView: controlView editor: textObject delegate: anObject range: NSMakeRange(0, 0)]; if ([theEvent type] == NSLeftMouseDown) { [textObject mouseDown: theEvent]; } } /**

This method does nothing if the controlView is nil, if text object does not exist or if the cell's type is not NSTextCellType

*/ - (void) selectWithFrame: (NSRect)aRect inView: (NSView*)controlView editor: (NSText*)textObject delegate: (id)anObject start: (int)selStart length: (int)selLength { if (!controlView || !textObject || (_cell.type != NSTextCellType)) return; [self _setupTextWithFrame: aRect inView: controlView editor: textObject delegate: anObject range: NSMakeRange(selStart, selLength)]; } - (BOOL) sendsActionOnEndEditing { return _cell.sends_action_on_end_editing; } - (void) setSendsActionOnEndEditing: (BOOL)flag { _cell.sends_action_on_end_editing = flag; } - (BOOL) allowsUndo { return _cell.allows_undo; } - (void) setAllowsUndo: (BOOL)flag { _cell.allows_undo = flag; } /* * Copying */ - (id) copyWithZone: (NSZone*)zone { NSCell *c = (NSCell*)NSCopyObject (self, 0, zone); /* Hmmm. */ c->_contents = [_contents copyWithZone: zone]; /* Because of performance issues (and because so the doc says) only pointers to the objects are copied. We need to RETAIN them all though. */ TEST_RETAIN (_font); TEST_RETAIN (_object_value); TEST_RETAIN (_menu); TEST_RETAIN (_cell_image); TEST_RETAIN (_formatter); TEST_RETAIN (_represented_object); return c; } /* * NSCoding protocol */ - (void) encodeWithCoder: (NSCoder*)aCoder { if ([aCoder allowsKeyedCoding]) { unsigned long cFlags = 0; unsigned int cFlags2 = 0; id contents; // encode contents if (_cell.type == NSTextCellType) { contents = _contents; } else if (_cell.type == NSImageCellType) { contents = _cell_image; } else { contents = [self objectValue]; } [aCoder encodeObject: contents forKey: @"NSContents"]; // flags cFlags |= [self focusRingType]; cFlags |= [self showsFirstResponder] ? 0x4 : 0; cFlags |= (_action_mask & NSLeftMouseUpMask) ? 0 : 0x20; cFlags |= [self wraps] ? 0x40 : 0; cFlags |= (_action_mask & NSLeftMouseDraggedMask) ? 0x100 : 0; cFlags |= ([self lineBreakMode] << 12); cFlags |= (_action_mask & NSLeftMouseDownMask) ? 0x40000 : 0; cFlags |= [self isContinuous] ? 0x80000 : 0; cFlags |= [self isScrollable] ? 0x100000 : 0; cFlags |= [self isSelectable] ? 0x200000 : 0; cFlags |= [self isBezeled] ? 0x400000 : 0; cFlags |= [self isBordered] ? 0x800000 : 0; cFlags |= ([self type] << 26); cFlags |= [self isEditable] ? 0x10000000 : 0; cFlags |= ([self isEnabled] == NO) ? 0x20000000 : 0; cFlags |= [self isHighlighted] ? 0x40000000 : 0; cFlags |= ([self state] == NSOnState) ? 0x80000000 : 0; [aCoder encodeInt: cFlags forKey: @"NSCellFlags"]; // flags part 2 cFlags2 |= ([self controlTint] << 5); cFlags2 |= ([self controlSize] << 17); cFlags2 |= [self sendsActionOnEndEditing] ? 0x400000 : 0; cFlags2 |= [self allowsMixedState] ? 0x1000000 : 0; cFlags2 |= [self refusesFirstResponder] ? 0x2000000 : 0; cFlags2 |= ([self alignment] << 26); cFlags2 |= [self importsGraphics] ? 0x20000000 : 0; cFlags2 |= [self allowsEditingTextAttributes] ? 0x40000000 : 0; [aCoder encodeInt: cFlags2 forKey: @"NSCellFlags2"]; if (_cell.type == NSTextCellType) { // font and formatter. if ([self font]) { [aCoder encodeObject: [self font] forKey: @"NSSupport"]; } if ([self formatter]) { [aCoder encodeObject: [self formatter] forKey: @"NSFormatter"]; } } else if ([self image]) { [aCoder encodeObject: [self image] forKey: @"NSSupport"]; } } else { BOOL flag; unsigned int tmp_int; [aCoder encodeObject: _contents]; [aCoder encodeObject: _cell_image]; [aCoder encodeObject: _font]; [aCoder encodeObject: _object_value]; flag = _cell.contents_is_attributed_string; [aCoder encodeValueOfObjCType: @encode(BOOL) at: &flag]; flag = _cell.is_highlighted; [aCoder encodeValueOfObjCType: @encode(BOOL) at: &flag]; flag = _cell.is_disabled; [aCoder encodeValueOfObjCType: @encode(BOOL) at: &flag]; flag = _cell.is_editable; [aCoder encodeValueOfObjCType: @encode(BOOL) at: &flag]; flag = _cell.is_rich_text; [aCoder encodeValueOfObjCType: @encode(BOOL) at: &flag]; flag = _cell.imports_graphics; [aCoder encodeValueOfObjCType: @encode(BOOL) at: &flag]; flag = _cell.shows_first_responder; [aCoder encodeValueOfObjCType: @encode(BOOL) at: &flag]; flag = _cell.refuses_first_responder; [aCoder encodeValueOfObjCType: @encode(BOOL) at: &flag]; flag = _cell.sends_action_on_end_editing; [aCoder encodeValueOfObjCType: @encode(BOOL) at: &flag]; flag = _cell.is_bordered; [aCoder encodeValueOfObjCType: @encode(BOOL) at: &flag]; flag = _cell.is_bezeled; [aCoder encodeValueOfObjCType: @encode(BOOL) at: &flag]; flag = _cell.is_scrollable; [aCoder encodeValueOfObjCType: @encode(BOOL) at: &flag]; flag = _cell.is_selectable; [aCoder encodeValueOfObjCType: @encode(BOOL) at: &flag]; // This used to be is_continuous, which has been replaced. /* Ayers 20.03.2003: But we must continue to encode it for backward compatibility or current releases will have undefined behavior when decoding archives (i.e. .gorm files) encoded by this version. */ flag = [self isContinuous]; [aCoder encodeValueOfObjCType: @encode(BOOL) at: &flag]; flag = _cell.allows_mixed_state; [aCoder encodeValueOfObjCType: @encode(BOOL) at: &flag]; flag = [self wraps]; [aCoder encodeValueOfObjCType: @encode(BOOL) at: &flag]; tmp_int = _cell.text_align; [aCoder encodeValueOfObjCType: @encode(unsigned int) at: &tmp_int]; tmp_int = _cell.type; [aCoder encodeValueOfObjCType: @encode(unsigned int) at: &tmp_int]; tmp_int = _cell.image_position; [aCoder encodeValueOfObjCType: @encode(unsigned int) at: &tmp_int]; tmp_int = _cell.entry_type; [aCoder encodeValueOfObjCType: @encode(unsigned int) at: &tmp_int]; // FIXME: State may be -1, why do we encode it as unsigned? tmp_int = _cell.state; [aCoder encodeValueOfObjCType: @encode(unsigned int) at: &tmp_int]; tmp_int = _cell.mnemonic_location; [aCoder encodeValueOfObjCType: @encode(unsigned int) at: &tmp_int]; [aCoder encodeValueOfObjCType: @encode(unsigned int) at: &_mouse_down_flags]; [aCoder encodeValueOfObjCType: @encode(unsigned int) at: &_action_mask]; [aCoder encodeValueOfObjCType: @encode(id) at: &_formatter]; [aCoder encodeValueOfObjCType: @encode(id) at: &_menu]; [aCoder encodeValueOfObjCType: @encode(id) at: &_represented_object]; tmp_int = _cell.allows_undo; [aCoder encodeValueOfObjCType: @encode(unsigned int) at: &tmp_int]; tmp_int = _cell.line_break_mode; [aCoder encodeValueOfObjCType: @encode(unsigned int) at: &tmp_int]; tmp_int = _cell.control_tint; [aCoder encodeValueOfObjCType: @encode(unsigned int) at: &tmp_int]; tmp_int = _cell.control_size; [aCoder encodeValueOfObjCType: @encode(unsigned int) at: &tmp_int]; tmp_int = _cell.focus_ring_type; [aCoder encodeValueOfObjCType: @encode(unsigned int) at: &tmp_int]; tmp_int = _cell.base_writing_direction; [aCoder encodeValueOfObjCType: @encode(unsigned int) at: &tmp_int]; } } - (id) initWithCoder: (NSCoder*)aDecoder { if ([aDecoder allowsKeyedCoding]) { id contents = [aDecoder decodeObjectForKey: @"NSContents"]; // initialize based on content... if ([contents isKindOfClass: [NSString class]]) { self = [self initTextCell: contents]; } else if ([contents isKindOfClass: [NSImage class]]) { self = [self initImageCell: contents]; } else { self = [self init]; [self setObjectValue: contents]; } if ([aDecoder containsValueForKey: @"NSCellFlags"]) { unsigned long cFlags; unsigned long mask = 0; cFlags = [aDecoder decodeIntForKey: @"NSCellFlags"]; [self setFocusRingType: (cFlags & 0x3)]; [self setShowsFirstResponder: ((cFlags & 0x4) == 0x4)]; // This bit flag is the other way around! if ((cFlags & 0x20) != 0x20) mask |= NSLeftMouseUpMask; // This bit flag is the other way around! [self setWraps: ((cFlags & 0x40) != 0x40)]; if ((cFlags & 0x100) == 0x100) mask |= NSLeftMouseDraggedMask; [self setLineBreakMode: ((cFlags & 0x7000) >> 12)]; if ((cFlags & 0x40000) == 0x40000) mask |= NSLeftMouseDownMask; if ((cFlags & 0x80000) == 0x80000) mask |= NSPeriodicMask; [self sendActionOn: mask]; [self setScrollable: ((cFlags & 0x100000) == 0x100000)]; [self setSelectable: ((cFlags & 0x200000) == 0x200000)]; [self setBezeled: ((cFlags & 0x400000) == 0x400000)]; [self setBordered: ((cFlags & 0x800000) == 0x800000)]; [self setType: ((cFlags & 0xC000000) >> 26)]; [self setEditable: ((cFlags & 0x10000000) == 0x10000000)]; // This bit flag is the other way around! [self setEnabled: ((cFlags & 0x20000000) != 0x20000000)]; [self setHighlighted: ((cFlags & 0x40000000) == 0x40000000)]; [self setState: ((cFlags & 0x80000000) == 0x80000000) ? NSOnState : NSOffState]; } if ([aDecoder containsValueForKey: @"NSCellFlags2"]) { int cFlags2; cFlags2 = [aDecoder decodeIntForKey: @"NSCellFlags2"]; [self setControlTint: ((cFlags2 & 0xE0) >> 5)]; [self setControlSize: ((cFlags2 & 0xE0000) >> 17)]; [self setSendsActionOnEndEditing: ((cFlags2 & 0x400000) == 0x400000)]; [self setAllowsMixedState: ((cFlags2 & 0x1000000) == 0x1000000)]; [self setRefusesFirstResponder: ((cFlags2 & 0x2000000) == 0x2000000)]; [self setAlignment: ((cFlags2 & 0x1C000000) >> 26)]; [self setImportsGraphics: ((cFlags2 & 0x20000000) == 0x20000000)]; [self setAllowsEditingTextAttributes: ((cFlags2 & 0x40000000) == 0x40000000)]; } if ([aDecoder containsValueForKey: @"NSSupport"]) { id support = [aDecoder decodeObjectForKey: @"NSSupport"]; if ([support isKindOfClass: [NSFont class]]) { [self setFont: support]; } else if ([support isKindOfClass: [NSImage class]]) { [self setImage: support]; } } if ([aDecoder containsValueForKey: @"NSFormatter"]) { NSFormatter *formatter = [aDecoder decodeObjectForKey: @"NSFormatter"]; [self setFormatter: formatter]; } } else { BOOL flag; unsigned int tmp_int; id formatter, menu; int version = [aDecoder versionForClassName: @"NSCell"]; [aDecoder decodeValueOfObjCType: @encode(id) at: &_contents]; [aDecoder decodeValueOfObjCType: @encode(id) at: &_cell_image]; [aDecoder decodeValueOfObjCType: @encode(id) at: &_font]; [aDecoder decodeValueOfObjCType: @encode(id) at: &_object_value]; [aDecoder decodeValueOfObjCType: @encode(BOOL) at: &flag]; _cell.contents_is_attributed_string = flag; [aDecoder decodeValueOfObjCType: @encode(BOOL) at: &flag]; _cell.is_highlighted = flag; [aDecoder decodeValueOfObjCType: @encode(BOOL) at: &flag]; _cell.is_disabled = flag; [aDecoder decodeValueOfObjCType: @encode(BOOL) at: &flag]; _cell.is_editable = flag; [aDecoder decodeValueOfObjCType: @encode(BOOL) at: &flag]; _cell.is_rich_text = flag; [aDecoder decodeValueOfObjCType: @encode(BOOL) at: &flag]; _cell.imports_graphics = flag; [aDecoder decodeValueOfObjCType: @encode(BOOL) at: &flag]; _cell.shows_first_responder = flag; [aDecoder decodeValueOfObjCType: @encode(BOOL) at: &flag]; _cell.refuses_first_responder = flag; [aDecoder decodeValueOfObjCType: @encode(BOOL) at: &flag]; _cell.sends_action_on_end_editing = flag; [aDecoder decodeValueOfObjCType: @encode(BOOL) at: &flag]; _cell.is_bordered = flag; [aDecoder decodeValueOfObjCType: @encode(BOOL) at: &flag]; _cell.is_bezeled = flag; [aDecoder decodeValueOfObjCType: @encode(BOOL) at: &flag]; _cell.is_scrollable = flag; [aDecoder decodeValueOfObjCType: @encode(BOOL) at: &flag]; _cell.is_selectable = flag; [aDecoder decodeValueOfObjCType: @encode(BOOL) at: &flag]; // This used to be is_continuous, which has been replaced. //_cell.is_continuous = flag; [aDecoder decodeValueOfObjCType: @encode(BOOL) at: &flag]; _cell.allows_mixed_state = flag; [aDecoder decodeValueOfObjCType: @encode(BOOL) at: &flag]; _cell.wraps = flag; [aDecoder decodeValueOfObjCType: @encode(unsigned int) at: &tmp_int]; _cell.text_align = tmp_int; [aDecoder decodeValueOfObjCType: @encode(unsigned int) at: &tmp_int]; _cell.type = tmp_int; [aDecoder decodeValueOfObjCType: @encode(unsigned int) at: &tmp_int]; _cell.image_position = tmp_int; [aDecoder decodeValueOfObjCType: @encode(unsigned int) at: &tmp_int]; _cell.entry_type = tmp_int; [aDecoder decodeValueOfObjCType: @encode(unsigned int) at: &tmp_int]; _cell.state = tmp_int; [aDecoder decodeValueOfObjCType: @encode(unsigned int) at: &tmp_int]; _cell.mnemonic_location = tmp_int; [aDecoder decodeValueOfObjCType: @encode(unsigned int) at: &_mouse_down_flags]; [aDecoder decodeValueOfObjCType: @encode(unsigned int) at: &_action_mask]; if (version < 3) { unsigned int mask = 0; // Convert old GNUstep mask value to Cocoa values if ((_action_mask & 0x1) == 0x1) { mask |= NSLeftMouseDownMask; } if ((_action_mask & 0x2) == 0x2) { mask |= NSLeftMouseUpMask; } if ((_action_mask & 0x4) == 0x4) { mask |= NSOtherMouseDownMask; } if ((_action_mask & 0x8) == 0x8) { mask |= NSOtherMouseUpMask; } if ((_action_mask & 0x10) == 0x10) { mask |= NSRightMouseDownMask; } if ((_action_mask & 0x20) == 0x20) { mask |= NSRightMouseUpMask; } if ((_action_mask & 0x40) == 0x40) { mask |= NSMouseMovedMask; } if ((_action_mask & 0x80) == 0x80) { mask |= NSLeftMouseDraggedMask; } if ((_action_mask & 0x100) == 0x100) { mask |= NSOtherMouseDraggedMask; } if ((_action_mask & 0x200) == 0x200) { mask |= NSRightMouseDraggedMask; } if ((_action_mask & 0x400) == 0x400) { mask |= NSMouseEnteredMask; } if ((_action_mask & 0x800) == 0x800) { mask |= NSMouseExitedMask; } if ((_action_mask & 0x1000) == 0x1000) { mask |= NSKeyDownMask; } if ((_action_mask & 0x2000) == 0x2000) { mask |= NSKeyUpMask; } if ((_action_mask & 0x4000) == 0x4000) { mask |= NSFlagsChangedMask; } if ((_action_mask & 0x8000) == 0x8000) { mask |= NSAppKitDefinedMask; } if ((_action_mask & 0x10000) == 0x10000) { mask |= NSSystemDefinedMask; } if ((_action_mask & 0x20000) == 0x20000) { mask |= NSApplicationDefinedMask; } if ((_action_mask & 0x40000) == 0x40000) { mask |= NSPeriodicMask; } if ((_action_mask & 0x80000) == 0x80000) { mask |= NSCursorUpdateMask; } if ((_action_mask & 0x100000) == 0x100000) { mask |= NSScrollWheelMask; } _action_mask = mask; } _action_mask = NSLeftMouseUpMask; [aDecoder decodeValueOfObjCType: @encode(id) at: &formatter]; [self setFormatter: formatter]; [aDecoder decodeValueOfObjCType: @encode(id) at: &menu]; [self setMenu: menu]; [aDecoder decodeValueOfObjCType: @encode(id) at: &_represented_object]; if (_formatter != nil) { NSString *contents; contents = [_formatter stringForObjectValue: _object_value]; if (contents != nil) { _cell.has_valid_object_value = YES; ASSIGN (_contents, contents); _cell.contents_is_attributed_string = NO; } } if (version >= 2) { [aDecoder decodeValueOfObjCType: @encode(unsigned int) at: &tmp_int]; _cell.allows_undo = tmp_int; [aDecoder decodeValueOfObjCType: @encode(unsigned int) at: &tmp_int]; _cell.line_break_mode = tmp_int; [aDecoder decodeValueOfObjCType: @encode(unsigned int) at: &tmp_int]; _cell.control_tint = tmp_int; [aDecoder decodeValueOfObjCType: @encode(unsigned int) at: &tmp_int]; _cell.control_size = tmp_int; [aDecoder decodeValueOfObjCType: @encode(unsigned int) at: &tmp_int]; _cell.focus_ring_type = tmp_int; [aDecoder decodeValueOfObjCType: @encode(unsigned int) at: &tmp_int]; _cell.base_writing_direction = tmp_int; } } return self; } @end @implementation NSCell (PrivateMethods) - (NSColor*) textColor { if (_cell.is_disabled) return dtxtCol; else return txtCol; } /* This method is an exception and returns a non-autoreleased dictionary, so that calling methods can deallocate it immediately using release. Otherwise if many cells are drawn/their size computed, we pile up hundreds or thousands of these objects before they are deallocated at the end of the run loop. */ - (NSDictionary*) _nonAutoreleasedTypingAttributes { NSDictionary *attr; NSColor *color; NSMutableParagraphStyle *paragraphStyle; color = [self textColor]; /* Note: There are only a few possible paragraph styles for cells. TODO: Cache them and reuse them for the whole app lifetime. */ paragraphStyle = [[NSParagraphStyle defaultParagraphStyle] mutableCopy]; [paragraphStyle setLineBreakMode: [self lineBreakMode]]; [paragraphStyle setBaseWritingDirection: [self baseWritingDirection]]; [paragraphStyle setAlignment: [self alignment]]; attr = [[NSDictionary alloc] initWithObjectsAndKeys: _font, NSFontAttributeName, color, NSForegroundColorAttributeName, paragraphStyle, NSParagraphStyleAttributeName, nil]; RELEASE (paragraphStyle); return attr; } - (NSSize) _sizeText: (NSString*)title { NSSize size; NSDictionary *dict; if (title == nil) { return NSMakeSize (0,0); } dict = [self _nonAutoreleasedTypingAttributes]; size = [title sizeWithAttributes: dict]; RELEASE (dict); return size; } /** * Private internal method, returns an attributed string to display. */ - (NSAttributedString*) _drawAttributedString { if (!_cell.is_disabled) { return [self attributedStringValue]; } else { NSAttributedString *attrStr = [self attributedStringValue]; NSDictionary *attribs; NSMutableDictionary *newAttribs; attribs = [attrStr attributesAtIndex: 0 effectiveRange: NULL]; newAttribs = [NSMutableDictionary dictionaryWithDictionary: attribs]; [newAttribs setObject: [NSColor disabledControlTextColor] forKey: NSForegroundColorAttributeName]; return AUTORELEASE([[NSAttributedString alloc] initWithString: [attrStr string] attributes: newAttribs]); } } /** * Private internal method to display an attributed string. */ - (void) _drawAttributedText: (NSAttributedString*)aString inFrame: (NSRect)aRect { NSSize titleSize; if (aString == nil) return; titleSize = [aString size]; /** Important: text should always be vertically centered without * considering descender [as if descender did not exist]. * This is particularly important for single line texts. * Please make sure the output remains always correct. */ aRect.origin.y = NSMidY (aRect) - titleSize.height/2; aRect.size.height = titleSize.height; [aString drawInRect: aRect]; } - (void) _drawText: (NSString*)aString inFrame: (NSRect)cellFrame { NSSize titleSize; NSDictionary *attributes; if (aString == nil) return; attributes = [self _nonAutoreleasedTypingAttributes]; titleSize = [aString sizeWithAttributes: attributes]; /** Important: text should always be vertically centered without * considering descender [as if descender did not exist]. * This is particularly important for single line texts. * Please make sure the output remains always correct. */ cellFrame.origin.y = NSMidY (cellFrame) - titleSize.height/2; cellFrame.size.height = titleSize.height; [aString drawInRect: cellFrame withAttributes: attributes]; RELEASE (attributes); } // Private helper method overridden in subclasses - (void) _drawBorderAndBackgroundWithFrame: (NSRect)cellFrame inView: (NSView*)controlView { if (_cell.is_bordered) { [shadowCol set]; NSFrameRect(cellFrame); } else if (_cell.is_bezeled) { [[GSTheme theme] drawWhiteBezel: cellFrame withClip: NSZeroRect]; } } // Private helper method - (void) _drawFocusRingWithFrame: (NSRect)cellFrame inView: (NSView*)controlView { if (_cell.shows_first_responder && [[controlView window] firstResponder] == controlView) { switch (_cell.focus_ring_type) { case NSFocusRingTypeDefault: [[GSTheme theme] drawFocusFrame: [self drawingRectForBounds: cellFrame] view: controlView]; break; case NSFocusRingTypeExterior: [[GSTheme theme] drawFocusFrame: cellFrame view: controlView]; break; case NSFocusRingTypeNone: default: break; } } } - (BOOL) _sendsActionOn:(int)eventTypeMask { return (_action_mask & eventTypeMask); } @end /* * Global function which should go somewhere else */ inline NSSize _sizeForBorderType (NSBorderType aType) { // Returns the size of a border switch (aType) { case NSLineBorder: return NSMakeSize(1, 1); case NSGrooveBorder: case NSBezelBorder: return NSMakeSize(2, 2); case NSNoBorder: default: return NSZeroSize; } }