libs-gdl2/EOControl/EOObserver.m
David Ayers f3e520ee9a 2003-03-30 David Ayers <d.ayers@inode.at>
* *.h/m, *.h/m):
	Used #include instead of depricated #import.  Avoid including
	entire library headers.  Use export macro where applicable.

	* EOControl/GNUmakefile: Removed EOKeyValueCodingBase.m/h and
	EOUndoMananger.h.  Added EOArrayDataSource.m/h, EODefines.h
	and EODepricated.h.
	* EOControl/EODefines.h: Updated for GDL2 & gnustep-make.
	* EOControl/EODebug.h: Updated.
	* EOControl/EODepricated.h: Updated for current state of GDL2.
	([NSObject +flushClassKeyBindings]): Added declaration.
	([EOClassDescription +setDelegate:]): Added declaration.
	([EOClassDescription +delegate]): Added declaration.
	(EOUndoManager): Moved declaration of interface here.
	* EOControl/EOKeyValueCoding.m
	([EOClassDescription +flushClassKeyBindings]): Added empty
	implementation.
	([NSObject takeStoredValuesFromDictionary:]): Cache EONull
	instance and use it instead of isKindOfClass:.
	* EOControl/EOEditingContext.m
	([EOEditingContext +initialize]): Tidied.
	([EOEditingContest -deleteObject]): Use NSUndoManager instead of
	EOUndoManager.
	* EOControl/EOQualifier.m
	([NSArray -filteredArrayUsingQualifier:]): Added minor
	optimization tweak.
	* EOControl/EOClassDescription.m: Removed inactive commented
	code.  Added private declerations of EOAccess methods to avoid
	compiler warnings.
	([EOClassDescription +initialize]): Tidied.
	([EOClassDescription -classDescriptionForClass:]): Use
	GSObjCName() instead of objc runtime routines.
	* EOControl/EOFault.m: Exchanged direct usages of ObjC runtime
	routines with NS/GSObjCRuntime abstraction API.
	([EOFault +superclass]): Ditto.
	([EOFault +targetClassForFault:]): Ditto.
	([EOFault -respondsToSelector:]): Ditto.
	([EOFault +initialize]): Cache static class variable.
	([EOFault +isKindOfClass]): Use static class variable.
	([EOFault +handlerForFault:]): Ditto.
	([EOFault +targetClassForFault:]): Ditto.
	([EOFault -dealloc]): Ditto.
	* EOControl/EOGenericRecord.m:
	([EOGenericRecord +initialize]): Tidied.
	* EOControl/EOKeyComparisonQualifier.m: Tidied documentation.
	* EOControl/EOKeyValueQualifier.m: Ditto.
	* EOControl/EONSAddOns.h/m: Added declarations to surpress
	compiler warnings.
	([NSObject -eoCompareOnName:]): Adjusted casts to surpress
	compiler warnings.
	(GSUseStrictWO451Compatibility): Added function.
	(GDL2GlobalLock, GDL2GlobalRecursive): Ditto.
	* EOControl/EONull: Remove implementations for foundation
	libraries without key value coding and fully rely on NSNull.
	Added assertions in all intance methods as instances should never
	be created.
	([EONull +allocWithZone:]) Corrected method name so it will
	actually be used.
	* EOControl/EOSortOrdering.m
	([NSArray sortedArrayUsingKeyOrderArray:]): Tidied.
	([NSMutableArray sortUsingKeyOrderArray:]): Ditto.
	([EONull compareAscending:]): Sync with referencs implementation.
	([EONull compareDescending:]): Ditto.
	([EONull compareCaseInsensitiveAscending:]): Ditto.
	([EONull compareCaseInsensitiveDescending:]): Ditto.

	* EOAccess/EOAdaptor.h: Added comment about API compatibility.
	* EOAccess/EOAdaptor.m ([EOAdaptor -contexts]): Return array of
	adaptor contexts rather tham GC-wrapper objects containing adaptor
	contexts.
	([EOAdaptor -databaseEncoding]): Use GSEncodingName() instead of
	GetEncodingName().
	* EOAccess/EODatabaseContext.m
	([EODatabaseContext -_turnFault:gid:editingContext:isComplete:]):
	Use GSObjCClass() instead of trying to access isa by dereferencing
	from id with incorrect member.
	* EOAccess/EOModel.m ([EOModel -entityNames]): Sort returned array
	to insure comparable output.
	* EOAccess/EOSQLExpression.m
	([EOSQLExpression sqlStringForArrayOfQualifiers:operation:]):
	Added cast to surpress compiler warning.
	* EOAccess/EOUtilities.m
	([EOObjectStoreCoordinator setModelGroup:]): Ditto.
	* EOAccess/EORelationship.h ([EORelationship -docComment]):
	Added declaration.
	* EOAccess/GNUmakefile: Added EODefines.h and EODepricated.h.

	* Tools/*.m: Use RCS_ID macro.
	* Tools/EOAttribute+GSDoc.h:
	([EOAttribute gsdocContentWithTagName:idPtr:]): Corrected Typo.
	* Tools/EOModel+GSDoc.h/m:
	([EOModel gsdocContentSplittedByEntities:idPtr:]): Ditto.
	* Tools/EORelationship+GSDoc.m:
	([EORelationship gsdocContentWithTagName:idPtr:]): Change variable
	type to supress compiler warnings.
	* Tools/eoutil.m (dump): Initialize variables to supress compiler
	warnings.
	* Tools/gsdoc-model.m: Include GSCategories.h to supress compiler
	warnings.
	(main): Added cast to supress compiler warning.  Fixed typo in
	method invocation.

2003-03-25  Stephane Corthesy  <stephane@sente.ch>

	* EOControl/EODefines.h: Added new file for export/win32 support.
	* EOControl/EODepricated.h: Added new file for depricated features.
	* EOControl/EOControl.h: Added EOArrayDataSource.h and EODefines.h.
	* EOControl/EOArrayDataSource.h/m: Added new files.  Some methods
	(<NSCoding> and qualifier bindings) are empty stubs.
	* EOControl/EODebug.h: Use export macro instead of explicit extern for
	function and symbol declarations.
	* EOControl/EOGlobalID.h: Ditto.
	* EOControl/EONull.h: Ditto.
	* EOControl/EOObjectStore.h: Ditto.
	* EOControl/EOOrQualifier.m: Replaced autorelease by AUTORELEASE
	and fixed typo.
	* EOControl/EOQualifier.m
	([NSArray -filteredArrayUsingQualifier:]): Implemented.
	* EOControl/EONSAddOns.m: Use volatile in some exception handlers
	(man longjmp for more info).
	* EOControl/EOSortOrdering.h/m
	([EOSortOrdering -copyWithZone:]): Implemented <NSCopying>.
	([EOSortOrdering -encodeWithKeyValueArchiver:]): Implemented.

	* EOAccess/EODefines.h: Added new file for export/win32 support.
	* EOAccess/EODepricated.h: Added new file for depricated features.
	* EOAccess/EOAccess.h: Added EODefines.h.
	* EOAccess/EOSQLExpression.h/m: Fixed typo for
	EOPrimaryKeyConstraintKey.
	* EOAccess/EOExpressionArray.h/m: Use volatile for variables usein
	in exception handlers. (man longjmp for more info)
	* EOAccess/EODatabase.h: Use export macro instead of explicit
	extern for function and symbol declarations.
	* EOAccess/EOEntity.h: Ditto.
	* EOAccess/EOModel.h: Ditto.
	* EOAccess/EOSchemaGeneration.h: Ditto.
	* EOAccess/EOSQLExpression.h: Ditto.
	* EOAccess/EOUtilities.h: Ditto.
	* Tools/eoutil.m (dump): Implemented use of -postinstall option.
	Corrected bug when getting adaptor's expression class.  Renamed
	symbol EOPrimaryKeyContraintsKey into EOPrimaryKeyConstraintsKey.


git-svn-id: svn+ssh://svn.gna.org/svn/gnustep/libs/gdl2/trunk@16298 72102866-910b-0410-8b05-ffd578937521
2003-03-31 00:24:15 +00:00

429 lines
8.7 KiB
Objective-C

/**
EOObserver.m <title>EOObserver</title>
Copyright (C) 2000 Free Software Foundation, Inc.
Author: Mirko Viviani <mirko.viviani@rccr.cremona.it>
Date: June 2000
$Revision$
$Date$
<abstract></abstract>
This file is part of the GNUstep Database Library.
<license>
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; see the file COPYING.LIB.
If not, write to the Free Software Foundation,
59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
</license>
**/
#include "config.h"
RCS_ID("$Id$")
#ifndef NeXT_Foundation_LIBRARY
#include <Foundation/NSObject.h>
#include <Foundation/NSMapTable.h>
#include <Foundation/NSNotification.h>
#include <Foundation/NSEnumerator.h>
#include <Foundation/NSRunLoop.h>
#include <Foundation/NSDebug.h>
#else
#include <Foundation/Foundation.h>
#endif
#include <EOControl/EOClassDescription.h>
#include <EOControl/EOKeyValueCoding.h>
#include <EOControl/EONull.h>
#include <EOControl/EOObserver.h>
#include <EOControl/EODebug.h>
#include <string.h>
@implementation NSObject (EOObserver)
- (void)willChange
{
EOFLOGObjectFnStart();
EOFLOGObjectLevelArgs(@"EOObserver", @"willChange self=%p", self);
[EOObserverCenter notifyObserversObjectWillChange: self];
EOFLOGObjectFnStop();
}
@end
@implementation EOObserverCenter
static NSMapTable *observersMap = NULL;
static NSMutableArray *omniscientObservers=nil;
static unsigned int notificationSuppressCount=0;
static id lastObject;
+ (void)initialize
{
observersMap = NSCreateMapTable(NSNonOwnedPointerMapKeyCallBacks, //No because it de-fault observed objects NSObjectMapKeyCallBacks,
NSObjectMapValueCallBacks,
32);
omniscientObservers = [NSMutableArray new];
lastObject = nil;
notificationSuppressCount = 0;
}
+ (void)addObserver: (id <EOObserving>)observer forObject: (id)object
{
NSMutableArray *observersArray;
observersArray = NSMapGet(observersMap, object);
if (observersArray == nil)
{
observersArray = [NSMutableArray arrayWithCapacity: 16];
[observersArray addObject: observer];
RETAIN(object); //because not owned by observersMap
NSMapInsert(observersMap, object, observersArray);
}
else
{
if ([observersArray containsObject: observer] == NO)
[observersArray addObject: observer];
}
}
+ (void)removeObserver: (id <EOObserving>)observer forObject: (id)object
{
NSMutableArray *observersArray;
observersArray = NSMapGet(observersMap, object);
if (observersArray)
{
[observersArray removeObject: observer];
if (![observersArray count])
{
NSMapRemove(observersMap, object);
RELEASE(object); //because not owned by observersMap
}
}
}
+ (void)notifyObserversObjectWillChange: (id)object
{
EOFLOGClassFnStart();
EOFLOGObjectLevelArgs(@"EOObserver", @"object=%p", object);
if (!notificationSuppressCount)
{
//EOFLOGObjectLevelArgs(@"EOObserver", @"object=%p", object);
EOFLOGObjectLevelArgs(@"EOObserver", @"object=%p lastObject=%p",
object, lastObject);
if (object == nil)
lastObject = nil;
else if (lastObject != object)
{
NSMutableArray *observersArray;
NSEnumerator *obsEnum;
id<EOObserving> observer;
lastObject = object;
observersArray = NSMapGet(observersMap, object);
EOFLOGObjectLevelArgs(@"EOObserver", @"observersArray count=%d",
[observersArray count]);
obsEnum = [observersArray objectEnumerator];
while ((observer = [obsEnum nextObject]))
[observer objectWillChange: object];
EOFLOGObjectLevelArgs(@"EOObserver", @"omniscientObservers count=%d",
[omniscientObservers count]);
obsEnum = [omniscientObservers objectEnumerator];
while ((observer = [obsEnum nextObject]))
[observer objectWillChange: nil];
}
}
EOFLOGClassFnStop();
}
+ (NSArray *)observersForObject: (id)object
{
return NSMapGet(observersMap, object);
}
+ (id)observerForObject: (id)object ofClass: (Class)targetClass
{
NSArray *observersArray;
observersArray = NSMapGet(observersMap, object);
if (observersArray)
{
NSEnumerator *obsEnum;
id observer;
obsEnum = [observersArray objectEnumerator];
while ((observer = [obsEnum nextObject]))
if ([observer isKindOfClass: targetClass])
return observer;
}
return nil;
}
+ (void)suppressObserverNotification
{
notificationSuppressCount++;
}
+ (void)enableObserverNotification
{
if (notificationSuppressCount)
notificationSuppressCount--;
else
{
NSLog(@"enableObserverNotification called more than suppressObserverNotification");
}
}
+ (unsigned int)observerNotificationSuppressCount
{
return notificationSuppressCount;
}
+ (void)addOmniscientObserver: (id <EOObserving>)observer
{
if ([omniscientObservers containsObject: observer] == NO)
[omniscientObservers addObject: observer];
}
+ (void)removeOmniscientObserver: (id <EOObserving>)observer
{
[omniscientObservers removeObject: observer];
}
@end
@implementation EODelayedObserver
- (void)objectWillChange: (id)subject
{
[[EODelayedObserverQueue defaultObserverQueue] enqueueObserver: self];
}
- (EOObserverPriority)priority
{
return EOObserverPriorityThird;
}
- (EODelayedObserverQueue *)observerQueue
{
return [EODelayedObserverQueue defaultObserverQueue];
}
- (void)subjectChanged
{
}
- (void)discardPendingNotification
{
[[EODelayedObserverQueue defaultObserverQueue] dequeueObserver: self];
}
@end
static EODelayedObserverQueue *observerQueue;
@implementation EODelayedObserverQueue
+ (EODelayedObserverQueue *)defaultObserverQueue
{
if (!observerQueue)
observerQueue = [[self alloc] init];
return observerQueue;
}
- init
{
if ((self == [super init]))
{
ASSIGN(_modes, [NSArray arrayWithObject: NSDefaultRunLoopMode]);
}
return self;
}
- (void)_notifyObservers
{
[self notifyObserversUpToPriority: EOObserverPrioritySixth];
}
- (void)enqueueObserver: (EODelayedObserver *)observer
{
EOObserverPriority priority = [observer priority];
if (priority == EOObserverPriorityImmediate)
[observer subjectChanged];
else
{
if (_queue[priority])
{
EODelayedObserver *obj = _queue[priority];
while (YES)
{
if (obj == observer)
return;
obj = obj->_next;
if (!obj)
{
obj->_next = observer;
if (priority > _highestNonEmptyQueue)
{
_highestNonEmptyQueue = priority;
_haveEntryInNotificationQueue = YES;
}
break;;
}
}
}
else
_queue[priority] = observer;
[[NSRunLoop currentRunLoop]
performSelector: @selector(_notifyObservers)
target: self
argument: nil
order: EOFlushDelayedObserversRunLoopOrdering
modes: _modes];
}
}
- (void)dequeueObserver: (EODelayedObserver *)observer
{
EOObserverPriority priority;
EODelayedObserver *obj, *last = nil;
if (!observer)
return;
priority = [observer priority];
obj = _queue[priority];
while (obj)
{
if (obj == observer)
{
if (last)
last->_next = obj->_next;
else
_queue[priority] = obj->_next;
if (!_queue[priority])
{
int i = priority;
if (priority >= _highestNonEmptyQueue)
{
for (; i > EOObserverPriorityImmediate; --i)
{
if (_queue[priority])
{
_highestNonEmptyQueue = priority;
break;
}
}
}
if (priority == EOObserverPriorityFirst
|| i == EOObserverPriorityImmediate)
{
_highestNonEmptyQueue = EOObserverPriorityImmediate;
_haveEntryInNotificationQueue = NO;
}
}
return;
}
last = obj;
obj = obj->_next;
}
}
- (void)notifyObserversUpToPriority: (EOObserverPriority)priority
{
int i = _highestNonEmptyQueue;
for (; i > EOObserverPriorityImmediate; i--)
{
EODelayedObserver *observer = _queue[i];
while (observer)
{
[observer subjectChanged];
observer = observer->_next;
}
}
}
- (void)setRunLoopModes: (NSArray *)modes
{
ASSIGN(_modes, modes);
}
- (NSArray *)runLoopModes
{
return _modes;
}
@end
@implementation EOObserverProxy
- initWithTarget: (id)target
action: (SEL)action
priority: (EOObserverPriority)priority
{
NSEmitTODO();
return [self notImplemented: _cmd]; //TODO
}
@end