mirror of
https://github.com/gnustep/libs-gui.git
synced 2025-04-23 15:11:37 +00:00
git-svn-id: svn+ssh://svn.gna.org/svn/gnustep/libs/gui/trunk@32451 72102866-910b-0410-8b05-ffd578937521
478 lines
9.9 KiB
Objective-C
478 lines
9.9 KiB
Objective-C
/** <title>NSResponder</title>
|
|
|
|
<abstract>Abstract class which is basis of command and event processing</abstract>
|
|
|
|
Copyright (C) 1996,1999 Free Software Foundation, Inc.
|
|
|
|
Author: Scott Christley <scottc@net-community.com>
|
|
Date: 1996
|
|
|
|
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/NSCoder.h>
|
|
#import <Foundation/NSDebug.h>
|
|
#import <Foundation/NSInvocation.h>
|
|
#import "AppKit/NSResponder.h"
|
|
#import "AppKit/NSApplication.h"
|
|
#import "AppKit/NSMenu.h"
|
|
#import "AppKit/NSEvent.h"
|
|
#import "AppKit/NSGraphics.h"
|
|
#import "AppKit/NSHelpManager.h"
|
|
#import "AppKit/NSInputManager.h"
|
|
|
|
@implementation NSResponder
|
|
|
|
/*
|
|
* Class methods
|
|
*/
|
|
+ (void) initialize
|
|
{
|
|
if (self == [NSResponder class])
|
|
{
|
|
[self setVersion: 1];
|
|
|
|
/* Gets the current input manager - this forces it to read the
|
|
key binding files at this stage. */
|
|
[NSInputManager currentInputManager];
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Instance methods
|
|
*/
|
|
/*
|
|
* Managing the next responder
|
|
*/
|
|
- (NSResponder*) nextResponder
|
|
{
|
|
return _next_responder;
|
|
}
|
|
|
|
- (void) setNextResponder: (NSResponder*)aResponder
|
|
{
|
|
_next_responder = aResponder;
|
|
}
|
|
|
|
/**
|
|
* Returns YES if the receiver is able to become the first responder,
|
|
* NO otherwise.
|
|
*/
|
|
- (BOOL) acceptsFirstResponder
|
|
{
|
|
return NO;
|
|
}
|
|
|
|
- (BOOL) becomeFirstResponder
|
|
{
|
|
return YES;
|
|
}
|
|
|
|
- (BOOL) resignFirstResponder
|
|
{
|
|
return YES;
|
|
}
|
|
|
|
/*
|
|
* Aid event processing
|
|
*/
|
|
- (BOOL) performKeyEquivalent: (NSEvent*)theEvent
|
|
{
|
|
return NO;
|
|
}
|
|
|
|
/**
|
|
* If the receiver responds to anAction, it performs that method with
|
|
* anObject as its argument, discards any return value, and return YES.<br />
|
|
* Otherwise, the next responder in the chain is asked to perform
|
|
* anAction and the result of that is returned.<br />
|
|
* If no responder in the chain is able to respond to anAction, then
|
|
* NO is returned.
|
|
*/
|
|
- (BOOL) tryToPerform: (SEL)anAction with: (id)anObject
|
|
{
|
|
/* Can we perform the action -then do it */
|
|
if ([self respondsToSelector: anAction])
|
|
{
|
|
IMP actionIMP = [self methodForSelector: anAction];
|
|
if (0 != actionIMP)
|
|
{
|
|
actionIMP(self, anAction, anObject);
|
|
return YES;
|
|
}
|
|
return YES;
|
|
}
|
|
else
|
|
{
|
|
/* If we cannot perform then try the next responder */
|
|
if (!_next_responder)
|
|
return NO;
|
|
else
|
|
return [_next_responder tryToPerform: anAction with: anObject];
|
|
}
|
|
}
|
|
|
|
- (BOOL) performMnemonic: (NSString*)aString
|
|
{
|
|
return NO;
|
|
}
|
|
|
|
- (void) interpretKeyEvents:(NSArray*)eventArray
|
|
{
|
|
[[NSInputManager currentInputManager] handleKeyboardEvents: eventArray
|
|
client: self];
|
|
}
|
|
|
|
- (void) flushBufferedKeyEvents
|
|
{
|
|
}
|
|
|
|
- (void) doCommandBySelector:(SEL)aSelector
|
|
{
|
|
if (![self tryToPerform: aSelector with: nil])
|
|
{
|
|
NSBeep();
|
|
}
|
|
}
|
|
|
|
- (void) insertText: (id)aString
|
|
{
|
|
if (_next_responder)
|
|
[_next_responder insertText: aString];
|
|
else
|
|
{
|
|
NSBeep ();
|
|
}
|
|
}
|
|
|
|
|
|
/*
|
|
* Forwarding event messages
|
|
*/
|
|
- (void) flagsChanged: (NSEvent*)theEvent
|
|
{
|
|
if (_next_responder)
|
|
[_next_responder flagsChanged: theEvent];
|
|
else
|
|
[self noResponderFor: @selector(flagsChanged:)];
|
|
}
|
|
|
|
- (void) helpRequested: (NSEvent*)theEvent
|
|
{
|
|
if ([[NSHelpManager sharedHelpManager]
|
|
showContextHelpForObject: self
|
|
locationHint: [theEvent locationInWindow]] == NO)
|
|
{
|
|
if (_next_responder)
|
|
{
|
|
[_next_responder helpRequested: theEvent];
|
|
return;
|
|
}
|
|
}
|
|
[NSHelpManager setContextHelpModeActive: NO];
|
|
}
|
|
|
|
- (void) keyDown: (NSEvent*)theEvent
|
|
{
|
|
if (_next_responder)
|
|
[_next_responder keyDown: theEvent];
|
|
else
|
|
[self noResponderFor: @selector(keyDown:)];
|
|
}
|
|
|
|
- (void) keyUp: (NSEvent*)theEvent
|
|
{
|
|
if (_next_responder)
|
|
[_next_responder keyUp: theEvent];
|
|
else
|
|
[self noResponderFor: @selector(keyUp:)];
|
|
}
|
|
|
|
- (void) otherMouseDown: (NSEvent*)theEvent
|
|
{
|
|
if (_next_responder)
|
|
[_next_responder otherMouseDown: theEvent];
|
|
else
|
|
[self noResponderFor: @selector(otherMouseDown:)];
|
|
}
|
|
|
|
- (void) otherMouseDragged: (NSEvent*)theEvent
|
|
{
|
|
if (_next_responder)
|
|
[_next_responder otherMouseDragged: theEvent];
|
|
else
|
|
[self noResponderFor: @selector(otherMouseDragged:)];
|
|
}
|
|
|
|
- (void) otherMouseUp: (NSEvent*)theEvent
|
|
{
|
|
if (_next_responder)
|
|
[_next_responder otherMouseUp: theEvent];
|
|
else
|
|
[self noResponderFor: @selector(otherMouseUp:)];
|
|
}
|
|
|
|
- (void) mouseDown: (NSEvent*)theEvent
|
|
{
|
|
if (_next_responder)
|
|
[_next_responder mouseDown: theEvent];
|
|
else
|
|
[self noResponderFor: @selector(mouseDown:)];
|
|
}
|
|
|
|
- (void) mouseDragged: (NSEvent*)theEvent
|
|
{
|
|
if (_next_responder)
|
|
[_next_responder mouseDragged: theEvent];
|
|
else
|
|
[self noResponderFor: @selector(mouseDragged:)];
|
|
}
|
|
|
|
- (void) mouseEntered: (NSEvent*)theEvent
|
|
{
|
|
if (_next_responder)
|
|
[_next_responder mouseEntered: theEvent];
|
|
else
|
|
[self noResponderFor: @selector(mouseEntered:)];
|
|
}
|
|
|
|
- (void) mouseExited: (NSEvent*)theEvent
|
|
{
|
|
if (_next_responder)
|
|
[_next_responder mouseExited: theEvent];
|
|
else
|
|
[self noResponderFor: @selector(mouseExited:)];
|
|
}
|
|
|
|
- (void) mouseMoved: (NSEvent*)theEvent
|
|
{
|
|
if (_next_responder)
|
|
[_next_responder mouseMoved: theEvent];
|
|
else
|
|
[self noResponderFor: @selector(mouseMoved:)];
|
|
}
|
|
|
|
- (void) mouseUp: (NSEvent*)theEvent
|
|
{
|
|
if (_next_responder)
|
|
[_next_responder mouseUp: theEvent];
|
|
else
|
|
[self noResponderFor: @selector(mouseUp:)];
|
|
}
|
|
|
|
- (void) noResponderFor: (SEL)eventSelector
|
|
{
|
|
/* Only beep for key down events */
|
|
if (sel_isEqual(eventSelector, @selector(keyDown:)))
|
|
NSBeep();
|
|
}
|
|
|
|
- (void) rightMouseDown: (NSEvent*)theEvent
|
|
{
|
|
if (_next_responder != nil)
|
|
[_next_responder rightMouseDown: theEvent];
|
|
else
|
|
[self noResponderFor: @selector(rightMouseDown:)];
|
|
}
|
|
|
|
- (void) rightMouseDragged: (NSEvent*)theEvent
|
|
{
|
|
if (_next_responder)
|
|
[_next_responder rightMouseDragged: theEvent];
|
|
else
|
|
[self noResponderFor: @selector(rightMouseDragged:)];
|
|
}
|
|
|
|
- (void) rightMouseUp: (NSEvent*)theEvent
|
|
{
|
|
if (_next_responder)
|
|
[_next_responder rightMouseUp: theEvent];
|
|
else
|
|
[self noResponderFor: @selector(rightMouseUp:)];
|
|
}
|
|
|
|
- (void) scrollWheel: (NSEvent *)theEvent
|
|
{
|
|
if (_next_responder)
|
|
[_next_responder scrollWheel: theEvent];
|
|
else
|
|
[self noResponderFor: @selector(scrollWheel:)];
|
|
}
|
|
|
|
/*
|
|
* Services menu support
|
|
*/
|
|
- (id) validRequestorForSendType: (NSString*)typeSent
|
|
returnType: (NSString*)typeReturned
|
|
{
|
|
if (_next_responder)
|
|
return [_next_responder validRequestorForSendType: typeSent
|
|
returnType: typeReturned];
|
|
else
|
|
return nil;
|
|
}
|
|
|
|
/*
|
|
* NSCoding protocol
|
|
* NB. Don't encode responder chain - it's transient information that should
|
|
* be reconstructed from else where in the encoded archive.
|
|
*/
|
|
- (void) encodeWithCoder: (NSCoder*)aCoder
|
|
{
|
|
if ([aCoder allowsKeyedCoding])
|
|
{
|
|
if (_interface_style != NSNoInterfaceStyle)
|
|
{
|
|
[aCoder encodeInt: _interface_style
|
|
forKey: @"NSInterfaceStyle"];
|
|
}
|
|
|
|
if ([self menu] != nil)
|
|
{
|
|
[aCoder encodeObject: [self menu]
|
|
forKey: @"NSMenu"];
|
|
}
|
|
if ([self nextResponder] != nil)
|
|
{
|
|
[aCoder encodeConditionalObject: [self nextResponder]
|
|
forKey: @"NSNextResponder"];
|
|
}
|
|
}
|
|
else
|
|
{
|
|
[aCoder encodeValueOfObjCType: @encode(int) at: &_interface_style];
|
|
[aCoder encodeObject: _menu];
|
|
}
|
|
}
|
|
|
|
- (id) initWithCoder: (NSCoder*)aDecoder
|
|
{
|
|
id obj = nil;
|
|
|
|
if ([aDecoder allowsKeyedCoding])
|
|
{
|
|
if ([aDecoder containsValueForKey: @"NSInterfaceStyle"])
|
|
{
|
|
_interface_style = [aDecoder decodeIntForKey: @"NSInterfaceStyle"];
|
|
}
|
|
|
|
if ([aDecoder containsValueForKey: @"NSMenu"])
|
|
{
|
|
obj = [aDecoder decodeObjectForKey: @"NSMenu"];
|
|
}
|
|
|
|
if ([aDecoder containsValueForKey: @"NSNextResponder"])
|
|
{
|
|
[self setNextResponder: [aDecoder decodeObjectForKey: @"NSNextResponder"]];
|
|
}
|
|
}
|
|
else
|
|
{
|
|
[aDecoder decodeValueOfObjCType: @encode(int) at: &_interface_style];
|
|
obj = [aDecoder decodeObject];
|
|
}
|
|
|
|
[self setMenu: obj];
|
|
|
|
return self;
|
|
}
|
|
- (void) dealloc
|
|
{
|
|
RELEASE(_menu);
|
|
[super dealloc];
|
|
}
|
|
- (NSMenu*) menu
|
|
{
|
|
return _menu;
|
|
}
|
|
|
|
- (void) setMenu: (NSMenu*)aMenu
|
|
{
|
|
ASSIGN(_menu, aMenu);
|
|
}
|
|
|
|
- (NSInterfaceStyle) interfaceStyle
|
|
{
|
|
return _interface_style;
|
|
}
|
|
|
|
- (void) setInterfaceStyle: (NSInterfaceStyle)aStyle
|
|
{
|
|
_interface_style = aStyle;
|
|
}
|
|
|
|
- (NSUndoManager*) undoManager
|
|
{
|
|
if (_next_responder)
|
|
return [_next_responder undoManager];
|
|
else
|
|
return nil;
|
|
}
|
|
|
|
- (BOOL) shouldBeTreatedAsInkEvent: (NSEvent *)theEvent
|
|
{
|
|
return NO;
|
|
}
|
|
|
|
- (BOOL)presentError:(NSError *)error
|
|
{
|
|
error = [self willPresentError: error];
|
|
|
|
if (_next_responder)
|
|
{
|
|
return [_next_responder presentError: error];
|
|
}
|
|
else
|
|
{
|
|
return [NSApp presentError: error];
|
|
}
|
|
}
|
|
|
|
- (void)presentError:(NSError *)error
|
|
modalForWindow:(NSWindow *)window
|
|
delegate:(id)delegate
|
|
didPresentSelector:(SEL)sel
|
|
contextInfo:(void *)context
|
|
{
|
|
error = [self willPresentError: error];
|
|
if (_next_responder)
|
|
{
|
|
[_next_responder presentError: error
|
|
modalForWindow: window
|
|
delegate: delegate
|
|
didPresentSelector: sel
|
|
contextInfo: context];
|
|
}
|
|
else
|
|
{
|
|
[NSApp presentError: error
|
|
modalForWindow: window
|
|
delegate: delegate
|
|
didPresentSelector: sel
|
|
contextInfo: context];
|
|
}
|
|
}
|
|
|
|
- (NSError *) willPresentError: (NSError *)error
|
|
{
|
|
return error;
|
|
}
|
|
|
|
@end
|