2001-12-17 14:31:42 +00:00
|
|
|
|
/** NSClassDescription
|
2000-11-27 22:26:08 +00:00
|
|
|
|
Copyright (C) 2000 Free Software Foundation, Inc.
|
|
|
|
|
|
|
|
|
|
Written by: Richard Frith-Macdonald <rfm@gnu.org>
|
|
|
|
|
Date: 2000
|
|
|
|
|
|
|
|
|
|
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 Library 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 Library General Public
|
|
|
|
|
License along with this library; if not, write to the Free
|
|
|
|
|
Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111 USA.
|
2001-12-18 16:54:15 +00:00
|
|
|
|
|
|
|
|
|
<title>NSClassDescription class reference</title>
|
|
|
|
|
$Date$ $Revision$
|
2000-11-27 22:26:08 +00:00
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
#include <Foundation/NSClassDescription.h>
|
|
|
|
|
#include <Foundation/NSLock.h>
|
|
|
|
|
#include <Foundation/NSMapTable.h>
|
|
|
|
|
#include <Foundation/NSNotification.h>
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@implementation NSClassDescription
|
|
|
|
|
|
|
|
|
|
static NSRecursiveLock *mapLock = nil;
|
|
|
|
|
static NSMapTable *classMap;
|
|
|
|
|
|
2002-02-12 07:43:47 +00:00
|
|
|
|
/**
|
|
|
|
|
* Returns the class descriptuion for aClass. If there is no such description
|
|
|
|
|
* available, sends an NSClassDescriptionNeededForClassNotification (with
|
|
|
|
|
* aClass as its object) so that objects providing class descriptions can
|
|
|
|
|
* register one, and tries again to find one.<br />
|
|
|
|
|
* Returns nil if there is no description found.<br />
|
|
|
|
|
* Handles locking to ensure thread safety and ensures that the returned
|
|
|
|
|
* object will not be destroyed by other threads.
|
|
|
|
|
*/
|
2000-11-27 22:26:08 +00:00
|
|
|
|
+ (NSClassDescription*) classDescriptionForClass: (Class)aClass
|
|
|
|
|
{
|
|
|
|
|
NSClassDescription *description;
|
|
|
|
|
|
|
|
|
|
[mapLock lock];
|
|
|
|
|
description = NSMapGet(classMap, aClass);
|
|
|
|
|
if (description == nil)
|
|
|
|
|
{
|
|
|
|
|
NSNotificationCenter *nc;
|
|
|
|
|
|
2002-02-12 07:43:47 +00:00
|
|
|
|
/*
|
|
|
|
|
* As we have a recursive lock, we can ask other objects to
|
|
|
|
|
* supply descriptuions right now, without having to unlock
|
|
|
|
|
* and re-lock
|
|
|
|
|
*/
|
2000-11-27 22:26:08 +00:00
|
|
|
|
nc = [NSNotificationCenter defaultCenter];
|
|
|
|
|
[nc postNotificationName: NSClassDescriptionNeededForClassNotification
|
|
|
|
|
object: aClass];
|
|
|
|
|
description = NSMapGet(classMap, aClass);
|
|
|
|
|
}
|
|
|
|
|
RETAIN(description);
|
|
|
|
|
[mapLock unlock];
|
|
|
|
|
|
|
|
|
|
return AUTORELEASE(description);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
+ (void) initialize
|
|
|
|
|
{
|
|
|
|
|
if (self == [NSClassDescription class])
|
|
|
|
|
{
|
|
|
|
|
classMap = NSCreateMapTable(NSObjectMapKeyCallBacks,
|
|
|
|
|
NSObjectMapValueCallBacks, 100);
|
|
|
|
|
mapLock = [NSRecursiveLock new];
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2002-02-12 07:43:47 +00:00
|
|
|
|
/**
|
|
|
|
|
* Invalidates the cache of class descriptions so the new descriptions
|
|
|
|
|
* will be fetched as required and begin to refil the cache. You need
|
|
|
|
|
* this only if you suspect that a class description should have
|
|
|
|
|
* changed.
|
|
|
|
|
*/
|
2000-11-27 22:26:08 +00:00
|
|
|
|
+ (void) invalidateClassDescriptionCache
|
|
|
|
|
{
|
|
|
|
|
[mapLock lock];
|
|
|
|
|
NSResetMapTable(classMap);
|
|
|
|
|
[mapLock unlock];
|
|
|
|
|
}
|
|
|
|
|
|
2002-02-12 07:43:47 +00:00
|
|
|
|
/**
|
|
|
|
|
* Registers aDescription for aClass ... placing it in the cache and
|
|
|
|
|
* replacing any previous version.
|
|
|
|
|
*/
|
2000-11-27 22:26:08 +00:00
|
|
|
|
+ (void) registerClassDescription: (NSClassDescription*)aDescription
|
|
|
|
|
forClass: (Class)aClass
|
|
|
|
|
{
|
|
|
|
|
if (aDescription != nil && aClass != 0)
|
|
|
|
|
{
|
|
|
|
|
[mapLock lock];
|
|
|
|
|
NSMapInsert(classMap, aClass, aDescription);
|
|
|
|
|
[mapLock unlock];
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2002-02-12 07:43:47 +00:00
|
|
|
|
/** <override-subclass />
|
|
|
|
|
* Returns the attribute keys - default implementation returns nil.
|
|
|
|
|
*/
|
2000-11-27 22:26:08 +00:00
|
|
|
|
- (NSArray*) attributeKeys
|
|
|
|
|
{
|
|
|
|
|
return nil;
|
|
|
|
|
}
|
|
|
|
|
|
2002-02-12 07:43:47 +00:00
|
|
|
|
/** <override-subclass />
|
|
|
|
|
* Returns the inverse relationship keys - default implementation returns nil.
|
|
|
|
|
*/
|
2000-11-27 22:26:08 +00:00
|
|
|
|
- (NSString*) inverseForRelationshipKey: (NSString*)aKey
|
|
|
|
|
{
|
|
|
|
|
return nil;
|
|
|
|
|
}
|
|
|
|
|
|
2002-02-12 07:43:47 +00:00
|
|
|
|
/** <override-subclass />
|
|
|
|
|
* Returns the to many relationship keys - default implementation returns nil.
|
|
|
|
|
*/
|
2000-11-27 22:26:08 +00:00
|
|
|
|
- (NSArray*) toManyRelationshipKeys
|
|
|
|
|
{
|
|
|
|
|
return nil;
|
|
|
|
|
}
|
|
|
|
|
|
2002-02-12 07:43:47 +00:00
|
|
|
|
/** <override-subclass />
|
|
|
|
|
* Returns the to one relationship keys - default implementation returns nil.
|
|
|
|
|
*/
|
2000-11-27 22:26:08 +00:00
|
|
|
|
- (NSArray*) toOneRelationshipKeys
|
|
|
|
|
{
|
|
|
|
|
return nil;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@end
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2002-02-12 07:43:47 +00:00
|
|
|
|
@implementation NSObject(NSClassDescriptionPrimitives)
|
2000-11-27 22:26:08 +00:00
|
|
|
|
|
|
|
|
|
static Class NSClassDescriptionClass = 0;
|
|
|
|
|
|
2002-02-12 07:43:47 +00:00
|
|
|
|
/**
|
|
|
|
|
* Returns the attribute keys supplied by the
|
2002-02-12 12:33:57 +00:00
|
|
|
|
* <ref id="NSClassDescription" type="class"/>
|
2002-02-12 07:43:47 +00:00
|
|
|
|
* object for the receivers class.
|
|
|
|
|
*/
|
2000-11-27 22:26:08 +00:00
|
|
|
|
- (NSArray*) attributeKeys
|
|
|
|
|
{
|
|
|
|
|
return [[self classDescription] attributeKeys];
|
|
|
|
|
}
|
|
|
|
|
|
2002-02-12 07:43:47 +00:00
|
|
|
|
/**
|
|
|
|
|
* Returns the <ref id="NSClassDescription" type="class" />
|
|
|
|
|
* object for the receivers class.
|
|
|
|
|
*/
|
2000-11-27 22:26:08 +00:00
|
|
|
|
- (NSClassDescription*) classDescription
|
|
|
|
|
{
|
|
|
|
|
if (NSClassDescriptionClass == 0)
|
|
|
|
|
{
|
|
|
|
|
NSClassDescriptionClass = [NSClassDescription class];
|
|
|
|
|
}
|
|
|
|
|
return [NSClassDescriptionClass classDescriptionForClass: [self class]];
|
|
|
|
|
}
|
|
|
|
|
|
2002-02-12 07:43:47 +00:00
|
|
|
|
/**
|
|
|
|
|
* Returns the inverse relationship keys supplied by the
|
|
|
|
|
* <ref id="NSClassDescription" type="class" />
|
|
|
|
|
* object for the receivers class.
|
|
|
|
|
*/
|
2000-11-27 22:26:08 +00:00
|
|
|
|
- (NSString*) inverseForRelationshipKey: (NSString*)aKey
|
|
|
|
|
{
|
|
|
|
|
return [[self classDescription] inverseForRelationshipKey: aKey];
|
|
|
|
|
}
|
|
|
|
|
|
2002-02-12 07:43:47 +00:00
|
|
|
|
/**
|
|
|
|
|
* Returns the to many relationship keys supplied by the
|
|
|
|
|
* <ref id="NSClassDescription" type="class" />
|
|
|
|
|
* object for the receivers class.
|
|
|
|
|
*/
|
2000-11-27 22:26:08 +00:00
|
|
|
|
- (NSArray*) toManyRelationshipKeys
|
|
|
|
|
{
|
|
|
|
|
return [[self classDescription] toManyRelationshipKeys];
|
|
|
|
|
}
|
|
|
|
|
|
2002-02-12 07:43:47 +00:00
|
|
|
|
/**
|
|
|
|
|
* Returns the to one relationship keys supplied by the
|
|
|
|
|
* <ref id="NSClassDescription" type="class" />
|
|
|
|
|
* object for the receivers class.
|
|
|
|
|
*/
|
2000-11-27 22:26:08 +00:00
|
|
|
|
- (NSArray*) toOneRelationshipKeys
|
|
|
|
|
{
|
|
|
|
|
return [[self classDescription] toOneRelationshipKeys];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@end
|
|
|
|
|
|