libs-base/Source/NSArray.m
Fred Kiefer e7d5480cd1 * Source/NSArray.m (-objectsAtIndexes:): Fix comparision of
unsigned variable with NSNotFound.


git-svn-id: svn+ssh://svn.gna.org/svn/gnustep/libs/base/trunk@34389 72102866-910b-0410-8b05-ffd578937521
2012-01-02 15:07:32 +00:00

2500 lines
61 KiB
Objective-C
Raw Permalink Blame History

This file contains invisible Unicode characters

This file contains invisible Unicode characters that are indistinguishable to humans but may be processed differently by a computer. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

/** NSArray - Array object to hold other objects.
Copyright (C) 1995, 1996, 1998 Free Software Foundation, Inc.
Written by: Andrew Kachites McCallum <mccallum@gnu.ai.mit.edu>
From skeleton by: Adam Fedor <fedor@boulder.colorado.edu>
Created: March 1995
Rewrite by: Richard Frith-Macdonald <richard@brainstorm.co.uk>
January 1998 - new methods and changes as documented for Rhapsody plus
changes of array indices to type unsigned, plus major efficiency hacks.
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>NSArray class reference</title>
$Date$ $Revision$
*/
#import "common.h"
#import "Foundation/NSArray.h"
#import "Foundation/NSEnumerator.h"
#import "Foundation/NSCoder.h"
#import "Foundation/NSData.h"
#import "Foundation/NSRange.h"
#import "Foundation/NSException.h"
#import "Foundation/NSAutoreleasePool.h"
#import "Foundation/NSThread.h"
#import "Foundation/NSMapTable.h"
#import "Foundation/NSLock.h"
#import "Foundation/NSValue.h"
#import "Foundation/NSNull.h"
#import "Foundation/NSKeyValueCoding.h"
#import "Foundation/NSSet.h"
#import "Foundation/NSUserDefaults.h"
#import "Foundation/NSIndexSet.h"
// For private method _decodeArrayOfObjectsForKey:
#import "Foundation/NSKeyedArchiver.h"
#import "GNUstepBase/NSObject+GNUstepBase.h"
#import "GSPrivate.h"
#import "GSFastEnumeration.h"
static BOOL GSMacOSXCompatiblePropertyLists(void)
{
if (GSPrivateDefaultsFlag(NSWriteOldStylePropertyLists) == YES)
return NO;
return GSPrivateDefaultsFlag(GSMacOSXCompatible);
}
extern void GSPropertyListMake(id,NSDictionary*,BOOL,BOOL,unsigned,id*);
@interface NSArrayEnumerator : NSEnumerator
{
NSArray *array;
unsigned pos;
IMP get;
unsigned (*cnt)(NSArray*, SEL);
}
- (id) initWithArray: (NSArray*)anArray;
@end
@interface NSArrayEnumeratorReverse : NSArrayEnumerator
@end
static Class NSArrayClass;
static Class GSArrayClass;
static Class NSMutableArrayClass;
static Class GSMutableArrayClass;
static Class GSPlaceholderArrayClass;
static GSPlaceholderArray *defaultPlaceholderArray;
static NSMapTable *placeholderMap;
static NSLock *placeholderLock;
/**
* A simple, low overhead, ordered container for objects. All the objects
* in the container are retained by it. The container may not contain nil
* (though it may contain [NSNull+null]).
*/
@implementation NSArray
static SEL addSel;
static SEL appSel;
static SEL countSel;
static SEL eqSel;
static SEL oaiSel;
static SEL remSel;
static SEL rlSel;
+ (void) atExit
{
DESTROY(defaultPlaceholderArray);
DESTROY(placeholderLock);
DESTROY(placeholderMap);
}
+ (void) initialize
{
if (self == [NSArray class])
{
[self setVersion: 1];
addSel = @selector(addObject:);
appSel = @selector(appendString:);
countSel = @selector(count);
eqSel = @selector(isEqual:);
oaiSel = @selector(objectAtIndex:);
remSel = @selector(removeObjectAtIndex:);
rlSel = @selector(removeLastObject);
NSArrayClass = [NSArray class];
NSMutableArrayClass = [NSMutableArray class];
GSArrayClass = [GSArray class];
GSMutableArrayClass = [GSMutableArray class];
GSPlaceholderArrayClass = [GSPlaceholderArray class];
/*
* Set up infrastructure for placeholder arrays.
*/
defaultPlaceholderArray = (GSPlaceholderArray*)
NSAllocateObject(GSPlaceholderArrayClass, 0, NSDefaultMallocZone());
placeholderMap = NSCreateMapTable(NSNonOwnedPointerMapKeyCallBacks,
NSNonRetainedObjectMapValueCallBacks, 0);
placeholderLock = [NSLock new];
[self registerAtExit];
}
}
+ (id) allocWithZone: (NSZone*)z
{
if (self == NSArrayClass)
{
/*
* For a constant array, we return a placeholder object that can
* be converted to a real object when its initialisation method
* is called.
*/
if (z == NSDefaultMallocZone() || z == 0)
{
/*
* As a special case, we can return a placeholder for an array
* in the default malloc zone extremely efficiently.
*/
return defaultPlaceholderArray;
}
else
{
id obj;
/*
* For anything other than the default zone, we need to
* locate the correct placeholder in the (lock protected)
* table of placeholders.
*/
[placeholderLock lock];
obj = (id)NSMapGet(placeholderMap, (void*)z);
if (obj == nil)
{
/*
* There is no placeholder object for this zone, so we
* create a new one and use that.
*/
obj = (id)NSAllocateObject(GSPlaceholderArrayClass, 0, z);
NSMapInsert(placeholderMap, (void*)z, (void*)obj);
}
[placeholderLock unlock];
return obj;
}
}
else
{
return NSAllocateObject(self, 0, z);
}
}
/**
* Returns an empty autoreleased array.
*/
+ (id) array
{
id o;
o = [self allocWithZone: NSDefaultMallocZone()];
o = [o initWithObjects: (id*)0 count: 0];
return AUTORELEASE(o);
}
/**
* Returns a new autoreleased NSArray instance containing all the objects from
* array, in the same order as the original.
*/
+ (id) arrayWithArray: (NSArray*)array
{
id o;
o = [self allocWithZone: NSDefaultMallocZone()];
o = [o initWithArray: array];
return AUTORELEASE(o);
}
/**
* Returns an autoreleased array based upon the file. The new array is
* created using [NSObject+allocWithZone:] and initialised using the
* [NSArray-initWithContentsOfFile:] method. See the documentation for those
* methods for more detail.
*/
+ (id) arrayWithContentsOfFile: (NSString*)file
{
id o;
o = [self allocWithZone: NSDefaultMallocZone()];
o = [o initWithContentsOfFile: file];
return AUTORELEASE(o);
}
/**
* Returns an autoreleased array from the contents of aURL. The new array is
* created using [NSObject+allocWithZone:] and initialised using the
* -initWithContentsOfURL: method. See the documentation for those
* methods for more detail.
*/
+ (id) arrayWithContentsOfURL: (NSURL*)aURL
{
id o;
o = [self allocWithZone: NSDefaultMallocZone()];
o = [o initWithContentsOfURL: aURL];
return AUTORELEASE(o);
}
/**
* Returns an autoreleased array containing anObject.
*/
+ (id) arrayWithObject: (id)anObject
{
id o;
o = [self allocWithZone: NSDefaultMallocZone()];
o = [o initWithObjects: &anObject count: 1];
return AUTORELEASE(o);
}
/**
* Returns an autoreleased array containing the list
* of objects, preserving order.
*/
+ (id) arrayWithObjects: firstObject, ...
{
id a = [self allocWithZone: NSDefaultMallocZone()];
GS_USEIDLIST(firstObject,
a = [a initWithObjects: __objects count: __count]);
return AUTORELEASE(a);
}
/**
* Returns an autoreleased array containing the specified
* objects, preserving order.
*/
+ (id) arrayWithObjects: (const id[])objects count: (NSUInteger)count
{
return AUTORELEASE([[self allocWithZone: NSDefaultMallocZone()]
initWithObjects: objects count: count]);
}
/**
* Returns an autoreleased array formed from the contents of
* the receiver and adding anObject as the last item.
*/
- (NSArray*) arrayByAddingObject: (id)anObject
{
id na;
unsigned c = [self count];
if (anObject == nil)
[NSException raise: NSInvalidArgumentException
format: @"Attempt to add nil to an array"];
if (c == 0)
{
na = [[GSArrayClass allocWithZone: NSDefaultMallocZone()]
initWithObjects: &anObject count: 1];
}
else
{
GS_BEGINIDBUF(objects, c+1);
[self getObjects: objects];
objects[c] = anObject;
na = [[GSArrayClass allocWithZone: NSDefaultMallocZone()]
initWithObjects: objects count: c+1];
GS_ENDIDBUF();
}
return AUTORELEASE(na);
}
/**
* Returns a new array which is the concatenation of self and
* otherArray (in this precise order).
*/
- (NSArray*) arrayByAddingObjectsFromArray: (NSArray*)anotherArray
{
id na;
unsigned c;
unsigned l;
unsigned e;
c = [self count];
l = [anotherArray count];
e = c + l;
{
GS_BEGINIDBUF(objects, e);
[self getObjects: objects];
if ([anotherArray isProxy])
{
unsigned i = c;
unsigned j = 0;
while (i < e)
{
objects[i++] = [anotherArray objectAtIndex: j++];
}
}
else
{
[anotherArray getObjects: &objects[c]];
}
na = [NSArrayClass arrayWithObjects: objects count: e];
GS_ENDIDBUF();
}
return na;
}
/**
* Returns the abstract class ... arrays are coded as abstract arrays.
*/
- (Class) classForCoder
{
return NSArrayClass;
}
/**
* Returns YES if anObject belongs to self. No otherwise.<br />
* The [NSObject-isEqual:] method of anObject is used to test for equality.
*/
- (BOOL) containsObject: (id)anObject
{
return ([self indexOfObject: anObject] != NSNotFound);
}
/**
* Returns a new copy of the receiver.<br />
* The default abstract implementation of a copy is to use the
* -initWithArray:copyItems: method with the flag set to YES.<br />
* Immutable subclasses generally simply retain and return the receiver.
*/
- (id) copyWithZone: (NSZone*)zone
{
NSArray *copy = [NSArrayClass allocWithZone: zone];
return [copy initWithArray: self copyItems: YES];
}
/** <override-subclass />
* Returns the number of elements contained in the receiver.
*/
- (NSUInteger) count
{
[self subclassResponsibility: _cmd];
return 0;
}
- (NSUInteger) countByEnumeratingWithState: (NSFastEnumerationState*)state
objects: (__unsafe_unretained id[])stackbuf
count: (NSUInteger)len
{
NSUInteger size = [self count];
NSInteger count;
/* This is cached in the caller at the start and compared at each
* iteration. If it changes during the iteration then
* objc_enumerationMutation() will be called, throwing an exception.
*/
state->mutationsPtr = (unsigned long *)size;
count = MIN(len, size - state->state);
/* If a mutation has occurred then it's possible that we are being asked to
* get objects from after the end of the array. Don't pass negative values
* to memcpy.
*/
if (count > 0)
{
IMP imp = [self methodForSelector: @selector(objectAtIndex:)];
int p = state->state;
int i;
for (i = 0; i < count; i++, p++)
{
stackbuf[i] = (*imp)(self, @selector(objectAtIndex:), p);
}
state->state += count;
}
else
{
count = 0;
}
state->itemsPtr = stackbuf;
return count;
}
/**
* Encodes the receiver for storing to archive or sending over an
* [NSConnection].
*/
- (void) encodeWithCoder: (NSCoder*)aCoder
{
unsigned count = [self count];
if ([aCoder allowsKeyedCoding])
{
/* HACK ... MacOS-X seems to code differently if the coder is an
* actual instance of NSKeyedArchiver
*/
if ([aCoder class] == [NSKeyedArchiver class])
{
[(NSKeyedArchiver*)aCoder _encodeArrayOfObjects: self
forKey: @"NS.objects"];
}
else
{
unsigned i;
for (i = 0; i < count; i++)
{
NSString *key;
key = [NSString stringWithFormat: @"NS.object.%u", i];
[(NSKeyedArchiver*)aCoder encodeObject: [self objectAtIndex: i]
forKey: key];
}
}
}
else
{
[aCoder encodeValueOfObjCType: @encode(unsigned)
at: &count];
if (count > 0)
{
GS_BEGINIDBUF(a, count);
[self getObjects: a];
[aCoder encodeArrayOfObjCType: @encode(id)
count: count
at: a];
GS_ENDIDBUF();
}
}
}
/**
* Copies the objects from the receiver to aBuffer, which must be
* an area of memory large enough to hold them.
*/
- (void) getObjects: (__unsafe_unretained id[])aBuffer
{
unsigned i, c = [self count];
IMP get = [self methodForSelector: oaiSel];
for (i = 0; i < c; i++)
aBuffer[i] = (*get)(self, oaiSel, i);
}
/**
* Copies the objects from the range aRange of the receiver to aBuffer,
* which must be an area of memory large enough to hold them.
*/
- (void) getObjects: (__unsafe_unretained id[])aBuffer range: (NSRange)aRange
{
unsigned i, j = 0, c = [self count], e = aRange.location + aRange.length;
IMP get = [self methodForSelector: oaiSel];
GS_RANGE_CHECK(aRange, c);
for (i = aRange.location; i < e; i++)
aBuffer[j++] = (*get)(self, oaiSel, i);
}
/**
* Returns the same value as -count
*/
- (NSUInteger) hash
{
return [self count];
}
/**
* Returns the index of the specified object in the receiver, or
* NSNotFound if the object is not present.
*/
- (NSUInteger) indexOfObjectIdenticalTo: (id)anObject
{
unsigned c = [self count];
if (c > 0)
{
IMP get = [self methodForSelector: oaiSel];
unsigned i;
for (i = 0; i < c; i++)
if (anObject == (*get)(self, oaiSel, i))
return i;
}
return NSNotFound;
}
/**
* Returns the index of the specified object in the range of the receiver,
* or NSNotFound if the object is not present.
*/
- (NSUInteger) indexOfObjectIdenticalTo: anObject inRange: (NSRange)aRange
{
unsigned i, e = aRange.location + aRange.length, c = [self count];
IMP get = [self methodForSelector: oaiSel];
GS_RANGE_CHECK(aRange, c);
for (i = aRange.location; i < e; i++)
if (anObject == (*get)(self, oaiSel, i))
return i;
return NSNotFound;
}
/**
* Returns the index of the first object found in the receiver
* which is equal to anObject (using anObject's [NSObject-isEqual:] method).
* Returns NSNotFound on failure.
*/
- (NSUInteger) indexOfObject: (id)anObject
{
unsigned c = [self count];
if (c > 0 && anObject != nil)
{
unsigned i;
IMP get = [self methodForSelector: oaiSel];
BOOL (*eq)(id, SEL, id)
= (BOOL (*)(id, SEL, id))[anObject methodForSelector: eqSel];
for (i = 0; i < c; i++)
if ((*eq)(anObject, eqSel, (*get)(self, oaiSel, i)) == YES)
return i;
}
return NSNotFound;
}
/**
* Returns the index of the first object found in aRange of receiver
* which is equal to anObject (using anObject's [NSObject-isEqual:] method).
* Returns NSNotFound on failure.
*/
- (NSUInteger) indexOfObject: (id)anObject inRange: (NSRange)aRange
{
unsigned i, e = aRange.location + aRange.length, c = [self count];
IMP get = [self methodForSelector: oaiSel];
BOOL (*eq)(id, SEL, id)
= (BOOL (*)(id, SEL, id))[anObject methodForSelector: eqSel];
GS_RANGE_CHECK(aRange, c);
for (i = aRange.location; i < e; i++)
{
if ((*eq)(anObject, eqSel, (*get)(self, oaiSel, i)) == YES)
return i;
}
return NSNotFound;
}
/**
* <p>In MacOS-X class clusters do not have designated initialisers,
* and there is a general rule that -init is treated as the designated
* initialiser of the class cluster, but that other intitialisers
* may not work s expected an would need to be individually overridden
* in any subclass.
* </p>
* <p>GNUstep tries to make it easier to subclass a class cluster,
* by making class clusters follow the same convention as normal
* classes, so the designated initialiser is the <em>richest</em>
* initialiser. This means that all other initialisers call the
* documented designated initialiser (which calls -init only for
* MacOS-X compatibility), and anyone writing a subclass only needs
* to override that one initialiser in order to have all the other
* ones work.
* </p>
* <p>For MacOS-X compatibility, you may also need to override various
* other initialisers. Exactly which ones, you will need to determine
* by trial on a MacOS-X system ... and may vary between releases of
* MacOS-X. So to be safe, on MacOS-X you probably need to re-implement
* <em>all</em> the class cluster initialisers you might use in conjunction
* with your subclass.
* </p>
*/
- (id) init
{
self = [super init];
return self;
}
/**
* Initialize the receiver with the contents of array.
* The order of array is preserved.<br />
* If shouldCopy is YES then the objects are copied
* rather than simply retained.<br />
* Invokes -initWithObjects:count:
*/
- (id) initWithArray: (NSArray*)array copyItems: (BOOL)shouldCopy
{
unsigned c = [array count];
GS_BEGINIDBUF(objects, c);
if ([array isProxy])
{
unsigned i;
for (i = 0; i < c; i++)
{
objects[i] = [array objectAtIndex: i];
}
}
else
{
[array getObjects: objects];
}
if (shouldCopy == YES)
{
unsigned i;
for (i = 0; i < c; i++)
{
objects[i] = [objects[i] copy];
}
self = [self initWithObjects: objects count: c];
#if GS_WITH_GC == 0
while (i > 0)
{
[objects[--i] release];
}
#endif
}
else
{
self = [self initWithObjects: objects count: c];
}
GS_ENDIDBUF();
return self;
}
/**
* Initialize the receiver with the contents of array.
* The order of array is preserved.<br />
* Invokes -initWithObjects:count:
*/
- (id) initWithArray: (NSArray*)array
{
unsigned c = [array count];
GS_BEGINIDBUF(objects, c);
if ([array isProxy])
{
unsigned i;
for (i = 0; i < c; i++)
{
objects[i] = [array objectAtIndex: i];
}
}
else
{
[array getObjects: objects];
}
self = [self initWithObjects: objects count: c];
GS_ENDIDBUF();
return self;
}
/**
* Initialize the array by decoding from an archive.<br />
* Invokes -initWithObjects:count:
*/
- (id) initWithCoder: (NSCoder*)aCoder
{
if ([aCoder allowsKeyedCoding])
{
id array;
array = [(NSKeyedUnarchiver*)aCoder _decodeArrayOfObjectsForKey:
@"NS.objects"];
if (array == nil)
{
unsigned i = 0;
NSString *key;
id val;
array = [NSMutableArray arrayWithCapacity: 2];
key = [NSString stringWithFormat: @"NS.object.%u", i];
val = [(NSKeyedUnarchiver*)aCoder decodeObjectForKey: key];
while (val != nil)
{
[array addObject: val];
i++;
key = [NSString stringWithFormat: @"NS.object.%u", i];
val = [(NSKeyedUnarchiver*)aCoder decodeObjectForKey: key];
}
}
self = [self initWithArray: array];
}
else
{
unsigned count;
[aCoder decodeValueOfObjCType: @encode(unsigned)
at: &count];
if (count > 0)
{
GS_BEGINIDBUF(contents, count);
[aCoder decodeArrayOfObjCType: @encode(id)
count: count
at: contents];
self = [self initWithObjects: contents count: count];
#if GS_WITH_GC == 0
while (count-- > 0)
{
[contents[count] release];
}
#endif
GS_ENDIDBUF();
}
else
{
self = [self initWithObjects: 0 count: 0];
}
}
return self;
}
/**
* <p>Initialises the array with the contents of the specified file,
* which must contain an array in property-list format.
* </p>
* <p>In GNUstep, the property-list format may be either the OpenStep
* format (ASCII data), or the MacOS-X format (UTF-8 XML data) ... this
* method will recognise which it is.
* </p>
* <p>If there is a failure to load the file for any reason, the receiver
* will be released, the method will return nil, and a warning may be logged.
* </p>
* <p>Works by invoking [NSString-initWithContentsOfFile:] and
* [NSString-propertyList] then checking that the result is an array.
* </p>
*/
- (id) initWithContentsOfFile: (NSString*)file
{
NSString *myString;
myString = [[NSString allocWithZone: NSDefaultMallocZone()]
initWithContentsOfFile: file];
if (myString == nil)
{
DESTROY(self);
}
else
{
id result;
NS_DURING
{
result = [myString propertyList];
}
NS_HANDLER
{
result = nil;
}
NS_ENDHANDLER
RELEASE(myString);
if ([result isKindOfClass: NSArrayClass])
{
//self = [self initWithArray: result];
/* OSX appears to always return a mutable array rather than
* the class of the receiver.
*/
RELEASE(self);
self = RETAIN(result);
}
else
{
NSWarnMLog(@"Contents of file '%@' does not contain an array", file);
DESTROY(self);
}
}
return self;
}
/**
* <p>Initialises the array with the contents of the specified URL,
* which must contain an array in property-list format.
* </p>
* <p>In GNUstep, the property-list format may be either the OpenStep
* format (ASCII data), or the MacOS-X format (UTF8 XML data) ... this
* method will recognise which it is.
* </p>
* <p>If there is a failure to load the URL for any reason, the receiver
* will be released, the method will return nil, and a warning may be logged.
* </p>
* <p>Works by invoking [NSString-initWithContentsOfURL:] and
* [NSString-propertyList] then checking that the result is an array.
* </p>
*/
- (id) initWithContentsOfURL: (NSURL*)aURL
{
NSString *myString;
myString = [[NSString allocWithZone: NSDefaultMallocZone()]
initWithContentsOfURL: aURL];
if (myString == nil)
{
DESTROY(self);
}
else
{
id result;
NS_DURING
{
result = [myString propertyList];
}
NS_HANDLER
{
result = nil;
}
NS_ENDHANDLER
RELEASE(myString);
if ([result isKindOfClass: NSArrayClass])
{
self = [self initWithArray: result];
}
else
{
NSWarnMLog(@"Contents of URL '%@' does not contain an array", aURL);
DESTROY(self);
}
}
return self;
}
/** <init /> <override-subclass />
* This should initialize the array with count (may be zero) objects.<br />
* Retains each object placed in the array.<br />
* Calls -init (which does nothing but maintain MacOS-X compatibility),
* and needs to be re-implemented in subclasses in order to have all
* other initialisers work.
*/
- (id) initWithObjects: (const id[])objects count: (NSUInteger)count
{
self = [self init];
return self;
}
/**
* Initialize the array the list of objects.
* <br />May change the value of self before returning it.
*/
- (id) initWithObjects: firstObject, ...
{
GS_USEIDLIST(firstObject,
self = [self initWithObjects: __objects count: __count]);
return self;
}
/**
* Returns an NSMutableArray instance containing the same objects as
* the receiver.<br />
* The default implementation does this by calling the
* -initWithArray:copyItems: method on a newly created object,
* and passing it NO to tell it just to retain the items.
*/
- (id) mutableCopyWithZone: (NSZone*)zone
{
NSMutableArray *copy = [NSMutableArrayClass allocWithZone: zone];
return [copy initWithArray: self copyItems: NO];
}
/** <override-subclass />
* Returns the object at the specified index.
* Raises an exception of the index is beyond the array.
*/
- (id) objectAtIndex: (NSUInteger)index
{
[self subclassResponsibility: _cmd];
return nil;
}
- (NSArray *) objectsAtIndexes: (NSIndexSet *)indexes
{
//FIXME: probably slow!
NSMutableArray *group = [NSMutableArray arrayWithCapacity: [indexes count]];
NSUInteger i = [indexes firstIndex];
while (i != NSNotFound)
{
[group addObject: [self objectAtIndex: i]];
i = [indexes indexGreaterThanIndex: i];
}
return [group makeImmutableCopyOnFail: NO];
}
- (BOOL) isEqual: (id)anObject
{
if (self == anObject)
return YES;
if ([anObject isKindOfClass: NSArrayClass])
return [self isEqualToArray: anObject];
return NO;
}
/**
* Returns YES if the receiver is equal to otherArray, NO otherwise.
*/
- (BOOL) isEqualToArray: (NSArray*)otherArray
{
unsigned i, c;
if (self == (id)otherArray)
return YES;
c = [self count];
if (c != [otherArray count])
return NO;
if (c > 0)
{
IMP get0 = [self methodForSelector: oaiSel];
IMP get1 = [otherArray methodForSelector: oaiSel];
for (i = 0; i < c; i++)
if (![(*get0)(self, oaiSel, i) isEqual: (*get1)(otherArray, oaiSel, i)])
return NO;
}
return YES;
}
/**
* Returns the last object in the receiver, or nil if the receiver is empty.
*/
- (id) lastObject
{
unsigned count = [self count];
if (count == 0)
return nil;
return [self objectAtIndex: count-1];
}
/**
* Makes each object in the array perform aSelector.<br />
* This is done sequentially from the first to the last object.
*/
- (void) makeObjectsPerformSelector: (SEL)aSelector
{
unsigned c = [self count];
if (c > 0)
{
IMP get = [self methodForSelector: oaiSel];
unsigned i = 0;
while (i < c)
{
[(*get)(self, oaiSel, i++) performSelector: aSelector];
}
}
}
/**
* Obsolete version of -makeObjectsPerformSelector:
*/
- (void) makeObjectsPerform: (SEL)aSelector
{
[self makeObjectsPerformSelector: aSelector];
}
/**
* Makes each object in the array perform aSelector with arg.<br />
* This is done sequentially from the first to the last object.
*/
- (void) makeObjectsPerformSelector: (SEL)aSelector withObject: (id)arg
{
unsigned c = [self count];
if (c > 0)
{
IMP get = [self methodForSelector: oaiSel];
unsigned i = 0;
while (i < c)
{
[(*get)(self, oaiSel, i++) performSelector: aSelector
withObject: arg];
}
}
}
/**
* Obsolete version of -makeObjectsPerformSelector:withObject:
*/
- (void) makeObjectsPerform: (SEL)aSelector withObject: (id)argument
{
[self makeObjectsPerformSelector: aSelector withObject: argument];
}
static NSComparisonResult
compare(id elem1, id elem2, void* context)
{
NSComparisonResult (*imp)(id, SEL, id);
if (context == 0)
{
[NSException raise: NSInvalidArgumentException
format: @"compare null selector given"];
}
imp = (NSComparisonResult (*)(id, SEL, id))
[elem1 methodForSelector: context];
if (imp == NULL)
{
[NSException raise: NSGenericException
format: @"invalid selector passed to compare"];
}
return (*imp)(elem1, context, elem2);
}
/**
* Returns an autoreleased array in which the objects are ordered
* according to a sort with comparator.
*/
- (NSArray*) sortedArrayUsingSelector: (SEL)comparator
{
return [self sortedArrayUsingFunction: compare context: (void *)comparator];
}
/**
* Returns an autoreleased array in which the objects are ordered
* according to a sort with comparator. This invokes
* -sortedArrayUsingFunction:context:hint: with a nil hint.
*/
- (NSArray*) sortedArrayUsingFunction: (NSComparisonResult(*)(id,id,void*))comparator
context: (void*)context
{
return [self sortedArrayUsingFunction: comparator context: context hint: nil];
}
/**
* Subclasses may provide a hint for sorting ... The default GNUstep
* implementation just returns nil.
*/
- (NSData*) sortedArrayHint
{
return nil;
}
/**
* Returns an autoreleased array in which the objects are ordered
* according to a sort with comparator, where the comparator function
* is passed two objects to compare, and the context as the third
* argument. The hint argument is currently ignored, and may be nil.
*/
- (NSArray*) sortedArrayUsingFunction: (NSComparisonResult(*)(id,id,void*))comparator
context: (void*)context
hint: (NSData*)hint
{
NSMutableArray *sortedArray;
sortedArray = [[[NSMutableArrayClass allocWithZone:
NSDefaultMallocZone()] initWithArray: self copyItems: NO] autorelease];
[sortedArray sortUsingFunction: comparator context: context];
return [sortedArray makeImmutableCopyOnFail: NO];
}
/**
* Returns a string formed by concatenating the objects in the receiver,
* with the specified separator string inserted between each part.
*/
- (NSString*) componentsJoinedByString: (NSString*)separator
{
unsigned int c = [self count];
NSMutableString *s;
s = [[[NSMutableString alloc] initWithCapacity: c] autorelease];
if (c > 0)
{
unsigned l = [separator length];
unsigned i;
[s appendString: [[self objectAtIndex: 0] description]];
for (i = 1; i < c; i++)
{
if (l > 0)
{
[s appendString: separator];
}
[s appendString: [[self objectAtIndex: i] description]];
}
}
return [s makeImmutableCopyOnFail: NO];
}
/**
* Assumes that the receiver is an array of paths, and returns an
* array formed by selecting the subset of those patch matching
* the specified array of extensions.
*/
- (NSArray*) pathsMatchingExtensions: (NSArray*)extensions
{
unsigned i, c = [self count];
NSMutableArray *a = [[[NSMutableArray alloc] initWithCapacity: 1] autorelease];
Class cls = [NSString class];
IMP get = [self methodForSelector: oaiSel];
IMP add = [a methodForSelector: addSel];
for (i = 0; i < c; i++)
{
id o = (*get)(self, oaiSel, i);
if ([o isKindOfClass: cls])
{
if ([extensions containsObject: [o pathExtension]])
{
(*add)(a, addSel, o);
}
}
}
return [a makeImmutableCopyOnFail: NO];
}
/**
* Returns the first object found in the receiver (starting at index 0)
* which is present in the otherArray as determined by using the
* -containsObject: method.
*/
- (id) firstObjectCommonWithArray: (NSArray*)otherArray
{
unsigned i, c = [self count];
id o;
for (i = 0; i < c; i++)
{
if ([otherArray containsObject: (o = [self objectAtIndex: i])])
{
return o;
}
}
return nil;
}
/**
* Returns a subarray of the receiver containing the objects found in
* the specified range aRange.
*/
- (NSArray*) subarrayWithRange: (NSRange)aRange
{
id na;
unsigned c = [self count];
GS_RANGE_CHECK(aRange, c);
if (aRange.length == 0)
{
na = [NSArray array];
}
else
{
GS_BEGINIDBUF(objects, aRange.length);
[self getObjects: objects range: aRange];
na = [NSArray arrayWithObjects: objects count: aRange.length];
GS_ENDIDBUF();
}
return na;
}
/**
* Returns an enumerator describing the array sequentially
* from the first to the last element.<br/>
* If you use a mutable subclass of NSArray,
* you should not modify the array during enumeration.
*/
- (NSEnumerator*) objectEnumerator
{
id e;
e = [NSArrayEnumerator allocWithZone: NSDefaultMallocZone()];
e = [e initWithArray: self];
return AUTORELEASE(e);
}
/**
* Returns an enumerator describing the array sequentially
* from the last to the first element.<br/>
* If you use a mutable subclass of NSArray,
* you should not modify the array during enumeration.
*/
- (NSEnumerator*) reverseObjectEnumerator
{
id e;
e = [NSArrayEnumeratorReverse allocWithZone: NSDefaultMallocZone()];
e = [e initWithArray: self];
return AUTORELEASE(e);
}
/**
* Returns the result of invoking -descriptionWithLocale:indent: with a nil
* locale and zero indent.
*/
- (NSString*) description
{
return [self descriptionWithLocale: nil];
}
/**
* Returns the result of invoking -descriptionWithLocale:indent:
* with a zero indent.
*/
- (NSString*) descriptionWithLocale: (id)locale
{
return [self descriptionWithLocale: locale indent: 0];
}
/**
* Returns the receiver as a text property list in the traditional format.<br />
* See [NSString-propertyList] for details.<br />
* If locale is nil, no formatting is done, otherwise entries are formatted
* according to the locale, and indented according to level.<br />
* Unless locale is nil, a level of zero indents items by four spaces,
* while a level of one indents them by a tab.<br />
* The items in the property list string appear in the same order as
* they appear in the receiver.
*/
- (NSString*) descriptionWithLocale: (id)locale
indent: (NSUInteger)level
{
NSString *result = nil;
GSPropertyListMake(self, locale, NO, YES, level == 1 ? 3 : 2, &result);
return result;
}
/**
* <p>Writes the contents of the array to the file specified by path.
* The file contents will be in property-list format ... under GNUstep
* this is either OpenStep style (ASCII characters using \U hexadecimal
* escape sequences for unicode), or MacOS-X style (XML in the UTF8
* character set).
* </p>
* <p>If the useAuxiliaryFile flag is YES, the file write operation is
* atomic ... the data is written to a temporary file, which is then
* renamed to the actual file name.
* </p>
* <p>If the conversion of data into the correct property-list format fails
* or the write operation fails, the method returns NO, otherwise it
* returns YES.
* </p>
* <p>NB. The fact that the file is in property-list format does not
* necessarily mean that it can be used to reconstruct the array using
* the -initWithContentsOfFile: method. If the original array contains
* non-property-list objects, the descriptions of those objects will
* have been written, and reading in the file as a property-list will
* result in a new array containing the string descriptions.
* </p>
*/
- (BOOL) writeToFile: (NSString *)path atomically: (BOOL)useAuxiliaryFile
{
NSDictionary *loc;
NSString *desc = nil;
NSData *data;
loc = [[NSUserDefaults standardUserDefaults] dictionaryRepresentation];
if (GSMacOSXCompatiblePropertyLists() == YES)
{
GSPropertyListMake(self, loc, YES, NO, 2, &desc);
data = [desc dataUsingEncoding: NSUTF8StringEncoding];
}
else
{
GSPropertyListMake(self, loc, NO, NO, 2, &desc);
data = [desc dataUsingEncoding: NSASCIIStringEncoding];
}
return [data writeToFile: path atomically: useAuxiliaryFile];
}
/**
* <p>Writes the contents of the array to the specified url.
* This functions just like -writeToFile:atomically: except that the
* output may be written to any URL, not just a local file.
* </p>
*/
- (BOOL) writeToURL: (NSURL *)url atomically: (BOOL)useAuxiliaryFile
{
NSDictionary *loc;
NSString *desc = nil;
NSData *data;
loc = [[NSUserDefaults standardUserDefaults] dictionaryRepresentation];
if (GSMacOSXCompatiblePropertyLists() == YES)
{
GSPropertyListMake(self, loc, YES, NO, 2, &desc);
data = [desc dataUsingEncoding: NSUTF8StringEncoding];
}
else
{
GSPropertyListMake(self, loc, NO, NO, 2, &desc);
data = [desc dataUsingEncoding: NSASCIIStringEncoding];
}
return [data writeToURL: url atomically: useAuxiliaryFile];
}
/**
* This overrides NSObjects implementation of this method.
* This method returns an array of objects returned by
* invoking -valueForKey: for each item in the receiver,
* substituting NSNull for nil.
* A special case: the key "count" is not forwarded to each object
* of the receiver but returns the number of objects of the receiver.<br/>
*/
- (id) valueForKey: (NSString*)key
{
id result = nil;
if ([key isEqualToString: @"@count"] == YES)
{
result = [NSNumber numberWithUnsignedInt: [self count]];
}
else if ([key isEqualToString: @"count"] == YES)
{
GSOnceMLog(
@"[NSArray-valueForKey:] called wth 'count' is deprecated .. use '@count'");
result = [NSNumber numberWithUnsignedInt: [self count]];
}
else
{
NSMutableArray *results = nil;
static NSNull *null = nil;
unsigned i;
unsigned count = [self count];
volatile id object = nil;
results = [NSMutableArray arrayWithCapacity: count];
for (i = 0; i < count; i++)
{
id result;
object = [self objectAtIndex: i];
result = [object valueForKey: key];
if (result == nil)
{
if (null == nil)
{
null = RETAIN([NSNull null]);
}
result = null;
}
[results addObject: result];
}
result = results;
}
return result;
}
- (id) valueForKeyPath: (NSString*)path
{
id result = nil;
if ([path hasPrefix: @"@"])
{
NSRange r;
r = [path rangeOfString: @"."];
if (r.length == 0)
{
if ([path isEqualToString: @"@count"] == YES)
{
result = [NSNumber numberWithUnsignedInt: [self count]];
}
else
{
result = [self valueForKey: path];
}
}
else
{
NSString *op = [path substringToIndex: r.location];
NSString *rem = [path substringFromIndex: NSMaxRange(r)];
unsigned count = [self count];
if ([op isEqualToString: @"@count"] == YES)
{
result = [NSNumber numberWithUnsignedInt: count];
}
else if ([op isEqualToString: @"@avg"] == YES)
{
double d = 0;
if (count > 0)
{
NSEnumerator *e = [self objectEnumerator];
id o;
while ((o = [e nextObject]) != nil)
{
d += [[o valueForKeyPath: rem] doubleValue];
}
d /= count;
}
result = [NSNumber numberWithDouble: d];
}
else if ([op isEqualToString: @"@max"] == YES)
{
if (count > 0)
{
NSEnumerator *e = [self objectEnumerator];
id o;
while ((o = [e nextObject]) != nil)
{
o = [o valueForKeyPath: rem];
if (result == nil
|| [result compare: o] == NSOrderedAscending)
{
result = o;
}
}
}
}
else if ([op isEqualToString: @"@min"] == YES)
{
if (count > 0)
{
NSEnumerator *e = [self objectEnumerator];
id o;
while ((o = [e nextObject]) != nil)
{
o = [o valueForKeyPath: rem];
if (result == nil
|| [result compare: o] == NSOrderedDescending)
{
result = o;
}
}
}
}
else if ([op isEqualToString: @"@sum"] == YES)
{
double d = 0;
if (count > 0)
{
NSEnumerator *e = [self objectEnumerator];
id o;
while ((o = [e nextObject]) != nil)
{
d += [[o valueForKeyPath: rem] doubleValue];
}
}
result = [NSNumber numberWithDouble: d];
}
else if ([op isEqualToString: @"@distinctUnionOfArrays"] == YES)
{
if (count > 0)
{
NSEnumerator *e = [self objectEnumerator];
id o;
result = [NSMutableSet set];
while ((o = [e nextObject]) != nil)
{
o = [o valueForKeyPath: rem];
[result addObjectsFromArray: o];
}
result = [result allObjects];
}
else
{
result = [NSArray array];
}
}
else if ([op isEqualToString: @"@distinctUnionOfObjects"] == YES)
{
if (count > 0)
{
NSEnumerator *e = [self objectEnumerator];
id o;
result = [NSMutableSet set];
while ((o = [e nextObject]) != nil)
{
o = [o valueForKeyPath: rem];
[result addObject: o];
}
result = [result allObjects];
}
else
{
result = [NSArray array];
}
}
else if ([op isEqualToString: @"@distinctUnionOfSets"] == YES)
{
if (count > 0)
{
NSEnumerator *e = [self objectEnumerator];
id o;
result = [NSMutableSet set];
while ((o = [e nextObject]) != nil)
{
o = [o valueForKeyPath: rem];
[result addObjectsFromArray: [o allObjects]];
}
result = [result allObjects];
}
else
{
result = [NSArray array];
}
}
else if ([op isEqualToString: @"@unionOfArrays"] == YES)
{
if (count > 0)
{
NSEnumerator *e = [self objectEnumerator];
id o;
result = [GSMutableArray array];
while ((o = [e nextObject]) != nil)
{
o = [o valueForKeyPath: rem];
[result addObjectsFromArray: o];
}
[result makeImmutableCopyOnFail: NO];
}
else
{
result = [NSArray array];
}
}
else if ([op isEqualToString: @"@unionOfObjects"] == YES)
{
if (count > 0)
{
NSEnumerator *e = [self objectEnumerator];
id o;
result = [GSMutableArray array];
while ((o = [e nextObject]) != nil)
{
o = [o valueForKeyPath: rem];
[result addObject: o];
}
[result makeImmutableCopyOnFail: NO];
}
else
{
result = [NSArray array];
}
}
else if ([op isEqualToString: @"@unionOfSets"] == YES)
{
if (count > 0)
{
NSEnumerator *e = [self objectEnumerator];
id o;
result = [GSMutableArray array];
while ((o = [e nextObject]) != nil)
{
o = [o valueForKeyPath: rem];
[result addObjectsFromArray: [o allObjects]];
}
[result makeImmutableCopyOnFail: NO];
}
else
{
result = [NSArray array];
}
}
else
{
result = [super valueForKeyPath: path];
}
}
}
else
{
result = [super valueForKeyPath: path];
}
return result;
}
- (void) enumerateObjectsUsingBlock: (GSEnumeratorBlock)aBlock
{
[self enumerateObjectsWithOptions: 0 usingBlock: aBlock];
}
- (void) enumerateObjectsWithOptions: (NSEnumerationOptions)opts
usingBlock: (GSEnumeratorBlock)aBlock
{
NSUInteger count = 0;
BOOL shouldStop = NO;
id<NSFastEnumeration> enumerator = self;
/* If we are enumerating in reverse, use the reverse enumerator for fast
* enumeration. */
if (opts & NSEnumerationReverse)
{
enumerator = [self reverseObjectEnumerator];
}
FOR_IN (id, obj, enumerator)
CALL_BLOCK(aBlock, obj, count++, &shouldStop);
if (shouldStop)
{
return;
}
END_FOR_IN(enumerator)
}
- (void) enumerateObjectsAtIndexes: (NSIndexSet*)indexSet
options: (NSEnumerationOptions)opts
usingBlock: (GSEnumeratorBlock)block
{
[[self objectsAtIndexes: indexSet] enumerateObjectsWithOptions: opts
usingBlock: block];
}
- (NSIndexSet *) indexesOfObjectsWithOptions: (NSEnumerationOptions)opts
passingTest: (GSPredicateBlock)predicate
{
/* TODO: Concurrency. */
NSMutableIndexSet *set = [NSMutableIndexSet indexSet];
BOOL shouldStop = NO;
id<NSFastEnumeration> enumerator = self;
NSUInteger count = 0;
/* If we are enumerating in reverse, use the reverse enumerator for fast
* enumeration. */
if (opts & NSEnumerationReverse)
{
enumerator = [self reverseObjectEnumerator];
}
FOR_IN (id, obj, enumerator)
if (CALL_BLOCK(predicate, obj, count, &shouldStop))
{
/* TODO: It would be more efficient to collect an NSRange and only
* pass it to the index set when CALL_BLOCK returned NO. */
[set addIndex: count];
}
if (shouldStop)
{
return set;
}
count++;
END_FOR_IN(enumerator)
return set;
}
- (NSIndexSet*) indexesOfObjectsPassingTest: (GSPredicateBlock)predicate
{
return [self indexesOfObjectsWithOptions: 0 passingTest: predicate];
}
- (NSIndexSet*) indexesOfObjectsAtIndexes: (NSIndexSet*)indexSet
options: (NSEnumerationOptions)opts
passingTest: (GSPredicateBlock)predicate
{
return [[self objectsAtIndexes: indexSet]
indexesOfObjectsWithOptions: opts
passingTest: predicate];
}
- (NSUInteger)indexOfObjectWithOptions: (NSEnumerationOptions)opts
passingTest: (GSPredicateBlock)predicate
{
/* TODO: Concurrency. */
id<NSFastEnumeration> enumerator = self;
BOOL shouldStop = NO;
NSUInteger count = 0;
/* If we are enumerating in reverse, use the reverse enumerator for fast
* enumeration. */
if (opts & NSEnumerationReverse)
{
enumerator = [self reverseObjectEnumerator];
}
FOR_IN (id, obj, enumerator)
if (CALL_BLOCK(predicate, obj, count, &shouldStop))
{
return count;
}
if (shouldStop)
{
return NSNotFound;
}
count++;
END_FOR_IN(enumerator)
return NSNotFound;
}
- (NSUInteger) indexOfObjectPassingTest: (GSPredicateBlock)predicate
{
return [self indexOfObjectWithOptions: 0 passingTest: predicate];
}
- (NSUInteger)indexOfObjectAtIndexes: (NSIndexSet*)indexSet
options: (NSEnumerationOptions)opts
passingTest: (GSPredicateBlock)predicate
{
return [[self objectsAtIndexes: indexSet]
indexOfObjectWithOptions: 0
passingTest: predicate];
}
@end
/**
* <code>NSMutableArray</code> is the mutable version of [NSArray]. It
* provides methods for altering the contents of the array.
*/
@implementation NSMutableArray
+ (void) initialize
{
if (self == [NSMutableArray class])
{
}
}
+ (id) allocWithZone: (NSZone*)z
{
if (self == NSMutableArrayClass)
{
return NSAllocateObject(GSMutableArrayClass, 0, z);
}
else
{
return NSAllocateObject(self, 0, z);
}
}
+ (id) arrayWithObject: (id)anObject
{
NSMutableArray *obj = [self allocWithZone: NSDefaultMallocZone()];
obj = [obj initWithObjects: &anObject count: 1];
return AUTORELEASE(obj);
}
- (Class) classForCoder
{
return NSMutableArrayClass;
}
/** <init /> <override-subclass />
* Initialise the array with the specified capacity ... this
* should ensure that the array can have numItems added efficiently.<br />
* Calls -init (which does nothing but maintain MacOS-X compatibility),
* and needs to be re-implemented in subclasses in order to have all
* other initialisers work.
*/
- (id) initWithCapacity: (NSUInteger)numItems
{
self = [self init];
return self;
}
/** <override-subclass />
* Adds anObject at the end of the array, thus increasing the size of
* the array. The object is retained upon addition.
*/
- (void) addObject: (id)anObject
{
[self subclassResponsibility: _cmd];
}
/**
* Swaps the positions of two objects in the array. Raises an exception
* if either array index is out of bounds.
*/
- (void) exchangeObjectAtIndex: (NSUInteger)i1
withObjectAtIndex: (NSUInteger)i2
{
id tmp = [self objectAtIndex: i1];
IF_NO_GC([tmp retain];)
[self replaceObjectAtIndex: i1 withObject: [self objectAtIndex: i2]];
[self replaceObjectAtIndex: i2 withObject: tmp];
RELEASE(tmp);
}
/** <override-subclass />
* Places an object into the receiver at the specified location.<br />
* Raises an exception if given an array index which is too large.<br />
* The object is retained by the array.
*/
- (void) replaceObjectAtIndex: (NSUInteger)index withObject: (id)anObject
{
[self subclassResponsibility: _cmd];
}
/** Replaces the values in the receiver at the locations given by the
* indexes set with values from the objects array.
*/
- (void) replaceObjectsAtIndexes: (NSIndexSet *)indexes
withObjects: (NSArray *)objects
{
NSUInteger index = [indexes firstIndex];
NSEnumerator *enumerator = [objects objectEnumerator];
id object = [enumerator nextObject];
while (object != nil && index != NSNotFound)
{
[self replaceObjectAtIndex: index withObject: object];
object = [enumerator nextObject];
index = [indexes indexGreaterThanIndex: index];
}
}
/**
* Replaces objects in the receiver with those from anArray.<br />
* Raises an exception if given a range extending beyond the array.<br />
*/
- (void) replaceObjectsInRange: (NSRange)aRange
withObjectsFromArray: (NSArray*)anArray
{
id e, o;
if ([self count] < (aRange.location + aRange.length))
[NSException raise: NSRangeException
format: @"Replacing objects beyond end of array."];
[self removeObjectsInRange: aRange];
e = [anArray reverseObjectEnumerator];
while ((o = [e nextObject]))
[self insertObject: o atIndex: aRange.location];
}
/**
* Replaces objects in the receiver with some of those from anArray.<br />
* Raises an exception if given a range extending beyond the array.<br />
*/
- (void) replaceObjectsInRange: (NSRange)aRange
withObjectsFromArray: (NSArray*)anArray
range: (NSRange)anotherRange
{
[self replaceObjectsInRange: aRange
withObjectsFromArray: [anArray subarrayWithRange: anotherRange]];
}
/** <override-subclass />
* Inserts an object into the receiver at the specified location.<br />
* Raises an exception if given an array index which is too large.<br />
* The size of the array increases by one.<br />
* The object is retained by the array.
*/
- (void) insertObject: anObject atIndex: (NSUInteger)index
{
[self subclassResponsibility: _cmd];
}
/** Inserts the values from the objects array into the receiver at the
* locations given by the indexes set.<br />
* The values are inserted in the same order that they appear in the
* array.
*/
- (void) insertObjects: (NSArray *)objects atIndexes: (NSIndexSet *)indexes
{
NSUInteger index = [indexes firstIndex];
NSEnumerator *enumerator = [objects objectEnumerator];
id object = [enumerator nextObject];
while (object != nil && index != NSNotFound)
{
[self insertObject: object atIndex: index];
object = [enumerator nextObject];
index = [indexes indexGreaterThanIndex: index];
}
}
/** <override-subclass />
* Removes an object from the receiver at the specified location.<br />
* The size of the array decreases by one.<br />
* Raises an exception if given an array index which is too large.<br />
*/
- (void) removeObjectAtIndex: (NSUInteger)index
{
[self subclassResponsibility: _cmd];
}
/**
* Creates an autoreleased mutable array able to store at least numItems.
* See the -initWithCapacity: method.
*/
+ (id) arrayWithCapacity: (NSUInteger)numItems
{
return AUTORELEASE([[self allocWithZone: NSDefaultMallocZone()]
initWithCapacity: numItems]);
}
/**
* Override our superclass's designated initializer to go our's
*/
- (id) initWithObjects: (const id[])objects count: (NSUInteger)count
{
self = [self initWithCapacity: count];
if (count > 0)
{
unsigned i;
IMP add = [self methodForSelector: addSel];
for (i = 0; i < count; i++)
(*add)(self, addSel, objects[i]);
}
return self;
}
/**
* Removes the last object in the array. Raises an exception if the array
* is already empty.
*/
- (void) removeLastObject
{
unsigned count = [self count];
if (count == 0)
[NSException raise: NSRangeException
format: @"Trying to remove from an empty array."];
[self removeObjectAtIndex: count-1];
}
/**
* Removes all occurrences of anObject (found by pointer equality)
* from the receiver.
*/
- (void) removeObjectIdenticalTo: (id)anObject
{
unsigned i;
if (anObject == nil)
{
NSWarnMLog(@"attempt to remove nil object");
return;
}
i = [self count];
if (i > 0)
{
IMP rem = 0;
IMP get = [self methodForSelector: oaiSel];
while (i-- > 0)
{
id o = (*get)(self, oaiSel, i);
if (o == anObject)
{
if (rem == 0)
{
rem = [self methodForSelector: remSel];
}
(*rem)(self, remSel, i);
}
}
}
}
/**
* Removes all occurrences of anObject (found by the [NSObject-isEqual:] method
* of anObject) aRange in the receiver.
*/
- (void) removeObject: (id)anObject inRange: (NSRange)aRange
{
unsigned c;
unsigned s;
unsigned i;
if (anObject == nil)
{
NSWarnMLog(@"attempt to remove nil object");
return;
}
c = [self count];
s = aRange.location;
i = aRange.location + aRange.length;
if (i > c)
{
i = c;
}
if (i > s)
{
IMP rem = 0;
IMP get = [self methodForSelector: oaiSel];
BOOL (*eq)(id, SEL, id)
= (BOOL (*)(id, SEL, id))[anObject methodForSelector: eqSel];
while (i-- > s)
{
id o = (*get)(self, oaiSel, i);
if (o == anObject || (*eq)(anObject, eqSel, o) == YES)
{
if (rem == 0)
{
rem = [self methodForSelector: remSel];
/*
* We need to retain the object so that when we remove the
* first equal object we don't get left with a bad object
* pointer for later comparisons.
*/
IF_NO_GC([anObject retain];)
}
(*rem)(self, remSel, i);
}
}
#if GS_WITH_GC == 0
if (rem != 0)
{
RELEASE(anObject);
}
#endif
}
}
/**
* Removes all occurrences of anObject (found by pointer equality)
* from aRange in the receiver.
*/
- (void) removeObjectIdenticalTo: (id)anObject inRange: (NSRange)aRange
{
unsigned c;
unsigned s;
unsigned i;
if (anObject == nil)
{
NSWarnMLog(@"attempt to remove nil object");
return;
}
c = [self count];
s = aRange.location;
i = aRange.location + aRange.length;
if (i > c)
{
i = c;
}
if (i > s)
{
IMP rem = 0;
IMP get = [self methodForSelector: oaiSel];
while (i-- > s)
{
id o = (*get)(self, oaiSel, i);
if (o == anObject)
{
if (rem == 0)
{
rem = [self methodForSelector: remSel];
}
(*rem)(self, remSel, i);
}
}
}
}
/**
* Removes all occurrences of anObject (found by anObject's
* [NSObject-isEqual:] method) from the receiver.
*/
- (void) removeObject: (id)anObject
{
unsigned i;
if (anObject == nil)
{
NSWarnMLog(@"attempt to remove nil object");
return;
}
i = [self count];
if (i > 0)
{
IMP rem = 0;
IMP get = [self methodForSelector: oaiSel];
BOOL (*eq)(id, SEL, id)
= (BOOL (*)(id, SEL, id))[anObject methodForSelector: eqSel];
while (i-- > 0)
{
id o = (*get)(self, oaiSel, i);
if (o == anObject || (*eq)(anObject, eqSel, o) == YES)
{
if (rem == 0)
{
rem = [self methodForSelector: remSel];
/*
* We need to retain the object so that when we remove the
* first equal object we don't get left with a bad object
* pointer for later comparisons.
*/
IF_NO_GC([anObject retain];)
}
(*rem)(self, remSel, i);
}
}
#if GS_WITH_GC == 0
if (rem != 0)
{
RELEASE(anObject);
}
#endif
}
}
/**
* Removes all objects from the receiver, leaving an empty array.
*/
- (void) removeAllObjects
{
unsigned c = [self count];
if (c > 0)
{
IMP remLast = [self methodForSelector: rlSel];
while (c--)
{
(*remLast)(self, rlSel);
}
}
}
/**
* Adds each object from otherArray to the receiver, in first to last order.
*/
- (void) addObjectsFromArray: (NSArray*)otherArray
{
unsigned c = [otherArray count];
if (c > 0)
{
unsigned i;
IMP get = [otherArray methodForSelector: oaiSel];
IMP add = [self methodForSelector: addSel];
for (i = 0; i < c; i++)
(*add)(self, addSel, (*get)(otherArray, oaiSel, i));
}
}
/**
* Sets the contents of the receiver to be identical to the contents
* of otherArray.
*/
- (void) setArray: (NSArray *)otherArray
{
[self removeAllObjects];
[self addObjectsFromArray: otherArray];
}
/**
* Removes objects from the receiver at the indices supplied by an NSIndexSet
*/
- (void) removeObjectsAtIndexes: (NSIndexSet *)indexes
{
NSUInteger count = [indexes count];
NSUInteger indexArray[count];
[indexes getIndexes: indexArray
maxCount: count
inIndexRange: NULL];
[self removeObjectsFromIndices: indexArray
numIndices: count];
}
/**
* Supplied with a C array of indices containing count values, this method
* removes all corresponding objects from the receiver. The objects are
* removed in such a way that the removal is <em>safe</em> irrespective
* of the order in which they are specified in the indices array.
*/
- (void) removeObjectsFromIndices: (NSUInteger*)indices
numIndices: (NSUInteger)count
{
if (count > 0)
{
unsigned to = 0;
unsigned from = 0;
unsigned i;
GS_BEGINITEMBUF(sorted, count, NSUInteger);
while (from < count)
{
unsigned val = indices[from++];
i = to;
while (i > 0 && sorted[i-1] > val)
{
i--;
}
if (i == to)
{
sorted[to++] = val;
}
else if (sorted[i] != val)
{
unsigned j = to++;
if (sorted[i] < val)
{
i++;
}
while (j > i)
{
sorted[j] = sorted[j-1];
j--;
}
sorted[i] = val;
}
}
if (to > 0)
{
IMP rem = [self methodForSelector: remSel];
while (to--)
{
(*rem)(self, remSel, sorted[to]);
}
}
GS_ENDITEMBUF();
}
}
/**
* Removes from the receiver, all the objects present in otherArray,
* as determined by using the [NSObject-isEqual:] method.
*/
- (void) removeObjectsInArray: (NSArray*)otherArray
{
unsigned c = [otherArray count];
if (c > 0)
{
unsigned i;
IMP get = [otherArray methodForSelector: oaiSel];
IMP rem = [self methodForSelector: @selector(removeObject:)];
for (i = 0; i < c; i++)
(*rem)(self, @selector(removeObject:), (*get)(otherArray, oaiSel, i));
}
}
/**
* Removes all the objects in aRange from the receiver.
*/
- (void) removeObjectsInRange: (NSRange)aRange
{
unsigned i;
unsigned s = aRange.location;
unsigned c = [self count];
i = aRange.location + aRange.length;
if (c < i)
i = c;
if (i > s)
{
IMP rem = [self methodForSelector: remSel];
while (i-- > s)
{
(*rem)(self, remSel, i);
}
}
}
/**
* Sorts the array according to the supplied comparator.
*/
- (void) sortUsingSelector: (SEL)comparator
{
[self sortUsingFunction: compare context: (void *)comparator];
}
/**
* Sorts the array according to the supplied compare function
* with the context information.
*/
- (void) sortUsingFunction: (NSComparisonResult (*)(id,id,void*))compare
context: (void*)context
{
/* Shell sort algorithm taken from SortingInAction - a NeXT example */
#define STRIDE_FACTOR 3 // good value for stride factor is not well-understood
// 3 is a fairly good choice (Sedgewick)
unsigned int c;
unsigned int d;
unsigned int stride = 1;
BOOL found;
unsigned int count = [self count];
#ifdef GSWARN
BOOL badComparison = NO;
#endif
while (stride <= count)
{
stride = stride * STRIDE_FACTOR + 1;
}
while (stride > (STRIDE_FACTOR - 1))
{
// loop to sort for each value of stride
stride = stride / STRIDE_FACTOR;
for (c = stride; c < count; c++)
{
found = NO;
if (stride > c)
{
break;
}
d = c - stride;
while (!found) /* move to left until correct place */
{
id a = [self objectAtIndex: d + stride];
id b = [self objectAtIndex: d];
NSComparisonResult r;
r = (*compare)(a, b, context);
if (r < 0)
{
#ifdef GSWARN
if (r != NSOrderedAscending)
{
badComparison = YES;
}
#endif
IF_NO_GC(RETAIN(a));
[self replaceObjectAtIndex: d + stride withObject: b];
[self replaceObjectAtIndex: d withObject: a];
RELEASE(a);
if (stride > d)
{
break;
}
d -= stride; // jump by stride factor
}
else
{
#ifdef GSWARN
if (r != NSOrderedDescending && r != NSOrderedSame)
{
badComparison = YES;
}
#endif
found = YES;
}
}
}
}
#ifdef GSWARN
if (badComparison == YES)
{
NSWarnMLog(@"Detected bad return value from comparison");
}
#endif
}
/**
* Call setValue:forKey: on each of the receiver's items
* with the value and key.
*/
- (void) setValue: (id)value forKey: (NSString*)key
{
unsigned i;
unsigned count = [self count];
volatile id object = nil;
for (i = 0; i < count; i++)
{
object = [self objectAtIndex: i];
[object setValue: value
forKey: key];
}
}
@end
@implementation NSArrayEnumerator
- (id) initWithArray: (NSArray*)anArray
{
self = [super init];
if (self != nil)
{
array = anArray;
IF_NO_GC(RETAIN(array));
pos = 0;
get = [array methodForSelector: oaiSel];
cnt = (unsigned (*)(NSArray*, SEL))[array methodForSelector: countSel];
}
return self;
}
/**
* Returns the next object in the enumeration or nil if there are no more
* objects.<br />
* NB. modifying a mutable array during an enumeration can break things ...
* don't do it.
*/
- (id) nextObject
{
if (pos >= (*cnt)(array, countSel))
return nil;
return (*get)(array, oaiSel, pos++);
}
- (void) dealloc
{
RELEASE(array);
[super dealloc];
}
@end
@implementation NSArrayEnumeratorReverse
- (id) initWithArray: (NSArray*)anArray
{
self = [super initWithArray: anArray];
if (self != nil)
{
pos = (*cnt)(array, countSel);
}
return self;
}
/**
* Returns the next object in the enumeration or nil if there are no more
* objects.<br />
* NB. modifying a mutable array during an enumeration can break things ...
* don't do it.
*/
- (id) nextObject
{
if (pos == 0)
return nil;
return (*get)(array, oaiSel, --pos);
}
@end