1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240
|
/*
DocumentReadWrite.m
Copyright (c) 1995-1996, NeXT Software, Inc.
All rights reserved.
Author: Ali Ozer
You may freely copy, distribute and reuse the code in this example.
NeXT disclaims any warranty of any kind, expressed or implied,
as to its fitness for any particular use.
File I/O code... We define a few convenience methods on NSAttributedString as well.
*/
#import <AppKit/AppKit.h>
#import "Document.h"
#import "Preferences.h"
#import <sys/stat.h>
#define IgnoreRichText NO
#import <string.h> // For memcmp()...
@interface NSAttributedString(EditExtensions)
- (id)initWithRTF:(NSData *)data viewSize:(NSSize *)size hyphenationFactor:(float *)factor;
- (id)initWithRTFDFile:(NSString *)path viewSize:(NSSize *)size hyphenationFactor:(float *)factor;
- (BOOL)writeRTFDToFile:(NSString *)path updateFilenames:(BOOL)flag viewSize:(NSSize)size hyphenationFactor:(float)factor;
- (NSData *)RTFFromRange:(NSRange)range viewSize:(NSSize)size hyphenationFactor:(float)factor;
@end
@implementation NSAttributedString(EditExtensions)
static float defaultPadding(void) {
static float padding = -1;
if (padding < 0.0) {
NSTextContainer *container = [[NSTextContainer alloc] init];
padding = [container lineFragmentPadding];
[container release];
}
return padding;
}
- (id)initWithRTF:(NSData *)data viewSize:(NSSize *)size hyphenationFactor:(float *)factor {
NSDictionary *docAttrs;
if (self = [self initWithRTF:data documentAttributes:&docAttrs]) {
NSValue *value;
NSNumber *number;
if (size && (value = [docAttrs objectForKey:@"PaperSize"])) {
*size = [value sizeValue];
/* The size has the 12 pt padding from old Edit; compensate for that... Note that we should really be getting the margins here! */
if (size->width > 0 && size->height > 0) size->width = size->width - (6.0 * 2) + (defaultPadding() * 2);
}
if (factor && (number = [docAttrs objectForKey:@"HyphenationFactor"])) *factor = [number floatValue];
}
return self;
}
- (id)initWithRTFDFile:(NSString *)path viewSize:(NSSize *)size hyphenationFactor:(float *)factor {
NSDictionary *docAttrs;
if (self = [self initWithPath:path documentAttributes:&docAttrs]) {
NSValue *value;
NSNumber *number;
if (size && (value = [docAttrs objectForKey:@"PaperSize"])) {
*size = [value sizeValue];
/* The size has the 12 pt padding from old Edit; compensate for that... Note that we should really be getting the margins here! */
if (size->width > 0 && size->height > 0) size->width = size->width - (6.0 * 2) + (defaultPadding() * 2);
}
if (factor && (number = [docAttrs objectForKey:@"HyphenationFactor"])) *factor = [number floatValue];
}
return self;
}
- (BOOL)writeRTFDToFile:(NSString *)path updateFilenames:(BOOL)flag viewSize:(NSSize)size hyphenationFactor:(float)factor {
NSDictionary *dict = [NSDictionary dictionaryWithObjectsAndKeys:[NSValue valueWithSize:NSMakeSize(size.width + (6.0 * 2) - (defaultPadding() * 2), size.height)], @"PaperSize", [NSNumber numberWithFloat:factor], @"HyphenationFactor", nil];
NSFileWrapper *wrapper = [self RTFDFileWrapperFromRange:NSMakeRange(0, [self length]) documentAttributes:dict];
if (wrapper) {
return [wrapper writeToFile:path atomically:YES updateFilenames:flag];
} else {
return NO;
}
}
- (NSData *)RTFFromRange:(NSRange)range viewSize:(NSSize)size hyphenationFactor:(float)factor {
NSDictionary *dict = [NSDictionary dictionaryWithObjectsAndKeys:[NSValue valueWithSize:NSMakeSize(size.width + (6.0 * 2) - (defaultPadding() * 2), size.height)], @"PaperSize", [NSNumber numberWithFloat:factor], @"HyphenationFactor", nil];
return [self RTFFromRange:range documentAttributes:dict];
}
@end
@implementation Document(ReadWrite)
/* Loads from the specified path, sets encoding and textStorage. Note that if the file looks like RTF or RTFD, this method will open the file in rich text mode, regardless of the setting of encoding.
*/
- (BOOL)loadFromPath:(NSString *)fileName encoding:(int)encoding {
NSData *fileContentsAsData = nil;
NSDictionary *attrs;
BOOL isDirectory;
NSString *extension = [fileName pathExtension];
BOOL success = NO;
if (!(attrs = [[NSFileManager defaultManager] fileAttributesAtPath:fileName traverseLink:YES])) return NO;
isDirectory = [[attrs fileType] isEqualToString:NSFileTypeDirectory];
if (isDirectory) {
if (![@"rtfd" isEqual:extension]) return NO; /* If directory, should be .rtfd */
encoding = RichTextWithGraphicsStringEncoding;
} else if ([@"rtf" isEqual:extension] && !IgnoreRichText) { /* If file looks like RTF, ignore any choice they made */
encoding = RichTextStringEncoding;
} else if (encoding == UnknownStringEncoding) {
if (fileContentsAsData = [[NSData alloc] initWithContentsOfFile:fileName]) {
unsigned len = [fileContentsAsData length];
const unsigned char *bytes = [fileContentsAsData bytes];
static const unsigned char bigUnicodeHeader[] = {0xff, 0xfe};
static const unsigned char littleUnicodeHeader[] = {0xfe, 0xff};
static const unsigned char rtfHeader[] = {'{', '\\', 'r', 't', 'f'};
/* Unicode plain text files start with the Unicode BOM char; check for that first... */
if (((len & 1) == 0) && (len >= 2) && (!memcmp(bytes, bigUnicodeHeader, 2) || !memcmp(bytes, littleUnicodeHeader, 2))) {
encoding = NSUnicodeStringEncoding;
} else if (((len >= 6) && !memcmp(bytes, rtfHeader, 5)) && !IgnoreRichText) {
encoding = RichTextStringEncoding;
}
if (encoding == UnknownStringEncoding) {
encoding = [[Preferences objectForKey:PlainTextEncoding] intValue];
if (encoding == UnknownStringEncoding) {
encoding = [NSString defaultCStringEncoding];
}
}
} else {
return NO; /* File couldn't be opened, I guess */
}
}
if (encoding == RichTextWithGraphicsStringEncoding) {
NSSize size = NSZeroSize;
float factor = 0.0;
NSTextStorage *newTextStorage = [[NSTextStorage allocWithZone:[self zone]] initWithRTFDFile:fileName viewSize:&size hyphenationFactor:&factor];
if (newTextStorage) {
[self setRichText:YES];
if (size.width > 0 && size.height > 0 && ![self hasMultiplePages]) [self setViewSize:size];
[self setHyphenationFactor:factor];
[[self layoutManager] replaceTextStorage:newTextStorage];
[textStorage release];
textStorage = newTextStorage;
success = YES;
}
} else {
if (!fileContentsAsData) fileContentsAsData = [[NSData alloc] initWithContentsOfFile:fileName];
if (fileContentsAsData) {
if (encoding == RichTextStringEncoding) {
NSSize size = NSZeroSize;
float factor = 0.0;
NSTextStorage *newTextStorage = [[NSTextStorage allocWithZone:[self zone]] initWithRTF:fileContentsAsData viewSize:&size hyphenationFactor:&factor];
if (newTextStorage) {
[self setRichText:YES];
if (size.width > 0 && size.height > 0 && ![self hasMultiplePages]) [self setViewSize:size];
[self setHyphenationFactor:factor];
[[self layoutManager] replaceTextStorage:newTextStorage];
[textStorage release];
textStorage = newTextStorage;
success = YES;
}
} else {
NSString *fileContents = [[NSString alloc] initWithData:fileContentsAsData encoding:encoding];
if (fileContents) {
[textStorage beginEditing];
[[textStorage mutableString] setString:fileContents];
[self setRichText:NO];
[textStorage endEditing];
[fileContents release];
encodingIfPlainText = encoding;
success = YES;
}
}
}
}
[fileContentsAsData release];
return success;
}
- (BOOL)saveToPath:(NSString *)fileName encoding:(int)encoding updateFilenames:(BOOL)updateFileNamesFlag {
BOOL success = NO;
NSFileManager *fileManager = [NSFileManager defaultManager];
NSDictionary *curAttributes = [fileManager fileAttributesAtPath:fileName traverseLink:YES];
NSString *actualFileNameToSave = [fileName stringByResolvingSymlinksInPath]; /* Follow links to save */
if (curAttributes) { /* If not nil, the file exists... */
NSString *backupFileName = [actualFileNameToSave stringByAppendingString:@"~"];
/* If there was a backup file name, delete it */
if ([fileManager fileExistsAtPath:backupFileName]) {
(void)[fileManager removeFileAtPath:backupFileName handler:nil];
}
/* If the user wishes to keep backups, simply remove the old file aside */
if (![[Preferences objectForKey:DeleteBackup] boolValue]) {
(void)[fileManager movePath:actualFileNameToSave toPath:backupFileName handler:nil];
}
}
switch (encoding) {
case RichTextWithGraphicsStringEncoding: {
success = [textStorage writeRTFDToFile:actualFileNameToSave updateFilenames:updateFileNamesFlag viewSize:[self viewSize] hyphenationFactor:[self hyphenationFactor]];
break;
}
case RichTextStringEncoding: {
NSData *data = [textStorage RTFFromRange:NSMakeRange(0, [textStorage length]) viewSize:[self viewSize] hyphenationFactor:[self hyphenationFactor]];
success = data && [data writeToFile:actualFileNameToSave atomically:YES];
break;
}
default: {
NSData *data = [[textStorage string] dataUsingEncoding:encoding];
success = data && [data writeToFile:actualFileNameToSave atomically:YES];
break;
}
}
/* Apply the original permissions to the new file. Also make it writable if needed. */
if (success && curAttributes) {
id permissions = [curAttributes objectForKey:NSFilePosixPermissions];
if (permissions) {
if ([[Preferences objectForKey:SaveFilesWritable] boolValue]) {
permissions = [NSNumber numberWithUnsignedLong:([permissions unsignedLongValue] | 0200)];
}
[fileManager changeFileAttributes:[NSDictionary dictionaryWithObjectsAndKeys:permissions, NSFilePosixPermissions, nil] atPath:actualFileNameToSave];
}
}
return success;
}
@end
/*
2/21/95 aozer Created for Edit II.
4/11/95 aozer Added some preliminary paper size support
4/13/95 aozer Permissions
7/21/95 aozer Follow links correctly for saving
11/7/96 aozer Read/write hyphenation factor in RTF files
*/
|