libs-base/Source/NSInvocationOperation.m
Richard Frith-MacDonald 498b8300c3 add new class from Graham Lee <iamleeg@gmail.com>
git-svn-id: svn+ssh://svn.gna.org/svn/gnustep/libs/base/trunk@36860 72102866-910b-0410-8b05-ffd578937521
2013-07-09 18:59:38 +00:00

133 lines
3.4 KiB
Objective-C

/**Implementation for NSInvocationOperation for GNUStep
Copyright (C) 2013 Free Software Foundation, Inc.
Written by: Graham Lee <iamleeg@gmail.com>
Date: 2013
This file is part of the GNUstep Base 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
Library General Public License for more details.
You should have received a copy of the GNU Lesser General Public
License along with this library; if not, write to the Free
Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
Boston, MA 02111 USA.
<title>NSInvocationOperation class reference</title>
$Date$ $Revision$
*/
#import "common.h"
#import <Foundation/NSInvocationOperation.h>
#import <Foundation/NSException.h>
#import <Foundation/NSInvocation.h>
#import <Foundation/NSMethodSignature.h>
#import <Foundation/NSValue.h>
@implementation NSInvocationOperation
- (id) initWithInvocation: (NSInvocation *)inv
{
if (((self = [super init])) != nil)
{
[inv retainArguments];
_invocation = [inv retain];
}
return self;
}
- (id) initWithTarget: (id)target selector: (SEL)aSelector object: (id)arg
{
NSMethodSignature *methodSignature;
NSInvocation *inv;
methodSignature = [target methodSignatureForSelector: aSelector];
inv = [NSInvocation invocationWithMethodSignature: methodSignature];
[inv setTarget: target];
[inv setSelector: aSelector];
if ([methodSignature numberOfArguments] > 2)
[inv setArgument: &arg atIndex: 2];
return [self initWithInvocation: inv];
}
- (void) main
{
if (![self isCancelled])
{
NS_DURING
[_invocation invoke];
NS_HANDLER
_exception = [localException copy];
NS_ENDHANDLER
}
}
- (NSInvocation *) invocation
{
return [[_invocation retain] autorelease];
}
- (id) result
{
id result = nil;
if (![self isFinished])
{
return nil;
}
if (nil != _exception)
{
[_exception raise];
}
else if ([self isCancelled])
{
[NSException raise: (id)NSInvocationOperationCancelledException
format: @"*** %s: operation was cancelled", __PRETTY_FUNCTION__];
}
else
{
const char *returnType = [[_invocation methodSignature] methodReturnType];
if (0 == strncmp(@encode(void), returnType, 1))
{
[NSException raise: (id)NSInvocationOperationVoidResultException
format: @"*** %s: void result", __PRETTY_FUNCTION__];
}
else if (0 == strncmp(@encode(id), returnType, 1))
{
[_invocation getReturnValue: &result];
}
else
{
unsigned char *buffer = malloc([[_invocation methodSignature]
methodReturnLength]);
[_invocation getReturnValue: buffer];
result = [NSValue valueWithBytes: buffer objCType: returnType];
}
}
return result;
}
- (void) dealloc
{
[_invocation release];
[_exception release];
[super dealloc];
}
@end
const NSString * NSInvocationOperationVoidResultException
= @"NSInvocationOperationVoidResultException";
const NSString * NSInvocationOperationCancelledException
= @"NSInvcationOperationCancelledException";