1995-05-05 21:03:04 +00:00
|
|
|
/* NSCharacterSet - Character set holder
|
1997-01-09 17:02:56 +00:00
|
|
|
Copyright (C) 1995, 1996, 1997 Free Software Foundation, Inc.
|
1995-05-05 21:03:04 +00:00
|
|
|
|
|
|
|
Written by: Adam Fedor <fedor@boulder.colorado.edu>
|
|
|
|
Date: Apr 1995
|
|
|
|
|
1996-05-12 00:56:10 +00:00
|
|
|
This file is part of the GNUstep Base Library.
|
1995-05-05 21:03:04 +00:00
|
|
|
|
|
|
|
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., 675 Mass Ave, Cambridge, MA 02139, USA.
|
|
|
|
*/
|
|
|
|
|
1995-08-30 21:31:29 +00:00
|
|
|
#include <Foundation/NSBitmapCharSet.h>
|
|
|
|
#include <Foundation/NSException.h>
|
|
|
|
#include <Foundation/NSBundle.h>
|
|
|
|
#include <Foundation/NSData.h>
|
1996-09-02 13:53:26 +00:00
|
|
|
#include <Foundation/NSLock.h>
|
1995-05-05 21:03:04 +00:00
|
|
|
|
1997-09-13 17:52:31 +00:00
|
|
|
static NSString* NSCharacterSet_PATH = @"NSCharacterSets";
|
1997-09-18 14:56:47 +00:00
|
|
|
|
1997-05-03 17:25:25 +00:00
|
|
|
static NSString* gnustep_libdir =
|
|
|
|
#ifdef GNUSTEP_INSTALL_LIBDIR
|
|
|
|
@GNUSTEP_INSTALL_LIBDIR;
|
|
|
|
#else
|
|
|
|
nil;
|
|
|
|
#endif
|
1995-05-05 21:03:04 +00:00
|
|
|
|
1996-09-02 13:53:26 +00:00
|
|
|
/* A simple array for caching standard bitmap sets */
|
|
|
|
#define MAX_STANDARD_SETS 12
|
|
|
|
static NSCharacterSet* cache_set[MAX_STANDARD_SETS];
|
|
|
|
static NSLock* cache_lock = nil;
|
|
|
|
|
1995-05-05 21:03:04 +00:00
|
|
|
@implementation NSCharacterSet
|
|
|
|
|
1996-09-02 13:53:26 +00:00
|
|
|
+ (void) initialize
|
|
|
|
{
|
|
|
|
static BOOL one_time = NO;
|
|
|
|
|
|
|
|
if (one_time == NO)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
for (i = 0; i < MAX_STANDARD_SETS; i++)
|
|
|
|
cache_set[i] = 0;
|
|
|
|
one_time = YES;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
1995-08-30 21:31:29 +00:00
|
|
|
/* Provide a default object for allocation */
|
|
|
|
+ allocWithZone:(NSZone *)zone
|
|
|
|
{
|
|
|
|
return NSAllocateObject([NSBitmapCharSet self], 0, zone);
|
|
|
|
}
|
|
|
|
|
1995-05-05 21:03:04 +00:00
|
|
|
// Creating standard character sets
|
|
|
|
|
1996-09-02 13:53:26 +00:00
|
|
|
+ (NSCharacterSet *) _bitmapForSet: (NSString *)setname number: (int)number
|
1995-08-30 21:31:29 +00:00
|
|
|
{
|
1996-09-02 13:53:26 +00:00
|
|
|
NSCharacterSet* set;
|
1995-08-30 21:31:29 +00:00
|
|
|
NSString *path;
|
1997-05-03 17:25:25 +00:00
|
|
|
NSBundle *gstep_base_bundle = [NSBundle bundleWithPath: gnustep_libdir];
|
1995-08-30 21:31:29 +00:00
|
|
|
|
1996-09-02 13:53:26 +00:00
|
|
|
if (!cache_lock)
|
|
|
|
cache_lock = [NSLock new];
|
|
|
|
[cache_lock lock];
|
|
|
|
|
|
|
|
set = nil; /* Quiet warnings */
|
|
|
|
if (cache_set[number] == nil)
|
1997-05-03 17:16:10 +00:00
|
|
|
{
|
|
|
|
NS_DURING
|
1997-05-03 17:25:25 +00:00
|
|
|
path = [gstep_base_bundle pathForResource:setname
|
|
|
|
ofType:@"dat"
|
|
|
|
inDirectory:NSCharacterSet_PATH];
|
1997-09-18 14:56:47 +00:00
|
|
|
/* This is for testing purposes only! Look in uninstalled dir */
|
1996-09-02 13:53:26 +00:00
|
|
|
if (path == nil || [path length] == 0)
|
1997-05-03 17:16:10 +00:00
|
|
|
{
|
1997-09-18 14:56:47 +00:00
|
|
|
path = [@"../NSCharacterSets" stringByAppendingPathComponent:
|
|
|
|
setname];
|
|
|
|
path = [path stringByAppendingPathExtension: @"dat"];
|
1996-09-02 13:53:26 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (path == nil || [path length] == 0)
|
|
|
|
{
|
|
|
|
[NSException raise:NSGenericException
|
|
|
|
format:@"Could not find bitmap file %s", [setname cString]];
|
|
|
|
/* NOT REACHED */
|
|
|
|
}
|
|
|
|
|
|
|
|
set = [self characterSetWithBitmapRepresentation:
|
|
|
|
[NSData dataWithContentsOfFile: path]];
|
1996-09-07 17:56:16 +00:00
|
|
|
cache_set[number] = [set retain];
|
1996-09-02 13:53:26 +00:00
|
|
|
NS_HANDLER
|
|
|
|
[cache_lock unlock];
|
1996-09-02 13:58:02 +00:00
|
|
|
[localException raise];
|
1996-09-02 15:52:46 +00:00
|
|
|
abort (); /* quiet warnings about `set' clobbered by longjmp. */
|
1996-09-02 13:53:26 +00:00
|
|
|
NS_ENDHANDLER
|
1995-08-30 21:31:29 +00:00
|
|
|
}
|
1996-09-02 13:53:26 +00:00
|
|
|
else
|
|
|
|
set = cache_set[number];
|
1995-08-30 21:31:29 +00:00
|
|
|
|
1996-09-02 13:53:26 +00:00
|
|
|
[cache_lock unlock];
|
|
|
|
return set;
|
1995-08-30 21:31:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
1995-05-05 21:03:04 +00:00
|
|
|
+ (NSCharacterSet *)alphanumericCharacterSet
|
|
|
|
{
|
1996-09-02 13:53:26 +00:00
|
|
|
return [self _bitmapForSet:@"alphanumCharSet" number: 0];
|
1995-05-05 21:03:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
+ (NSCharacterSet *)controlCharacterSet
|
|
|
|
{
|
1996-09-02 13:53:26 +00:00
|
|
|
return [self _bitmapForSet:@"controlCharSet" number: 1];
|
1995-05-05 21:03:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
+ (NSCharacterSet *)decimalDigitCharacterSet
|
|
|
|
{
|
1996-09-02 13:53:26 +00:00
|
|
|
return [self _bitmapForSet:@"decimalCharSet" number: 2];
|
1995-05-05 21:03:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
+ (NSCharacterSet *)decomposableCharacterSet
|
|
|
|
{
|
1995-08-30 21:31:29 +00:00
|
|
|
fprintf(stderr, "Warning: Decomposable set not yet fully specified\n");
|
1996-09-02 13:53:26 +00:00
|
|
|
return [self _bitmapForSet:@"decomposableCharSet" number: 3];
|
1995-05-05 21:03:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
+ (NSCharacterSet *)illegalCharacterSet
|
|
|
|
{
|
1995-08-30 21:31:29 +00:00
|
|
|
fprintf(stderr, "Warning: Illegal set not yet fully specified\n");
|
1996-09-02 13:53:26 +00:00
|
|
|
return [self _bitmapForSet:@"illegalCharSet" number: 4];
|
1995-05-05 21:03:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
+ (NSCharacterSet *)letterCharacterSet
|
|
|
|
{
|
1996-09-02 13:53:26 +00:00
|
|
|
return [self _bitmapForSet:@"lettercharCharSet" number: 5];
|
1995-05-05 21:03:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
+ (NSCharacterSet *)lowercaseLetterCharacterSet
|
|
|
|
{
|
1996-09-02 13:53:26 +00:00
|
|
|
return [self _bitmapForSet:@"lowercaseCharSet" number: 6];
|
1995-05-05 21:03:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
+ (NSCharacterSet *)nonBaseCharacterSet
|
|
|
|
{
|
1996-09-02 13:53:26 +00:00
|
|
|
return [self _bitmapForSet:@"nonbaseCharSet" number: 7];
|
1995-05-05 21:03:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
+ (NSCharacterSet *)uppercaseLetterCharacterSet
|
|
|
|
{
|
1996-09-02 13:53:26 +00:00
|
|
|
return [self _bitmapForSet:@"uppercaseCharSet" number: 8];
|
1995-05-05 21:03:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
+ (NSCharacterSet *)whitespaceAndNewlineCharacterSet
|
|
|
|
{
|
1996-09-02 13:53:26 +00:00
|
|
|
return [self _bitmapForSet:@"whitespaceandnlCharSet" number: 9];
|
1995-05-05 21:03:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
+ (NSCharacterSet *)whitespaceCharacterSet
|
|
|
|
{
|
1996-09-02 13:53:26 +00:00
|
|
|
return [self _bitmapForSet:@"whitespaceCharSet" number: 10];
|
1995-05-05 21:03:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Creating custom character sets
|
|
|
|
|
|
|
|
+ (NSCharacterSet *)characterSetWithBitmapRepresentation:(NSData *)data
|
|
|
|
{
|
1995-08-30 21:31:29 +00:00
|
|
|
return [[[NSBitmapCharSet alloc] initWithBitmap:data] autorelease];
|
1995-05-05 21:03:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
+ (NSCharacterSet *)characterSetWithCharactersInString:(NSString *)aString
|
|
|
|
{
|
1995-08-30 21:31:29 +00:00
|
|
|
int i, length;
|
|
|
|
char *bytes;
|
|
|
|
NSMutableData *bitmap = [NSMutableData dataWithLength:BITMAP_SIZE];
|
|
|
|
|
|
|
|
if (!aString)
|
|
|
|
{
|
|
|
|
[NSException raise:NSInvalidArgumentException
|
|
|
|
format:@"Creating character set with nil string"];
|
|
|
|
/* NOT REACHED */
|
|
|
|
}
|
|
|
|
|
|
|
|
length = [aString length];
|
|
|
|
bytes = [bitmap mutableBytes];
|
|
|
|
for (i=0; i < length; i++)
|
|
|
|
{
|
|
|
|
unichar letter = [aString characterAtIndex:i];
|
|
|
|
SETBIT(bytes[letter/8], letter % 8);
|
|
|
|
}
|
|
|
|
|
|
|
|
return [self characterSetWithBitmapRepresentation:bitmap];
|
1995-05-05 21:03:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
+ (NSCharacterSet *)characterSetWithRange:(NSRange)aRange
|
|
|
|
{
|
1995-08-30 21:31:29 +00:00
|
|
|
int i;
|
|
|
|
char *bytes;
|
|
|
|
NSMutableData *bitmap = [NSMutableData dataWithLength:BITMAP_SIZE];
|
|
|
|
|
|
|
|
if (NSMaxRange(aRange) > UNICODE_SIZE)
|
|
|
|
{
|
|
|
|
[NSException raise:NSInvalidArgumentException
|
|
|
|
format:@"Specified range exceeds character set"];
|
|
|
|
/* NOT REACHED */
|
|
|
|
}
|
|
|
|
|
|
|
|
bytes = (char *)[bitmap mutableBytes];
|
|
|
|
for (i=aRange.location; i < NSMaxRange(aRange); i++)
|
|
|
|
SETBIT(bytes[i/8], i % 8);
|
|
|
|
|
|
|
|
return [self characterSetWithBitmapRepresentation:bitmap];
|
1995-05-05 21:03:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
- (NSData *)bitmapRepresentation
|
|
|
|
{
|
1995-08-30 21:31:29 +00:00
|
|
|
[self subclassResponsibility:_cmd];
|
1995-05-05 21:03:04 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
- (BOOL)characterIsMember:(unichar)aCharacter
|
|
|
|
{
|
1995-08-30 21:31:29 +00:00
|
|
|
[self subclassResponsibility:_cmd];
|
1995-05-05 21:03:04 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
- (NSCharacterSet *)invertedSet
|
|
|
|
{
|
1995-08-30 21:31:29 +00:00
|
|
|
int i, length;
|
|
|
|
char *bytes;
|
|
|
|
NSMutableData *bitmap = [[self bitmapRepresentation] mutableCopy];
|
1995-05-05 21:03:04 +00:00
|
|
|
|
1995-08-30 21:31:29 +00:00
|
|
|
length = [bitmap length];
|
|
|
|
bytes = [bitmap mutableBytes];
|
|
|
|
for (i=0; i < length; i++)
|
|
|
|
bytes[i] = ~bytes[i];
|
1995-05-05 21:03:04 +00:00
|
|
|
|
1995-08-30 21:31:29 +00:00
|
|
|
return [[self class] characterSetWithBitmapRepresentation:bitmap];
|
1995-05-05 21:03:04 +00:00
|
|
|
}
|
|
|
|
|
1995-08-30 21:31:29 +00:00
|
|
|
|
|
|
|
// NSCopying, NSMutableCopying
|
1995-05-05 21:03:04 +00:00
|
|
|
- (id)copyWithZone:(NSZone *)zone
|
|
|
|
{
|
|
|
|
if (NSShouldRetainWithZone(self, zone))
|
|
|
|
return [self retain];
|
|
|
|
else
|
1995-08-30 21:31:29 +00:00
|
|
|
return [super copyWithZone:zone];
|
1995-05-05 21:03:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
- (id)mutableCopyWithZone:(NSZone *)zone
|
|
|
|
{
|
1995-08-30 21:31:29 +00:00
|
|
|
NSData *bitmap;
|
|
|
|
bitmap = [self bitmapRepresentation];
|
|
|
|
return [[NSMutableBitmapCharSet allocWithZone:zone] initWithBitmap:bitmap];
|
1995-05-05 21:03:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
@end
|
|
|
|
|
|
|
|
@implementation NSMutableCharacterSet
|
|
|
|
|
1995-08-30 21:31:29 +00:00
|
|
|
/* Provide a default object for allocation */
|
|
|
|
+ allocWithZone:(NSZone *)zone
|
|
|
|
{
|
|
|
|
return NSAllocateObject([NSMutableBitmapCharSet self], 0, zone);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Override this from NSCharacterSet to create the correct class */
|
|
|
|
+ (NSCharacterSet *)characterSetWithBitmapRepresentation:(NSData *)data
|
|
|
|
{
|
|
|
|
return [[[NSMutableBitmapCharSet alloc] initWithBitmap:data] autorelease];
|
|
|
|
}
|
|
|
|
|
1995-05-05 21:03:04 +00:00
|
|
|
/* Mutable subclasses must implement ALL of these methods. */
|
|
|
|
- (void)addCharactersInRange:(NSRange)aRange
|
|
|
|
{
|
1995-08-30 21:31:29 +00:00
|
|
|
[self subclassResponsibility:_cmd];
|
1995-05-05 21:03:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
- (void)addCharactersInString:(NSString *)aString
|
|
|
|
{
|
1995-08-30 21:31:29 +00:00
|
|
|
[self subclassResponsibility:_cmd];
|
1995-05-05 21:03:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
- (void)formUnionWithCharacterSet:(NSCharacterSet *)otherSet
|
|
|
|
{
|
1995-08-30 21:31:29 +00:00
|
|
|
[self subclassResponsibility:_cmd];
|
1995-05-05 21:03:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
- (void)formIntersectionWithCharacterSet:(NSCharacterSet *)otherSet
|
|
|
|
{
|
1995-08-30 21:31:29 +00:00
|
|
|
[self subclassResponsibility:_cmd];
|
1995-05-05 21:03:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
- (void)removeCharactersInRange:(NSRange)aRange
|
|
|
|
{
|
1995-08-30 21:31:29 +00:00
|
|
|
[self subclassResponsibility:_cmd];
|
1995-05-05 21:03:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
- (void)removeCharactersInString:(NSString *)aString
|
|
|
|
{
|
1995-08-30 21:31:29 +00:00
|
|
|
[self subclassResponsibility:_cmd];
|
1995-05-05 21:03:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
- (void)invert
|
|
|
|
{
|
1995-08-30 21:31:29 +00:00
|
|
|
[self subclassResponsibility:_cmd];
|
1995-05-05 21:03:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// NSCopying, NSMutableCopying
|
|
|
|
- (id)copyWithZone:(NSZone *)zone
|
|
|
|
{
|
1995-08-30 21:31:29 +00:00
|
|
|
NSData *bitmap;
|
|
|
|
bitmap = [self bitmapRepresentation];
|
|
|
|
return [[NSBitmapCharSet allocWithZone:zone] initWithBitmap:bitmap];
|
1995-05-05 21:03:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
- (id)mutableCopyWithZone:(NSZone *)zone
|
|
|
|
{
|
1995-08-30 21:31:29 +00:00
|
|
|
return [super mutableCopyWithZone:zone];
|
1995-05-05 21:03:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
@end
|