mirror of
https://github.com/gnustep/libs-gui.git
synced 2025-04-23 12:20:49 +00:00
continious update, pass on the changed text. * Source/GSXibElement.m: Correct the handling of boolean values in attributes. * Source/GSBindingHelpers.h, * Source/NSKeyValueBinding.m: Prevent recursion when reverseSetValue: is used. * Source/NSTextField.m (-textDidChange:): Remove empty code line.
1157 lines
29 KiB
Objective-C
1157 lines
29 KiB
Objective-C
/** <title>NSControl</title>
|
|
|
|
<abstract>The abstract control class</abstract>
|
|
|
|
Copyright (C) 1996 Free Software Foundation, Inc.
|
|
|
|
Author: Scott Christley <scottc@net-community.com>
|
|
Date: 1996
|
|
Author: Richard Frith-Macdonald <richard@brainstorm.co.uk>
|
|
Date: August 1998
|
|
|
|
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 <http://www.gnu.org/licenses/> or write to the
|
|
Free Software Foundation, 51 Franklin Street, Fifth Floor,
|
|
Boston, MA 02110-1301, USA.
|
|
*/
|
|
|
|
#include "config.h"
|
|
|
|
#import <Foundation/NSDebug.h>
|
|
#import <Foundation/NSException.h>
|
|
#import <Foundation/NSKeyValueCoding.h>
|
|
#import <Foundation/NSKeyValueObserving.h>
|
|
#import <Foundation/NSNotification.h>
|
|
#import <Foundation/NSValue.h>
|
|
#import "AppKit/NSActionCell.h"
|
|
#import "AppKit/NSApplication.h"
|
|
#import "AppKit/NSCell.h"
|
|
#import "AppKit/NSControl.h"
|
|
#import "AppKit/NSColor.h"
|
|
#import "AppKit/NSEvent.h"
|
|
#import "AppKit/NSFont.h"
|
|
#import "AppKit/NSFontManager.h"
|
|
#import "AppKit/NSKeyValueBinding.h"
|
|
#import "AppKit/NSTextStorage.h"
|
|
#import "AppKit/NSTextView.h"
|
|
#import "AppKit/NSWindow.h"
|
|
#import "GSBindingHelpers.h"
|
|
#import "NSViewPrivate.h"
|
|
#import "GSGuiPrivate.h"
|
|
|
|
/*
|
|
* Class variables
|
|
*/
|
|
static Class usedCellClass;
|
|
static Class cellClass;
|
|
static Class actionCellClass;
|
|
static NSNotificationCenter *nc;
|
|
|
|
/**<p>TODO Description</p>
|
|
*/
|
|
|
|
@implementation NSControl
|
|
|
|
/*
|
|
* Class methods
|
|
*/
|
|
+ (void) initialize
|
|
{
|
|
if (self == [NSControl class])
|
|
{
|
|
[self setVersion: 1];
|
|
cellClass = [NSCell class];
|
|
usedCellClass = cellClass;
|
|
actionCellClass = [NSActionCell class];
|
|
// Cache the notifiaction centre for editing notifications
|
|
nc = [NSNotificationCenter defaultCenter];
|
|
|
|
// expose bindings
|
|
[self exposeBinding: NSValueBinding];
|
|
[self exposeBinding: NSEnabledBinding];
|
|
[self exposeBinding: NSAlignmentBinding];
|
|
[self exposeBinding: NSFontBinding];
|
|
[self exposeBinding: NSFontNameBinding];
|
|
[self exposeBinding: NSFontSizeBinding];
|
|
}
|
|
}
|
|
|
|
/**<p> Returns the cell Class used by NSControl. Used by subclasses.</p>
|
|
<p>See Also: +setCellClass:</p>
|
|
*/
|
|
+ (Class) cellClass
|
|
{
|
|
return usedCellClass;
|
|
}
|
|
|
|
/**<p> Sets the cell Class used by NSControl to <var>factoryId</var>.
|
|
Used by subclasses.</p> <p>See Also: +setCellClass:</p>
|
|
*/
|
|
+ (void) setCellClass: (Class)factoryId
|
|
{
|
|
usedCellClass = factoryId ? factoryId : cellClass;
|
|
}
|
|
|
|
/**<p>Initializes and returns a new NSControl into the rectangle
|
|
frameRect and create a new associated NSCell</p><p>See Also: -setCell:</p>
|
|
*/
|
|
- (id) initWithFrame: (NSRect)frameRect
|
|
{
|
|
NSCell *cell = [[[self class] cellClass] new];
|
|
|
|
[super initWithFrame: frameRect];
|
|
[self setCell: cell];
|
|
RELEASE(cell);
|
|
//_tag = 0;
|
|
|
|
return self;
|
|
}
|
|
|
|
- (void) dealloc
|
|
{
|
|
RELEASE(_cell);
|
|
[super dealloc];
|
|
}
|
|
|
|
/** <p>Returns the NSControl's cell.</p><p>See Also: -setCell:</p>
|
|
*/
|
|
- (id) cell
|
|
{
|
|
return _cell;
|
|
}
|
|
|
|
/** <p>Sets the NSControl's cell to aCell,
|
|
Raises an NSInvalidArgumentException exception if aCell is not nil and
|
|
if it is not a cell class.</p><p>See Also: -cell</p>
|
|
*/
|
|
- (void) setCell: (NSCell *)aCell
|
|
{
|
|
if (aCell != nil && [aCell isKindOfClass: cellClass] == NO)
|
|
[NSException raise: NSInvalidArgumentException
|
|
format: @"attempt to set non-cell object for control cell"];
|
|
|
|
ASSIGN(_cell, aCell);
|
|
}
|
|
|
|
/**<p>Returns whether the selected cell of the NSControl is enabled.</p>
|
|
<p>See Also: -setEnabled: [NSCell-isEnabled]</p>
|
|
*/
|
|
- (BOOL) isEnabled
|
|
{
|
|
return [[self selectedCell] isEnabled];
|
|
}
|
|
|
|
/**<p>Sets whether the NSControl's selected cell is enabled.
|
|
If flag is NO, this method abort the editing. This method marks self for
|
|
display.</p><p>See Also: -isEnabled [NSCell-setEnabled:]</p>
|
|
*/
|
|
- (void) setEnabled: (BOOL)flag
|
|
{
|
|
[[self selectedCell] setEnabled: flag];
|
|
if (!flag)
|
|
[self abortEditing];
|
|
|
|
[self setNeedsDisplay: YES];
|
|
}
|
|
|
|
/** <p>Returns the NSControl's selected cell.</p>
|
|
*/
|
|
- (id) selectedCell
|
|
{
|
|
return _cell;
|
|
}
|
|
|
|
/** <p>Returns the tag of the NSControl's selected cell (if exists).
|
|
-1 otherwise.</p><p>See Also: [NSCell-tag]</p>
|
|
*/
|
|
- (NSInteger) selectedTag
|
|
{
|
|
NSCell *selected = [self selectedCell];
|
|
|
|
if (selected == nil)
|
|
return -1;
|
|
else
|
|
return [selected tag];
|
|
}
|
|
|
|
/** <p>Returns the value of the NSControl's selected cell as double.</p>
|
|
<p>See Also: -setDoubleValue: [NSCell-doubleValue] -intValue -floatValue
|
|
-doubleValue -stringValue</p>
|
|
*/
|
|
- (double) doubleValue
|
|
{
|
|
// The validation is performed by the NSActionCell
|
|
return [[self selectedCell] doubleValue];
|
|
}
|
|
|
|
/** <p>Returns the value of the NSControl's selected cell as float.</p>
|
|
<p>See Also: -setFloatValue: [NSCell-floatValue] -intValue -stringValue
|
|
-doubleValue</p>
|
|
*/
|
|
- (float) floatValue
|
|
{
|
|
return [[self selectedCell] floatValue];
|
|
}
|
|
|
|
/** <p>Returns the value of the NSControl's selected cell as int.</p>
|
|
<p>See Also: -setIntValue: [NSCell-intValue] -floatValue -doubleValue
|
|
-stringValue</p>
|
|
*/
|
|
- (int) intValue
|
|
{
|
|
return [[self selectedCell] intValue];
|
|
}
|
|
|
|
/** <p>Returns the value of the NSControl's selected cell as int.</p>
|
|
<p>See Also: -setIntegerValue: [NSCell-integerValue] -floatValue -doubleValue
|
|
-stringValue</p>
|
|
*/
|
|
- (NSInteger) integerValue
|
|
{
|
|
return [[self selectedCell] integerValue];
|
|
}
|
|
|
|
/** <p>Returns the value of the NSControl's selected cell as NSString.</p>
|
|
<p>See Also: -setStringValue: [NSCell-stringValue] -intValue -floatValue
|
|
-doubleValue -stringValue</p>
|
|
*/
|
|
- (NSString *) stringValue
|
|
{
|
|
return [[self selectedCell] stringValue];
|
|
}
|
|
|
|
- (id) objectValue
|
|
{
|
|
return [[self selectedCell] objectValue];
|
|
}
|
|
|
|
/** <p>Sets the value of the NSControl's selected cell to double.
|
|
If the selected cell is an action cell, it marks self for display.</p>
|
|
<p>See Also: -doubleValue [NSCell-setDoubleValue:] -setIntValue:
|
|
-setStringValue: -setFloatValue:</p>
|
|
*/
|
|
- (void) setDoubleValue: (double)aDouble
|
|
{
|
|
NSCell *selected = [self selectedCell];
|
|
BOOL wasEditing = [self abortEditing];
|
|
|
|
[selected setDoubleValue: aDouble];
|
|
if (![selected isKindOfClass: actionCellClass])
|
|
[self setNeedsDisplay: YES];
|
|
|
|
if (wasEditing)
|
|
{
|
|
[[self window] makeFirstResponder: self];
|
|
}
|
|
}
|
|
|
|
/** <p>Sets the value of the NSControl's selected cell to float.
|
|
If the selected cell is an action cell, it marks self for display.</p>
|
|
<p>See Also: -floatValue [NSCell-setFloatValue:] -setIntValue:
|
|
-setStringValue: -setDoubleValue:</p>
|
|
*/
|
|
- (void) setFloatValue: (float)aFloat
|
|
{
|
|
NSCell *selected = [self selectedCell];
|
|
BOOL wasEditing = [self abortEditing];
|
|
|
|
[selected setFloatValue: aFloat];
|
|
if (![selected isKindOfClass: actionCellClass])
|
|
[self setNeedsDisplay: YES];
|
|
|
|
if (wasEditing)
|
|
{
|
|
[[self window] makeFirstResponder: self];
|
|
}
|
|
}
|
|
|
|
/** <p>Sets the value of the NSControl's selected cell to int.
|
|
If the selected cell is an action cell, it marks self for display.</p>
|
|
<p>See Also: -intValue [NSCell-setIntValue:] -setDoubleValue:
|
|
-setFloatValue: -setStringValue:</p>
|
|
*/
|
|
- (void) setIntValue: (int)anInt
|
|
{
|
|
NSCell *selected = [self selectedCell];
|
|
BOOL wasEditing = [self abortEditing];
|
|
|
|
[selected setIntValue: anInt];
|
|
if (![selected isKindOfClass: actionCellClass])
|
|
[self setNeedsDisplay: YES];
|
|
|
|
if (wasEditing)
|
|
{
|
|
[[self window] makeFirstResponder: self];
|
|
}
|
|
}
|
|
|
|
/** <p>Sets the value of the NSControl's selected cell to int.
|
|
If the selected cell is an action cell, it marks self for display.</p>
|
|
<p>See Also: -integerValue [NSCell-setIntegerValue:] -setDoubleValue:
|
|
-setFloatValue: -setStringValue:</p>
|
|
*/
|
|
- (void) setIntegerValue: (NSInteger)anInt
|
|
{
|
|
NSCell *selected = [self selectedCell];
|
|
BOOL wasEditing = [self abortEditing];
|
|
|
|
[selected setIntegerValue: anInt];
|
|
if (![selected isKindOfClass: actionCellClass])
|
|
[self setNeedsDisplay: YES];
|
|
|
|
if (wasEditing)
|
|
{
|
|
[[self window] makeFirstResponder: self];
|
|
}
|
|
}
|
|
|
|
/** <p>Sets the value of the NSControl's selected cell to NSString.
|
|
If the selected cell is an action cell, it marks self for display.</p>
|
|
<p>See Also: -stringValue [NSCell-setStringValue:] -setIntValue:
|
|
-setFloatValue: -setDoubleValue:</p>
|
|
*/
|
|
- (void) setStringValue: (NSString *)aString
|
|
{
|
|
NSCell *selected = [self selectedCell];
|
|
BOOL wasEditing = [self abortEditing];
|
|
|
|
[selected setStringValue: aString];
|
|
if (![selected isKindOfClass: actionCellClass])
|
|
[self setNeedsDisplay: YES];
|
|
|
|
if (wasEditing)
|
|
{
|
|
[[self window] makeFirstResponder: self];
|
|
}
|
|
}
|
|
|
|
- (void) setObjectValue: (id)anObject
|
|
{
|
|
NSCell *selected = [self selectedCell];
|
|
BOOL wasEditing = [self abortEditing];
|
|
|
|
[selected setObjectValue: anObject];
|
|
if (![selected isKindOfClass: actionCellClass])
|
|
[self setNeedsDisplay: YES];
|
|
|
|
if (wasEditing)
|
|
{
|
|
[[self window] makeFirstResponder: self];
|
|
}
|
|
}
|
|
|
|
/** <p>Marks self for display.</p>
|
|
*/
|
|
- (void) setNeedsDisplay
|
|
{
|
|
[super setNeedsDisplay: YES];
|
|
}
|
|
|
|
/**<p>Sets the NSControl's selected cell to the sender's double value.</p>
|
|
<p>See Also: [NSCell-takeDoubleValueFrom:] -takeFloatValueFrom:
|
|
takeIntValueFrom: takeStringValueFrom:</p>
|
|
*/
|
|
- (void) takeDoubleValueFrom: (id)sender
|
|
{
|
|
[[self selectedCell] takeDoubleValueFrom: sender];
|
|
[self setNeedsDisplay: YES];
|
|
}
|
|
|
|
/**<p>Sets the NSControl's selected cell to the sender's float value.</p>
|
|
<p>See Also: [NSCell-takeDoubleValueFrom:] -takeDoubleValueFrom:
|
|
-takeIntValueFrom: -takeStringValueFrom:</p>
|
|
*/
|
|
- (void) takeFloatValueFrom: (id)sender
|
|
{
|
|
[[self selectedCell] takeFloatValueFrom: sender];
|
|
[self setNeedsDisplay: YES];
|
|
}
|
|
|
|
/**<p>Sets the NSControl's selected cell to the sender's float int.</p>
|
|
<p>See Also: [NSCell-takeIntValueFrom:] -takeDoubleValueFrom:
|
|
-takeFloatValueFrom: -takeStringValueFrom:</p>
|
|
*/
|
|
- (void) takeIntValueFrom: (id)sender
|
|
{
|
|
[[self selectedCell] takeIntValueFrom: sender];
|
|
[self setNeedsDisplay: YES];
|
|
}
|
|
|
|
/**<p>Sets the NSControl's selected cell to the sender's float int.</p>
|
|
<p>See Also: [NSCell-takeIntegerValueFrom:] -takeDoubleValueFrom:
|
|
-takeFloatValueFrom: -takeStringValueFrom:</p>
|
|
*/
|
|
- (void) takeIntegerValueFrom: (id)sender
|
|
{
|
|
[[self selectedCell] takeIntegerValueFrom: sender];
|
|
[self setNeedsDisplay: YES];
|
|
}
|
|
|
|
- (void) takeObjectValueFrom: (id)sender
|
|
{
|
|
[[self selectedCell] takeObjectValueFrom: sender];
|
|
[self setNeedsDisplay: YES];
|
|
}
|
|
|
|
/**<p>Sets the NSControl's selected cell to the sender's float int.</p>
|
|
<p>See Also: [NSCell-takeDoubleValueFrom:] -takeDoubleValueFrom:
|
|
-takeFloatValueFrom: -takeIntValueFrom:</p>
|
|
*/
|
|
- (void) takeStringValueFrom: (id)sender
|
|
{
|
|
[[self selectedCell] takeStringValueFrom: sender];
|
|
[self setNeedsDisplay: YES];
|
|
}
|
|
|
|
/**<p>Returns the alignment of the text in the NSControl's cell.
|
|
Returns NSNaturalTextAlignment if the cell does not exists.
|
|
See <ref type="type" id="NSTextAlignment">NSTextAlignment</ref> for
|
|
more informations.</p><p>See Also: -setAlignment: [NSCell-alignment] </p>
|
|
*/
|
|
- (NSTextAlignment) alignment
|
|
{
|
|
if (_cell)
|
|
return [_cell alignment];
|
|
else
|
|
return NSNaturalTextAlignment;
|
|
}
|
|
|
|
/**<p>Returns the font of the text in the NSControl's cell. Returns nil if
|
|
the cell does not exists.</p><p>See Also: -setFont: [NSCell-font]</p>
|
|
*/
|
|
- (NSFont *) font
|
|
{
|
|
if (_cell)
|
|
return [_cell font];
|
|
else
|
|
return nil;
|
|
}
|
|
|
|
/**<p>Sets the alignment of the text in the NSControl's cell to
|
|
<var>mode</var>. This method abort the editing and marks self for display
|
|
if the cell is an NSActionCell. See <ref type="type" id="NSTextAlignment">
|
|
NSTextAlignment</ref> for more informations.</p>
|
|
<p>See Also: -alignment [NSCell-setAlignment:] -abortEditing</p>
|
|
*/
|
|
- (void) setAlignment: (NSTextAlignment)mode
|
|
{
|
|
if (_cell)
|
|
{
|
|
[self abortEditing];
|
|
|
|
[_cell setAlignment: mode];
|
|
if (![_cell isKindOfClass: actionCellClass])
|
|
[self setNeedsDisplay: YES];
|
|
}
|
|
}
|
|
|
|
/**<p>Sets the font of the text in the NSControl's cell and the
|
|
editor object (if exists) to fontObject.</p>
|
|
<p>See Also: -font [NSCell-setFont:] -currentEditor</p>
|
|
*/
|
|
- (void) setFont: (NSFont *)fontObject
|
|
{
|
|
if (_cell)
|
|
{
|
|
NSText *editor = [self currentEditor];
|
|
|
|
[_cell setFont: fontObject];
|
|
if (editor != nil)
|
|
[editor setFont: fontObject];
|
|
}
|
|
}
|
|
|
|
- (void) setFloatingPointFormat: (BOOL)autoRange
|
|
left: (NSUInteger)leftDigits
|
|
right: (NSUInteger)rightDigits
|
|
{
|
|
[self abortEditing];
|
|
|
|
[_cell setFloatingPointFormat: autoRange left: leftDigits
|
|
right: rightDigits];
|
|
if (![_cell isKindOfClass: actionCellClass])
|
|
[self setNeedsDisplay: YES];
|
|
}
|
|
|
|
- (void) setFormatter: (NSFormatter*)newFormatter
|
|
{
|
|
if (_cell)
|
|
{
|
|
[_cell setFormatter: newFormatter];
|
|
if (![_cell isKindOfClass: actionCellClass])
|
|
[self setNeedsDisplay: YES];
|
|
}
|
|
}
|
|
|
|
- (id) formatter
|
|
{
|
|
return [_cell formatter];
|
|
}
|
|
|
|
- (NSWritingDirection) baseWritingDirection
|
|
{
|
|
return [_cell baseWritingDirection];
|
|
}
|
|
|
|
- (void) setBaseWritingDirection: (NSWritingDirection)direction
|
|
{
|
|
if (_cell)
|
|
{
|
|
[_cell setBaseWritingDirection: direction];
|
|
if (![_cell isKindOfClass: actionCellClass])
|
|
[self setNeedsDisplay: YES];
|
|
}
|
|
}
|
|
|
|
/**<p>Sends an [NSCell-endEditing:] message to the current object used to
|
|
edit the NSControl. Returns NO if the the currentEditor does not exists,
|
|
YES otherwise.</p>
|
|
*/
|
|
- (BOOL) abortEditing
|
|
{
|
|
NSText *text;
|
|
|
|
text = [self currentEditor];
|
|
if (text == nil)
|
|
{
|
|
return NO;
|
|
}
|
|
|
|
[[self selectedCell] endEditing: text];
|
|
return YES;
|
|
}
|
|
|
|
/**<p>Returns the NSText object used when editing the NSControl.</p>
|
|
*/
|
|
- (NSText *) currentEditor
|
|
{
|
|
if (_cell != nil)
|
|
{
|
|
NSText *text;
|
|
|
|
text = [_window fieldEditor: NO forObject: self];
|
|
if (([text delegate] == self) && ([_window firstResponder] == text))
|
|
{
|
|
return text;
|
|
}
|
|
}
|
|
|
|
return nil;
|
|
}
|
|
|
|
/**
|
|
*/
|
|
- (void) validateEditing
|
|
{
|
|
NSText *text;
|
|
|
|
text = [self currentEditor];
|
|
if (text == nil)
|
|
{
|
|
return;
|
|
}
|
|
|
|
if ([text isRichText])
|
|
{
|
|
NSAttributedString *attr;
|
|
NSTextStorage *storage;
|
|
int len;
|
|
|
|
storage = [(NSTextView*)text textStorage];
|
|
len = [storage length];
|
|
attr = [storage attributedSubstringFromRange: NSMakeRange(0, len)];
|
|
[[self selectedCell] setAttributedStringValue: attr];
|
|
}
|
|
else
|
|
{
|
|
NSString *string;
|
|
|
|
string = AUTORELEASE([[text string] copy]);
|
|
[[self selectedCell] setStringValue: string];
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Text delegate methods
|
|
*/
|
|
|
|
/**<p>Invokes when the text cell starts to be editing.This methods posts
|
|
a NSControlTextDidBeginEditingNotification with a dictionary containing
|
|
the NSFieldEditor as user info </p><p>See Also:
|
|
[NSNotificationCenter-postNotificationName:object:userInfo:]</p>
|
|
*/
|
|
- (void) textDidBeginEditing: (NSNotification *)aNotification
|
|
{
|
|
NSMutableDictionary *dict;
|
|
|
|
dict = [[NSMutableDictionary alloc] initWithDictionary:
|
|
[aNotification userInfo]];
|
|
[dict setObject: [aNotification object] forKey: @"NSFieldEditor"];
|
|
|
|
[nc postNotificationName: NSControlTextDidBeginEditingNotification
|
|
object: self
|
|
userInfo: dict];
|
|
RELEASE(dict);
|
|
}
|
|
|
|
/**<p>Invokes when the text cell is changed. This methods posts a
|
|
NSControlTextDidChangeNotification with a dictionary containing the
|
|
NSFieldEditor as user info </p><p>See Also:
|
|
[NSNotificationCenter-postNotificationName:object:userInfo:]</p>
|
|
*/
|
|
- (void) textDidChange: (NSNotification *)aNotification
|
|
{
|
|
NSMutableDictionary *dict;
|
|
GSKeyValueBinding *theBinding;
|
|
|
|
dict = [[NSMutableDictionary alloc] initWithDictionary:
|
|
[aNotification userInfo]];
|
|
[dict setObject: [aNotification object] forKey: @"NSFieldEditor"];
|
|
|
|
[nc postNotificationName: NSControlTextDidChangeNotification
|
|
object: self
|
|
userInfo: dict];
|
|
RELEASE(dict);
|
|
|
|
theBinding = [GSKeyValueBinding getBinding: NSValueBinding
|
|
forObject: self];
|
|
if (theBinding != nil)
|
|
{
|
|
NSDictionary *options = [theBinding->info objectForKey: NSOptionsKey];
|
|
NSNumber *num = [options objectForKey: NSContinuouslyUpdatesValueBindingOption];
|
|
|
|
if ([num boolValue])
|
|
{
|
|
[theBinding reverseSetValueFor: @"objectValue"];
|
|
}
|
|
}
|
|
}
|
|
|
|
/**<p>Invokes when the text cell is changed.
|
|
This methods posts a NSControlTextDidEndEditingNotification
|
|
a dictionary containing the NSFieldEditor as user info </p><p>See Also:
|
|
[NSNotificationCenter-postNotificationName:object:userInfo:]</p>
|
|
*/
|
|
- (void) textDidEndEditing: (NSNotification *)aNotification
|
|
{
|
|
NSMutableDictionary *dict;
|
|
|
|
[self validateEditing];
|
|
[self abortEditing];
|
|
|
|
dict = [[NSMutableDictionary alloc] initWithDictionary:
|
|
[aNotification userInfo]];
|
|
[dict setObject: [aNotification object] forKey: @"NSFieldEditor"];
|
|
|
|
[nc postNotificationName: NSControlTextDidEndEditingNotification
|
|
object: self
|
|
userInfo: dict];
|
|
RELEASE(dict);
|
|
}
|
|
|
|
/**<p>Recalculates the internal size by sending [NSCell-calcDrawInfo:]
|
|
to the cell.</p>
|
|
*/
|
|
- (void) calcSize
|
|
{
|
|
[_cell calcDrawInfo: [self bounds]];
|
|
}
|
|
|
|
/**<p>Resizes the NSControl to fits the NSControl's cell size.</p>
|
|
<p>See Also: [NSCell-cellSize]</p>
|
|
*/
|
|
- (void) sizeToFit
|
|
{
|
|
[self setFrameSize: [_cell cellSize]];
|
|
}
|
|
|
|
/** <p>Returns whether the NSControl's cell is opaque</p>
|
|
*/
|
|
- (BOOL) isOpaque
|
|
{
|
|
return [_cell isOpaque];
|
|
}
|
|
|
|
- (void) drawRect: (NSRect)aRect
|
|
{
|
|
[self drawCell: _cell];
|
|
}
|
|
|
|
/**<p>Redraws a <var>aCell</var> if it is the NSControl's cell.</p>
|
|
<p>See Also: -setCell: [NSCell-drawWithFrame:inView:]</p>
|
|
*/
|
|
- (void) drawCell: (NSCell *)aCell
|
|
{
|
|
if (_cell == aCell)
|
|
{
|
|
[_cell drawWithFrame: _bounds
|
|
inView: self];
|
|
}
|
|
}
|
|
|
|
/**<p>Redraws a <var>aCell</var>'s inside if it is the NSControl's cell.</p>
|
|
<p>See Also: -setCell: [NSCell-drawInteriorWithFrame:inView:]</p>
|
|
*/
|
|
- (void) drawCellInside: (NSCell *)aCell
|
|
{
|
|
if (_cell == aCell)
|
|
{
|
|
[_cell drawInteriorWithFrame: _bounds
|
|
inView: self];
|
|
}
|
|
}
|
|
|
|
/** <p>Sets the aCell's state to NSOnState and marks self for display
|
|
if it is the NSControl's cell.</p>
|
|
*/
|
|
- (void) selectCell: (NSCell *)aCell
|
|
{
|
|
if (_cell == aCell)
|
|
{
|
|
[_cell setState: NSOnState];
|
|
[self setNeedsDisplay: YES];
|
|
}
|
|
}
|
|
|
|
/** <p>Marks self for display.</p>
|
|
*/
|
|
- (void) updateCell: (NSCell *)aCell
|
|
{
|
|
[self setNeedsDisplay: YES];
|
|
}
|
|
|
|
/** <p>Marks self for display.</p>
|
|
*/
|
|
- (void) updateCellInside: (NSCell *)aCell
|
|
{
|
|
[self setNeedsDisplay: YES];
|
|
}
|
|
|
|
/** <p>Returns the NSControl's cell action method.</p>
|
|
<p>See Also: -setAction: [NSCell-action] </p>
|
|
*/
|
|
- (SEL) action
|
|
{
|
|
return [_cell action];
|
|
}
|
|
|
|
/** <p>Returns whether the NSControl's cell can continuously sends its action
|
|
message.</p><p>See Also: -setContinuous: [NSCell-isContinuous]</p>
|
|
*/
|
|
- (BOOL) isContinuous
|
|
{
|
|
return [_cell isContinuous];
|
|
}
|
|
|
|
/**<p>Asks the NSApplication to send an action <var>theAction</var>
|
|
with <var>theTarget</var> as target to NSControl.
|
|
Returns NO if <var>theAction</var> is nil or if NSApplication can
|
|
not send the action.</p>
|
|
<p>See Also: [NSApplication-sendAction:to:from:]</p>
|
|
*/
|
|
- (BOOL) sendAction: (SEL)theAction to: (id)theTarget
|
|
{
|
|
GSKeyValueBinding *theBinding;
|
|
|
|
theBinding = [GSKeyValueBinding getBinding: NSValueBinding
|
|
forObject: self];
|
|
if (theBinding != nil)
|
|
[theBinding reverseSetValueFor: @"objectValue"];
|
|
|
|
if (theAction)
|
|
return [NSApp sendAction: theAction to: theTarget from: self];
|
|
else
|
|
return NO;
|
|
}
|
|
|
|
- (NSInteger) sendActionOn: (NSInteger)mask
|
|
{
|
|
return [_cell sendActionOn: mask];
|
|
}
|
|
|
|
/**<p>Sets the NSControl's cell action method.</p>
|
|
<p>See Also: -action [NSCell-setAction:]</p>
|
|
*/
|
|
- (void) setAction: (SEL)aSelector
|
|
{
|
|
[_cell setAction: aSelector];
|
|
}
|
|
|
|
/** <p>Sets whether the NSControl's cell can continuously sends its action
|
|
message.</p><p>See Also: -isContinuous [NSCell-setContinuous:]</p>
|
|
*/
|
|
- (void) setContinuous: (BOOL)flag
|
|
{
|
|
[_cell setContinuous: flag];
|
|
}
|
|
|
|
/** <p>Sets the target object of the NSControl's cell to anObject.</p>
|
|
<p>See Also: -target [NSCell-setTarget:]</p>
|
|
*/
|
|
- (void) setTarget: (id)anObject
|
|
{
|
|
[_cell setTarget: anObject];
|
|
}
|
|
|
|
/**<p>Returns the target object of the NSControl's cell.</p>
|
|
<p>See Also: -setTarget: [NSCell-target]</p>
|
|
*/
|
|
- (id) target
|
|
{
|
|
return [_cell target];
|
|
}
|
|
|
|
/*
|
|
* Attributed string handling
|
|
*/
|
|
- (void) setAttributedStringValue: (NSAttributedString*)attribStr
|
|
{
|
|
NSCell *selected = [self selectedCell];
|
|
|
|
[self abortEditing];
|
|
|
|
[selected setAttributedStringValue: attribStr];
|
|
if (![selected isKindOfClass: actionCellClass])
|
|
[self setNeedsDisplay: YES];
|
|
}
|
|
|
|
- (NSAttributedString*) attributedStringValue
|
|
{
|
|
NSCell *selected = [self selectedCell];
|
|
|
|
if (selected == nil)
|
|
{
|
|
return AUTORELEASE([NSAttributedString new]);
|
|
}
|
|
|
|
// As this method is not defined for NSActionCell, we have
|
|
// to do the validation here.
|
|
[self validateEditing];
|
|
|
|
return [selected attributedStringValue];
|
|
}
|
|
|
|
/** Assigning a Tag
|
|
*/
|
|
- (void) setTag: (NSInteger)anInt
|
|
{
|
|
_tag = anInt;
|
|
}
|
|
|
|
/**<p>Returns the NSControl tag</p>
|
|
<p>See Also: -setTag:</p>
|
|
*/
|
|
- (NSInteger) tag
|
|
{
|
|
return _tag;
|
|
}
|
|
|
|
/*
|
|
* Activation
|
|
*/
|
|
|
|
/**
|
|
* Simulates a single mouse click on the control. This method calls the cell's
|
|
* method performClickWithFrame:inView:. Take note that <var>sender</var> is not
|
|
* used.
|
|
*/
|
|
- (void) performClick: (id)sender
|
|
{
|
|
[_cell performClickWithFrame: [self bounds] inView: self];
|
|
}
|
|
|
|
- (BOOL) refusesFirstResponder
|
|
{
|
|
return [[self selectedCell] refusesFirstResponder];
|
|
}
|
|
|
|
- (void) setRefusesFirstResponder:(BOOL)flag
|
|
{
|
|
[[self selectedCell] setRefusesFirstResponder: flag];
|
|
}
|
|
|
|
- (BOOL) acceptsFirstResponder
|
|
{
|
|
return [[self selectedCell] acceptsFirstResponder];
|
|
}
|
|
|
|
|
|
- (void) mouseDown: (NSEvent *)theEvent
|
|
{
|
|
unsigned int event_mask = NSLeftMouseDownMask | NSLeftMouseUpMask
|
|
| NSMouseMovedMask | NSLeftMouseDraggedMask | NSOtherMouseDraggedMask
|
|
| NSRightMouseDraggedMask;
|
|
NSEvent *e = nil;
|
|
|
|
// If not enabled ignore mouse clicks
|
|
if (![self isEnabled])
|
|
{
|
|
[super mouseDown: theEvent];
|
|
return;
|
|
}
|
|
|
|
// Ignore multiple clicks, if configured to do so
|
|
if (_ignoresMultiClick && ([theEvent clickCount] > 1))
|
|
{
|
|
[super mouseDown: theEvent];
|
|
return;
|
|
}
|
|
|
|
// Make sure self does not go away during the processing of the event
|
|
RETAIN(self);
|
|
|
|
// loop until mouse goes up
|
|
e = theEvent;
|
|
while (1)
|
|
{
|
|
NSPoint location = [self convertPoint: [e locationInWindow]
|
|
fromView: nil];
|
|
|
|
// ask the cell to track the mouse only,
|
|
// if the mouse is within the cell
|
|
if ([self mouse: location inRect: _bounds])
|
|
{
|
|
BOOL done;
|
|
|
|
[_cell setHighlighted: YES];
|
|
[self setNeedsDisplay: YES];
|
|
done = [_cell trackMouse: e
|
|
inRect: _bounds
|
|
ofView: self
|
|
untilMouseUp: [[_cell class] prefersTrackingUntilMouseUp]];
|
|
[_cell setHighlighted: NO];
|
|
[self setNeedsDisplay: YES];
|
|
|
|
if (done)
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
|
|
e = [NSApp nextEventMatchingMask: event_mask
|
|
untilDate: [NSDate distantFuture]
|
|
inMode: NSEventTrackingRunLoopMode
|
|
dequeue: YES];
|
|
if ([e type] == NSLeftMouseUp)
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
|
|
// undo initial retain
|
|
RELEASE(self);
|
|
}
|
|
|
|
- (BOOL) shouldBeTreatedAsInkEvent: (NSEvent *)theEvent
|
|
{
|
|
return NO;
|
|
}
|
|
|
|
- (void) resetCursorRects
|
|
{
|
|
[_cell resetCursorRect: _bounds inView: self];
|
|
}
|
|
|
|
/**<p>Returns wheter multiple clicks are ignored.</p>
|
|
<p>See Also: -setIgnoresMultiClick: -mouseDown:</p>
|
|
*/
|
|
- (BOOL) ignoresMultiClick
|
|
{
|
|
return _ignoresMultiClick;
|
|
}
|
|
|
|
|
|
/** <p>Sets wheter multiple clicks are ignored.</p>
|
|
<p>See Also: -ignoresMultiClick -mouseDown:</p>
|
|
*/
|
|
- (void) setIgnoresMultiClick: (BOOL)flag
|
|
{
|
|
_ignoresMultiClick = flag;
|
|
}
|
|
|
|
/**<p>Returns the mouse flags. This flags are usally sets in
|
|
the NSCell-trackMouse:inRect:ofView:untilMouseUp: method.</p>
|
|
<p>This is a NeXTStep 3.3 method, no longer officially supported.</p>
|
|
*/
|
|
- (NSInteger) mouseDownFlags
|
|
{
|
|
return [[self selectedCell] mouseDownFlags];
|
|
}
|
|
|
|
/*
|
|
* NSCoding protocol
|
|
*/
|
|
- (void) encodeWithCoder: (NSCoder*)aCoder
|
|
{
|
|
[super encodeWithCoder: aCoder];
|
|
if ([aCoder allowsKeyedCoding])
|
|
{
|
|
[aCoder encodeObject: [self cell] forKey: @"NSCell"];
|
|
[aCoder encodeBool: [self isEnabled] forKey: @"NSEnabled"];
|
|
if (_tag)
|
|
{
|
|
[aCoder encodeInt: [self tag] forKey: @"NSTag"];
|
|
}
|
|
}
|
|
else
|
|
{
|
|
encode_NSInteger(aCoder, &_tag);
|
|
[aCoder encodeObject: _cell];
|
|
[aCoder encodeValueOfObjCType: @encode(BOOL) at: &_ignoresMultiClick];
|
|
}
|
|
}
|
|
|
|
- (id) initWithCoder: (NSCoder*)aDecoder
|
|
{
|
|
self = [super initWithCoder: aDecoder];
|
|
if (self == nil)
|
|
{
|
|
return nil;
|
|
}
|
|
|
|
if ([aDecoder allowsKeyedCoding])
|
|
{
|
|
NSCell *cell = [aDecoder decodeObjectForKey: @"NSCell"];
|
|
|
|
if (cell != nil)
|
|
{
|
|
[self setCell: cell];
|
|
}
|
|
else
|
|
{
|
|
// This is needed for subclasses without cells, like NSColorWeel
|
|
// as we store some properties only on the cell.
|
|
cell = [[[self class] cellClass] new];
|
|
|
|
[self setCell: cell];
|
|
RELEASE(cell);
|
|
}
|
|
if ([aDecoder containsValueForKey: @"NSEnabled"])
|
|
{
|
|
// Don't use this information as it also comes from the cell
|
|
// and NSComboBox has always YES here, even when disabled
|
|
//[self setEnabled: [aDecoder decodeBoolForKey: @"NSEnabled"]];
|
|
}
|
|
if ([aDecoder containsValueForKey: @"NSTag"])
|
|
{
|
|
int tag = [aDecoder decodeIntForKey: @"NSTag"];
|
|
[self setTag: tag];
|
|
}
|
|
}
|
|
else
|
|
{
|
|
decode_NSInteger(aDecoder, &_tag);
|
|
[aDecoder decodeValueOfObjCType: @encode(id) at: &_cell];
|
|
[aDecoder decodeValueOfObjCType: @encode(BOOL) at: &_ignoresMultiClick];
|
|
}
|
|
|
|
return self;
|
|
}
|
|
|
|
- (void) bind: (NSString *)binding
|
|
toObject: (id)anObject
|
|
withKeyPath: (NSString *)keyPath
|
|
options: (NSDictionary *)options
|
|
{
|
|
if ([binding isEqual: NSValueBinding])
|
|
{
|
|
GSKeyValueBinding *kvb;
|
|
|
|
[self unbind: binding];
|
|
kvb = [[GSKeyValueBinding alloc] initWithBinding: @"objectValue"
|
|
withName: NSValueBinding
|
|
toObject: anObject
|
|
withKeyPath: keyPath
|
|
options: options
|
|
fromObject: self];
|
|
// The binding will be retained in the binding table
|
|
RELEASE(kvb);
|
|
}
|
|
else if ([binding hasPrefix: NSEnabledBinding])
|
|
{
|
|
GSKeyValueBinding *kvb;
|
|
|
|
[self unbind: binding];
|
|
kvb = [[GSKeyValueAndBinding alloc] initWithBinding: NSEnabledBinding
|
|
withName: binding
|
|
toObject: anObject
|
|
withKeyPath: keyPath
|
|
options: options
|
|
fromObject: self];
|
|
// The binding will be retained in the binding table
|
|
RELEASE(kvb);
|
|
}
|
|
else
|
|
{
|
|
[super bind: binding
|
|
toObject: anObject
|
|
withKeyPath: keyPath
|
|
options: options];
|
|
}
|
|
}
|
|
|
|
- (void) setValue: (id)anObject forKey: (NSString*)aKey
|
|
{
|
|
if ([aKey isEqual: NSFontNameBinding])
|
|
{
|
|
[self setFont: [[NSFontManager sharedFontManager] convertFont: [self font]
|
|
toFace: anObject]];
|
|
}
|
|
else if ([aKey isEqual: NSFontSizeBinding])
|
|
{
|
|
[self setFont: [[NSFontManager sharedFontManager] convertFont: [self font]
|
|
toSize: [anObject doubleValue]]];
|
|
}
|
|
else
|
|
{
|
|
[super setValue: anObject forKey: aKey];
|
|
}
|
|
}
|
|
|
|
- (id) valueForKey: (NSString*)aKey
|
|
{
|
|
if ([aKey isEqual: NSFontNameBinding])
|
|
{
|
|
return [[self font] fontName];
|
|
}
|
|
else if ([aKey isEqual: NSFontSizeBinding])
|
|
{
|
|
return [NSNumber numberWithDouble: (double)[[self font] pointSize]];
|
|
}
|
|
else
|
|
{
|
|
return [super valueForKey: aKey];
|
|
}
|
|
}
|
|
|
|
- (NSSize) sizeThatFits: (NSSize)size
|
|
{
|
|
// FIXME: This is a stub
|
|
return size;
|
|
}
|
|
|
|
@end
|
|
|
|
@implementation NSControl(KeyViewLoop)
|
|
|
|
- (void) _setUpKeyViewLoopWithNextKeyView: (NSView *)nextKeyView
|
|
{
|
|
// Controls are expected to have no subviews
|
|
//NSLog(@"%@@%p -_setUpKeyViewLoopWithKeyKeyView:%@@%p", [self class], self, [nextKeyView class], nextKeyView);
|
|
[self setNextKeyView: nextKeyView];
|
|
}
|
|
|
|
@end
|