mirror of
https://github.com/gnustep/libs-gui.git
synced 2025-04-29 19:11:08 +00:00
Fun fixes.
git-svn-id: svn+ssh://svn.gna.org/svn/gnustep/libs/gui/trunk@4877 72102866-910b-0410-8b05-ffd578937521
This commit is contained in:
parent
4026532377
commit
3d2b84054e
6 changed files with 697 additions and 639 deletions
10
ChangeLog
10
ChangeLog
|
@ -1,4 +1,12 @@
|
||||||
1999-09-10 Pedro Ivo Andrade Tavares <ptavares@iname.com>
|
1999-09-11 Michael Hanni <mhanni@sprintmail.com>
|
||||||
|
|
||||||
|
* Source/NSMenu.m: many small fixes and optimizations for quicker
|
||||||
|
display. In addition several small fixes which should improve
|
||||||
|
functionality as well as usablity.
|
||||||
|
* Source/NSMenuView.m: ditto.
|
||||||
|
* Source/NSHelpManager.m: a small bugfix to allow for compilation.
|
||||||
|
|
||||||
|
1999-09-11 Pedro Ivo Andrade Tavares <ptavares@iname.com>
|
||||||
|
|
||||||
* NSResponder.m ([NSResponder -helpRequested:]):
|
* NSResponder.m ([NSResponder -helpRequested:]):
|
||||||
Now the application exits context help mode regardless of whether
|
Now the application exits context help mode regardless of whether
|
||||||
|
|
|
@ -17,148 +17,149 @@
|
||||||
|
|
||||||
+ (NSArray *)readableTypes
|
+ (NSArray *)readableTypes
|
||||||
{
|
{
|
||||||
return [[NSDocumentController sharedDocumentController]
|
return [[NSDocumentController sharedDocumentController]
|
||||||
_editorAndViewerTypesForClass:self];
|
_editorAndViewerTypesForClass:self];
|
||||||
}
|
}
|
||||||
|
|
||||||
+ (NSArray *)writableTypes
|
+ (NSArray *)writableTypes
|
||||||
{
|
{
|
||||||
return [[NSDocumentController sharedDocumentController] _editorTypesForClass:self];
|
return [[NSDocumentController sharedDocumentController] _editorTypesForClass:self];
|
||||||
}
|
}
|
||||||
|
|
||||||
+ (BOOL)isNativeType:(NSString *)type
|
+ (BOOL)isNativeType:(NSString *)type
|
||||||
{
|
{
|
||||||
return ([[self readableTypes] containsObject:type] &&
|
return ([[self readableTypes] containsObject:type] &&
|
||||||
[[self writableTypes] containsObject:type]);
|
[[self writableTypes] containsObject:type]);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
- (id)init
|
- (id)init
|
||||||
{
|
{
|
||||||
static int untitledCount = 1;
|
static int untitledCount = 1;
|
||||||
|
|
||||||
[super init];
|
[super init];
|
||||||
_documentIndex = untitledCount++;
|
_documentIndex = untitledCount++;
|
||||||
return self;
|
return self;
|
||||||
}
|
}
|
||||||
|
|
||||||
- (id)initWithContentsOfFile:(NSString *)fileName ofType:(NSString *)fileType
|
- (id)initWithContentsOfFile:(NSString *)fileName ofType:(NSString *)fileType
|
||||||
{
|
{
|
||||||
[super init];
|
[super init];
|
||||||
|
|
||||||
if ([self readFromFile:fileName ofType:fileType])
|
if ([self readFromFile:fileName ofType:fileType])
|
||||||
{
|
{
|
||||||
[self setFileType:fileType];
|
[self setFileType:fileType];
|
||||||
[self setFileName:fileName];
|
[self setFileName:fileName];
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
[self release];
|
[self release];
|
||||||
return nil;
|
return nil;
|
||||||
}
|
}
|
||||||
|
|
||||||
return self;
|
return self;
|
||||||
}
|
}
|
||||||
|
|
||||||
- (id)initWithContentsOfURL:(NSURL *)url ofType:(NSString *)fileType
|
- (id)initWithContentsOfURL:(NSURL *)url ofType:(NSString *)fileType
|
||||||
{
|
{
|
||||||
[super init];
|
[super init];
|
||||||
|
|
||||||
if ([self readFromURL:url ofType:fileType])
|
if ([self readFromURL:url ofType:fileType])
|
||||||
{
|
{
|
||||||
[self setFileType:fileType];
|
[self setFileType:fileType];
|
||||||
[self setFileName:[url path]];
|
[self setFileName:[url path]];
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
[self release];
|
[self release];
|
||||||
return nil;
|
return nil;
|
||||||
}
|
}
|
||||||
|
|
||||||
return self;
|
return self;
|
||||||
}
|
}
|
||||||
|
|
||||||
- (void)dealloc
|
- (void)dealloc
|
||||||
{
|
{
|
||||||
[[NSNotificationCenter defaultCenter] removeObserver:self];
|
[[NSNotificationCenter defaultCenter] removeObserver:self];
|
||||||
[(NSObject*)_undoManager release];
|
[(NSObject*)_undoManager release];
|
||||||
[_fileName release];
|
[_fileName release];
|
||||||
[_fileType release];
|
[_fileType release];
|
||||||
[_windowControllers release];
|
[_windowControllers release];
|
||||||
[_window release];
|
[_window release];
|
||||||
[_printInfo release];
|
[_printInfo release];
|
||||||
[savePanelAccessory release];
|
[savePanelAccessory release];
|
||||||
[spaButton release];
|
[spaButton release];
|
||||||
[super dealloc];
|
[super dealloc];
|
||||||
}
|
}
|
||||||
|
|
||||||
- (NSString *)fileName
|
- (NSString *)fileName
|
||||||
{
|
{
|
||||||
return _fileName;
|
return _fileName;
|
||||||
}
|
}
|
||||||
|
|
||||||
- (void)setFileName:(NSString *)fileName
|
- (void)setFileName:(NSString *)fileName
|
||||||
{
|
{
|
||||||
[fileName retain];
|
[fileName retain];
|
||||||
[_fileName release];
|
[_fileName release];
|
||||||
_fileName = fileName;
|
_fileName = fileName;
|
||||||
|
|
||||||
[_windowControllers makeObjectsPerformSelector:
|
[_windowControllers makeObjectsPerformSelector:
|
||||||
@selector(_synchronizeWindowTitleWithDocumentName)];
|
@selector(_synchronizeWindowTitleWithDocumentName)];
|
||||||
}
|
}
|
||||||
|
|
||||||
- (NSString *)fileType
|
- (NSString *)fileType
|
||||||
{
|
{
|
||||||
return _fileType;
|
return _fileType;
|
||||||
}
|
}
|
||||||
|
|
||||||
- (void)setFileType:(NSString *)type
|
- (void)setFileType:(NSString *)type
|
||||||
{
|
{
|
||||||
[type retain];
|
[type retain];
|
||||||
[_fileType release];
|
[_fileType release];
|
||||||
_fileType = type;
|
_fileType = type;
|
||||||
}
|
}
|
||||||
|
|
||||||
- (NSArray *)windowControllers
|
- (NSArray *)windowControllers
|
||||||
{
|
{
|
||||||
return _windowControllers;
|
return _windowControllers;
|
||||||
}
|
}
|
||||||
|
|
||||||
- (void)addWindowController:(NSWindowController *)windowController
|
- (void)addWindowController:(NSWindowController *)windowController
|
||||||
{
|
{
|
||||||
if (_windowControllers == nil) _windowControllers = [[NSMutableArray alloc] init];
|
if (_windowControllers == nil) _windowControllers = [[NSMutableArray alloc] init];
|
||||||
|
|
||||||
[_windowControllers addObject:windowController];
|
[_windowControllers addObject:windowController];
|
||||||
if ([windowController document] != self)
|
if ([windowController document] != self)
|
||||||
[windowController setDocument:self];
|
[windowController setDocument:self];
|
||||||
}
|
}
|
||||||
|
|
||||||
- (void)_removeWindowController:(NSWindowController *)windowController
|
- (void)_removeWindowController:(NSWindowController *)windowController
|
||||||
{
|
{
|
||||||
if ([_windowControllers containsObject:windowController])
|
if ([_windowControllers containsObject:windowController])
|
||||||
{
|
{
|
||||||
BOOL autoClose = [windowController shouldCloseDocument];
|
BOOL autoClose = [windowController shouldCloseDocument];
|
||||||
|
|
||||||
[windowController setDocument:nil];
|
[windowController setDocument:nil];
|
||||||
[_windowControllers removeObject:windowController];
|
[_windowControllers removeObject:windowController];
|
||||||
|
|
||||||
if (autoClose || [_windowControllers count] == 0)
|
if (autoClose || [_windowControllers count] == 0)
|
||||||
{
|
{
|
||||||
[self close];
|
[self close];
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
- (NSString *)windowNibName
|
- (NSString *)windowNibName
|
||||||
{
|
{
|
||||||
return nil;
|
return nil;
|
||||||
}
|
}
|
||||||
|
|
||||||
// private; called during nib load. // we do not retain the window, since it should
|
// private; called during nib load.
|
||||||
|
// we do not retain the window, since it should
|
||||||
// already have a retain from the nib.
|
// already have a retain from the nib.
|
||||||
- (void)setWindow:(NSWindow *)window
|
- (void)setWindow:(NSWindow *)window
|
||||||
{
|
{
|
||||||
_window = window;
|
_window = window;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -169,51 +170,51 @@
|
||||||
*/
|
*/
|
||||||
- (NSWindow *)_transferWindowOwnership
|
- (NSWindow *)_transferWindowOwnership
|
||||||
{
|
{
|
||||||
NSWindow *window = _window;
|
NSWindow *window = _window;
|
||||||
_window = nil;
|
_window = nil;
|
||||||
return [window autorelease];
|
return [window autorelease];
|
||||||
}
|
}
|
||||||
|
|
||||||
- (void)makeWindowControllers
|
- (void)makeWindowControllers
|
||||||
{
|
{
|
||||||
NSString *name = [self windowNibName];
|
NSString *name = [self windowNibName];
|
||||||
|
|
||||||
if ([name length] > 0)
|
if ([name length] > 0)
|
||||||
{
|
{
|
||||||
NSWindowController *controller;
|
NSWindowController *controller;
|
||||||
controller = [[NSWindowController alloc] initWithWindowNibName:name owner:self];
|
controller = [[NSWindowController alloc] initWithWindowNibName:name owner:self];
|
||||||
[self addWindowController:controller];
|
[self addWindowController:controller];
|
||||||
[controller release];
|
[controller release];
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
[NSException raise:NSInternalInconsistencyException
|
[NSException raise:NSInternalInconsistencyException
|
||||||
format:@"%@ must override either -windowNibName or -makeWindowControllers",
|
format:@"%@ must override either -windowNibName or -makeWindowControllers",
|
||||||
NSStringFromClass([self class])];
|
NSStringFromClass([self class])];
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
- (void)showWindows
|
- (void)showWindows
|
||||||
{
|
{
|
||||||
[_windowControllers makeObjectsPerformSelector:@selector(showWindow:) withObject:self];
|
[_windowControllers makeObjectsPerformSelector:@selector(showWindow:) withObject:self];
|
||||||
}
|
}
|
||||||
|
|
||||||
- (BOOL)isDocumentEdited
|
- (BOOL)isDocumentEdited
|
||||||
{
|
{
|
||||||
return _changeCount != 0;
|
return _changeCount != 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
- (void)updateChangeCount:(NSDocumentChangeType)change
|
- (void)updateChangeCount:(NSDocumentChangeType)change
|
||||||
{
|
{
|
||||||
int i, count = [_windowControllers count];
|
int i, count = [_windowControllers count];
|
||||||
BOOL isEdited;
|
BOOL isEdited;
|
||||||
|
|
||||||
switch (change)
|
switch (change)
|
||||||
{
|
{
|
||||||
case NSChangeDone: _changeCount++; break;
|
case NSChangeDone: _changeCount++; break;
|
||||||
case NSChangeUndone: _changeCount--; break;
|
case NSChangeUndone: _changeCount--; break;
|
||||||
case NSChangeCleared: _changeCount = 0; break;
|
case NSChangeCleared: _changeCount = 0; break;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* NOTE: Apple's implementation seems to not call -isDocumentEdited
|
* NOTE: Apple's implementation seems to not call -isDocumentEdited
|
||||||
|
@ -221,126 +222,129 @@
|
||||||
* would be better to call the method in case it's overridden by a
|
* would be better to call the method in case it's overridden by a
|
||||||
* subclass, but we may want to keep Apple's behavior.
|
* subclass, but we may want to keep Apple's behavior.
|
||||||
*/
|
*/
|
||||||
isEdited = [self isDocumentEdited];
|
isEdited = [self isDocumentEdited];
|
||||||
|
|
||||||
for (i=0; i<count; i++)
|
for (i=0; i<count; i++)
|
||||||
{
|
{
|
||||||
[[[_windowControllers objectAtIndex:i] window] setDocumentEdited:isEdited];
|
[[[_windowControllers objectAtIndex:i] window] setDocumentEdited:isEdited];
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
- (BOOL)canCloseDocument
|
- (BOOL)canCloseDocument
|
||||||
{
|
{
|
||||||
int result;
|
int result;
|
||||||
|
|
||||||
if (![self isDocumentEdited]) return YES;
|
if (![self isDocumentEdited])
|
||||||
|
return YES;
|
||||||
|
|
||||||
//FIXME -- localize.
|
//FIXME -- localize.
|
||||||
result = NSRunAlertPanel(@"Close", @"%@ has changed. Save?",
|
result = NSRunAlertPanel(@"Close", @"%@ has changed. Save?",
|
||||||
@"Save", @"Cancel", @"Don't Save", [self displayName]);
|
@"Save", @"Cancel", @"Don't Save", [self displayName]);
|
||||||
|
|
||||||
#define Save NSAlertDefaultReturn
|
#define Save NSAlertDefaultReturn
|
||||||
#define Cancel NSAlertAlternateReturn
|
#define Cancel NSAlertAlternateReturn
|
||||||
#define DontSave NSAlertOtherReturn
|
#define DontSave NSAlertOtherReturn
|
||||||
|
|
||||||
switch (result)
|
switch (result)
|
||||||
{
|
{
|
||||||
// return NO if save failed
|
// return NO if save failed
|
||||||
case Save: [self saveDocument:nil]; return ![self isDocumentEdited];
|
case Save: [self saveDocument:nil]; return ![self isDocumentEdited];
|
||||||
case DontSave: return YES;
|
case DontSave: return YES;
|
||||||
case Cancel:
|
case Cancel:
|
||||||
default: return NO;
|
default: return NO;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
- (BOOL)shouldCloseWindowController:(NSWindowController *)windowController
|
- (BOOL)shouldCloseWindowController:(NSWindowController *)windowController
|
||||||
{
|
{
|
||||||
if (![_windowControllers containsObject:windowController]) return YES;
|
if (![_windowControllers containsObject:windowController]) return YES;
|
||||||
|
|
||||||
/* If it's the last window controller, pop up a warning */
|
/* If it's the last window controller, pop up a warning */
|
||||||
/* maybe we should count only loaded window controllers (or visible windows). */
|
/* maybe we should count only loaded window controllers (or visible windows). */
|
||||||
if ([windowController shouldCloseDocument] || [_windowControllers count] == 1)
|
if ([windowController shouldCloseDocument]
|
||||||
{
|
|| [_windowControllers count] == 1)
|
||||||
return [self canCloseDocument];
|
{
|
||||||
}
|
return [self canCloseDocument];
|
||||||
|
}
|
||||||
|
|
||||||
return YES;
|
return YES;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
- (NSString *)displayName
|
- (NSString *)displayName
|
||||||
{
|
{
|
||||||
if ([self fileName] != nil)
|
if ([self fileName] != nil)
|
||||||
{
|
{
|
||||||
return [[[self fileName] lastPathComponent] stringByDeletingPathExtension];
|
return [[[self fileName] lastPathComponent] stringByDeletingPathExtension];
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
//FIXME -- localize.
|
//FIXME -- localize.
|
||||||
return [NSString stringWithFormat:@"Untitled-%d", _documentIndex];
|
return [NSString stringWithFormat:@"Untitled-%d", _documentIndex];
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
- (BOOL)keepBackupFile
|
- (BOOL)keepBackupFile
|
||||||
{
|
{
|
||||||
return NO;
|
return NO;
|
||||||
}
|
}
|
||||||
|
|
||||||
- (NSData *)dataRepresentationOfType:(NSString *)type
|
- (NSData *)dataRepresentationOfType:(NSString *)type
|
||||||
{
|
{
|
||||||
[NSException raise:NSInternalInconsistencyException format:@"%@ must implement %@",
|
[NSException raise:NSInternalInconsistencyException format:@"%@ must implement %@",
|
||||||
NSStringFromClass([self class]), NSStringFromSelector(_cmd)];
|
NSStringFromClass([self class]), NSStringFromSelector(_cmd)];
|
||||||
return nil;
|
return nil;
|
||||||
}
|
}
|
||||||
|
|
||||||
- (BOOL)loadDataRepresentation:(NSData *)data ofType:(NSString *)type
|
- (BOOL)loadDataRepresentation:(NSData *)data ofType:(NSString *)type
|
||||||
{
|
{
|
||||||
[NSException raise:NSInternalInconsistencyException format:@"%@ must implement %@",
|
[NSException raise:NSInternalInconsistencyException format:@"%@ must implement %@",
|
||||||
NSStringFromClass([self class]), NSStringFromSelector(_cmd)];
|
NSStringFromClass([self class]), NSStringFromSelector(_cmd)];
|
||||||
return NO;
|
return NO;
|
||||||
}
|
}
|
||||||
|
|
||||||
- (NSFileWrapper *)fileWrapperRepresentationOfType:(NSString *)type
|
- (NSFileWrapper *)fileWrapperRepresentationOfType:(NSString *)type
|
||||||
{
|
{
|
||||||
NSData *data = [self dataRepresentationOfType:type];
|
NSData *data = [self dataRepresentationOfType:type];
|
||||||
|
|
||||||
if (data == nil) return nil;
|
if (data == nil)
|
||||||
|
return nil;
|
||||||
|
|
||||||
return [[[NSFileWrapper alloc] initRegularFileWithContents:data] autorelease];
|
return [[[NSFileWrapper alloc] initRegularFileWithContents:data] autorelease];
|
||||||
}
|
}
|
||||||
|
|
||||||
- (BOOL)loadFileWrapperRepresentation:(NSFileWrapper *)wrapper ofType:(NSString *)type
|
- (BOOL)loadFileWrapperRepresentation:(NSFileWrapper *)wrapper ofType:(NSString *)type
|
||||||
{
|
{
|
||||||
if ([wrapper isRegularFile])
|
if ([wrapper isRegularFile])
|
||||||
{
|
{
|
||||||
return [self loadDataRepresentation:[wrapper regularFileContents] ofType:type];
|
return [self loadDataRepresentation:[wrapper regularFileContents] ofType:type];
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* This even happens on a symlink. May want to use
|
* This even happens on a symlink. May want to use
|
||||||
* -stringByResolvingAllSymlinksInPath somewhere, but Apple doesn't.
|
* -stringByResolvingAllSymlinksInPath somewhere, but Apple doesn't.
|
||||||
*/
|
*/
|
||||||
NSLog(@"%@ must be overridden if your document deals with file packages.",
|
NSLog(@"%@ must be overridden if your document deals with file packages.",
|
||||||
NSStringFromSelector(_cmd));
|
NSStringFromSelector(_cmd));
|
||||||
|
|
||||||
return NO;
|
return NO;
|
||||||
}
|
}
|
||||||
|
|
||||||
- (BOOL)writeToFile:(NSString *)fileName ofType:(NSString *)type
|
- (BOOL)writeToFile:(NSString *)fileName ofType:(NSString *)type
|
||||||
{
|
{
|
||||||
return [[self fileWrapperRepresentationOfType:type]
|
return [[self fileWrapperRepresentationOfType:type]
|
||||||
writeToFile:fileName atomically:YES updateFilenames:YES];
|
writeToFile:fileName atomically:YES updateFilenames:YES];
|
||||||
}
|
}
|
||||||
|
|
||||||
- (BOOL)readFromFile:(NSString *)fileName ofType:(NSString *)type
|
- (BOOL)readFromFile:(NSString *)fileName ofType:(NSString *)type
|
||||||
{
|
{
|
||||||
NSFileWrapper *wrapper = [[[NSFileWrapper alloc] initWithPath:fileName] autorelease];
|
NSFileWrapper *wrapper = [[[NSFileWrapper alloc] initWithPath:fileName] autorelease];
|
||||||
return [self loadFileWrapperRepresentation:wrapper ofType:type];
|
return [self loadFileWrapperRepresentation:wrapper ofType:type];
|
||||||
}
|
}
|
||||||
|
|
||||||
- (BOOL)revertToSavedFromFile:(NSString *)fileName ofType:(NSString *)type
|
- (BOOL)revertToSavedFromFile:(NSString *)fileName ofType:(NSString *)type
|
||||||
{
|
{
|
||||||
return [self readFromFile:fileName ofType:type];
|
return [self readFromFile:fileName ofType:type];
|
||||||
}
|
}
|
||||||
|
|
||||||
- (IBAction)changeSaveType:(id)sender
|
- (IBAction)changeSaveType:(id)sender
|
||||||
|
@ -349,13 +353,13 @@
|
||||||
|
|
||||||
- (int)runModalSavePanel:(NSSavePanel *)savePanel withAccessoryView:(NSView *)accessoryView
|
- (int)runModalSavePanel:(NSSavePanel *)savePanel withAccessoryView:(NSView *)accessoryView
|
||||||
{
|
{
|
||||||
[savePanel setAccessoryView:accessoryView];
|
[savePanel setAccessoryView:accessoryView];
|
||||||
return [savePanel runModal];
|
return [savePanel runModal];
|
||||||
}
|
}
|
||||||
|
|
||||||
- (BOOL)shouldRunSavePanelWithAccessoryView
|
- (BOOL)shouldRunSavePanelWithAccessoryView
|
||||||
{
|
{
|
||||||
return YES;
|
return YES;
|
||||||
}
|
}
|
||||||
|
|
||||||
- (void)_loadPanelAccessoryNib
|
- (void)_loadPanelAccessoryNib
|
||||||
|
@ -369,80 +373,81 @@
|
||||||
|
|
||||||
- (NSString *)fileNameFromRunningSavePanelForSaveOperation:(NSSaveOperationType)saveOperation
|
- (NSString *)fileNameFromRunningSavePanelForSaveOperation:(NSSaveOperationType)saveOperation
|
||||||
{
|
{
|
||||||
NSView *accessory = nil;
|
NSView *accessory = nil;
|
||||||
NSString *title = @"save";
|
NSString *title = @"save";
|
||||||
NSSavePanel *savePanel = [NSSavePanel savePanel];
|
NSSavePanel *savePanel = [NSSavePanel savePanel];
|
||||||
NSArray *extensions = [[NSDocumentController sharedDocumentController]
|
NSArray *extensions = [[NSDocumentController sharedDocumentController]
|
||||||
fileExtensionsFromType:[self fileType]];
|
fileExtensionsFromType:[self fileType]];
|
||||||
|
|
||||||
if ([self shouldRunSavePanelWithAccessoryView])
|
if ([self shouldRunSavePanelWithAccessoryView])
|
||||||
{
|
{
|
||||||
if (savePanelAccessory == nil)
|
if (savePanelAccessory == nil)
|
||||||
[self _loadPanelAccessoryNib];
|
[self _loadPanelAccessoryNib];
|
||||||
|
|
||||||
[self _addItemsToSpaButtonFromArray:extensions];
|
[self _addItemsToSpaButtonFromArray:extensions];
|
||||||
|
|
||||||
accessory = savePanelAccessory;
|
accessory = savePanelAccessory;
|
||||||
}
|
}
|
||||||
|
|
||||||
if ([extensions count] > 0)
|
if ([extensions count] > 0)
|
||||||
[savePanel setRequiredFileType:[extensions objectAtIndex:0]];
|
[savePanel setRequiredFileType:[extensions objectAtIndex:0]];
|
||||||
|
|
||||||
switch (saveOperation)
|
switch (saveOperation)
|
||||||
{
|
{
|
||||||
// FIXME -- localize.
|
// FIXME -- localize.
|
||||||
case NSSaveOperation: title = @"Save"; break;
|
case NSSaveOperation: title = @"Save"; break;
|
||||||
case NSSaveAsOperation: title = @"Save As"; break;
|
case NSSaveAsOperation: title = @"Save As"; break;
|
||||||
case NSSaveToOperation: title = @"Save To"; break;
|
case NSSaveToOperation: title = @"Save To"; break;
|
||||||
}
|
}
|
||||||
|
|
||||||
[savePanel setTitle:title];
|
[savePanel setTitle:title];
|
||||||
if ([self fileName])
|
|
||||||
[savePanel setDirectory:[[self fileName] stringByDeletingLastPathComponent]];
|
|
||||||
|
|
||||||
if ([self runModalSavePanel:savePanel withAccessoryView:accessory])
|
if ([self fileName])
|
||||||
{
|
[savePanel setDirectory:[[self fileName] stringByDeletingLastPathComponent]];
|
||||||
return [savePanel filename];
|
|
||||||
}
|
|
||||||
|
|
||||||
return nil;
|
if ([self runModalSavePanel:savePanel withAccessoryView:accessory])
|
||||||
|
{
|
||||||
|
return [savePanel filename];
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil;
|
||||||
}
|
}
|
||||||
|
|
||||||
- (BOOL)shouldChangePrintInfo:(NSPrintInfo *)newPrintInfo
|
- (BOOL)shouldChangePrintInfo:(NSPrintInfo *)newPrintInfo
|
||||||
{
|
{
|
||||||
return YES;
|
return YES;
|
||||||
}
|
}
|
||||||
|
|
||||||
- (NSPrintInfo *)printInfo
|
- (NSPrintInfo *)printInfo
|
||||||
{
|
{
|
||||||
return _printInfo? _printInfo : [NSPrintInfo sharedPrintInfo];
|
return _printInfo? _printInfo : [NSPrintInfo sharedPrintInfo];
|
||||||
}
|
}
|
||||||
|
|
||||||
- (void)setPrintInfo:(NSPrintInfo *)printInfo
|
- (void)setPrintInfo:(NSPrintInfo *)printInfo
|
||||||
{
|
{
|
||||||
[printInfo retain];
|
[printInfo retain];
|
||||||
[_printInfo release];
|
[_printInfo release];
|
||||||
_printInfo = printInfo;
|
_printInfo = printInfo;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
// Page layout panel (Page Setup)
|
// Page layout panel (Page Setup)
|
||||||
|
|
||||||
- (int)runModalPageLayoutWithPrintInfo:(NSPrintInfo *)printInfo
|
- (int)runModalPageLayoutWithPrintInfo:(NSPrintInfo *)printInfo
|
||||||
{
|
{
|
||||||
return [[NSPageLayout pageLayout] runModalWithPrintInfo:printInfo];
|
return [[NSPageLayout pageLayout] runModalWithPrintInfo:printInfo];
|
||||||
}
|
}
|
||||||
|
|
||||||
- (IBAction)runPageLayout:(id)sender
|
- (IBAction)runPageLayout:(id)sender
|
||||||
{
|
{
|
||||||
NSPrintInfo *printInfo = [self printInfo];
|
NSPrintInfo *printInfo = [self printInfo];
|
||||||
|
|
||||||
if ([self runModalPageLayoutWithPrintInfo:printInfo] &&
|
if ([self runModalPageLayoutWithPrintInfo:printInfo]
|
||||||
[self shouldChangePrintInfo:printInfo])
|
&& [self shouldChangePrintInfo:printInfo])
|
||||||
{
|
{
|
||||||
[self setPrintInfo:printInfo];
|
[self setPrintInfo:printInfo];
|
||||||
[self updateChangeCount:NSChangeDone];
|
[self updateChangeCount:NSChangeDone];
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* This is overridden by subclassers; the default implementation does nothing. */
|
/* This is overridden by subclassers; the default implementation does nothing. */
|
||||||
|
@ -457,114 +462,114 @@
|
||||||
|
|
||||||
- (BOOL)validateMenuItem:(NSMenuItem *)anItem
|
- (BOOL)validateMenuItem:(NSMenuItem *)anItem
|
||||||
{
|
{
|
||||||
if ([anItem action] == @selector(revertDocumentToSaved:))
|
if ([anItem action] == @selector(revertDocumentToSaved:))
|
||||||
return ([self fileName] != nil && [self isDocumentEdited]);
|
return ([self fileName] != nil && [self isDocumentEdited]);
|
||||||
|
|
||||||
// FIXME should validate spa popup items; return YES if it's a native type.
|
// FIXME should validate spa popup items; return YES if it's a native type.
|
||||||
|
|
||||||
return YES;
|
return YES;
|
||||||
}
|
}
|
||||||
|
|
||||||
- (NSString *)saveFileType
|
- (NSString *)saveFileType
|
||||||
{
|
{
|
||||||
// FIXME this should return type picked on save accessory
|
// FIXME this should return type picked on save accessory
|
||||||
// return [spaPopupButton title];
|
// return [spaPopupButton title];
|
||||||
return [self fileType];
|
return [self fileType];
|
||||||
}
|
}
|
||||||
|
|
||||||
- (void)_doSaveAs:(NSSaveOperationType)saveOperation
|
- (void)_doSaveAs:(NSSaveOperationType)saveOperation
|
||||||
{
|
{
|
||||||
NSFileManager *fileManager = [NSFileManager defaultManager];
|
NSFileManager *fileManager = [NSFileManager defaultManager];
|
||||||
NSString *filename = [self fileName];
|
NSString *filename = [self fileName];
|
||||||
NSString *backupFilename = nil;
|
NSString *backupFilename = nil;
|
||||||
|
|
||||||
if (filename == nil || saveOperation != NSSaveOperation)
|
if (filename == nil || saveOperation != NSSaveOperation)
|
||||||
|
{
|
||||||
|
filename = [self fileNameFromRunningSavePanelForSaveOperation:saveOperation];
|
||||||
|
if (saveOperation == NSSaveOperation) saveOperation = NSSaveAsOperation;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (filename)
|
||||||
|
{
|
||||||
|
if ([fileManager fileExistsAtPath:filename])
|
||||||
{
|
{
|
||||||
filename = [self fileNameFromRunningSavePanelForSaveOperation:saveOperation];
|
NSString *extension = [filename pathExtension];
|
||||||
if (saveOperation == NSSaveOperation) saveOperation = NSSaveAsOperation;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (filename)
|
backupFilename = [filename stringByDeletingPathExtension];
|
||||||
{
|
backupFilename = [backupFilename stringByAppendingString:@"~"];
|
||||||
if ([fileManager fileExistsAtPath:filename])
|
backupFilename = [backupFilename stringByAppendingPathExtension:extension];
|
||||||
{
|
|
||||||
NSString *extension = [filename pathExtension];
|
|
||||||
|
|
||||||
backupFilename = [filename stringByDeletingPathExtension];
|
/* Save panel has already asked if the user wants to replace it */
|
||||||
backupFilename = [backupFilename stringByAppendingString:@"~"];
|
|
||||||
backupFilename = [backupFilename stringByAppendingPathExtension:extension];
|
|
||||||
|
|
||||||
/* Save panel has already asked if the user wants to replace it */
|
/* NSFileManager movePath: will fail if destination exists */
|
||||||
|
if ([fileManager fileExistsAtPath:backupFilename])
|
||||||
|
[fileManager removeFileAtPath:backupFilename handler:nil];
|
||||||
|
|
||||||
/* NSFileManager movePath: will fail if destination exists */
|
// Move or copy?
|
||||||
if ([fileManager fileExistsAtPath:backupFilename])
|
if (![fileManager movePath:filename toPath:backupFilename handler:nil] &&
|
||||||
[fileManager removeFileAtPath:backupFilename handler:nil];
|
[self keepBackupFile])
|
||||||
|
|
||||||
// Move or copy?
|
|
||||||
if (![fileManager movePath:filename toPath:backupFilename handler:nil] &&
|
|
||||||
[self keepBackupFile])
|
|
||||||
{
|
{
|
||||||
//FIXME -- localize.
|
//FIXME -- localize.
|
||||||
int result = NSRunAlertPanel(@"File Error",
|
int result = NSRunAlertPanel(@"File Error",
|
||||||
@"Can't create backup file. Save anyways?",
|
@"Can't create backup file. Save anyways?",
|
||||||
@"Save", @"Cancel", nil);
|
@"Save", @"Cancel", nil);
|
||||||
|
|
||||||
if (result != NSAlertDefaultReturn) return;
|
if (result != NSAlertDefaultReturn) return;
|
||||||
}
|
}
|
||||||
}
|
|
||||||
if ([self writeToFile:filename ofType:[self saveFileType]])
|
|
||||||
{
|
|
||||||
if (saveOperation != NSSaveToOperation)
|
|
||||||
{
|
|
||||||
[self setFileName:filename];
|
|
||||||
[self setFileType:[self saveFileType]];
|
|
||||||
[self updateChangeCount:NSChangeCleared];
|
|
||||||
}
|
|
||||||
|
|
||||||
if (backupFilename && ![self keepBackupFile])
|
|
||||||
{
|
|
||||||
[fileManager removeFileAtPath:backupFilename handler:nil];
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
if ([self writeToFile:filename ofType:[self saveFileType]])
|
||||||
|
{
|
||||||
|
if (saveOperation != NSSaveToOperation)
|
||||||
|
{
|
||||||
|
[self setFileName:filename];
|
||||||
|
[self setFileType:[self saveFileType]];
|
||||||
|
[self updateChangeCount:NSChangeCleared];
|
||||||
|
}
|
||||||
|
|
||||||
|
if (backupFilename && ![self keepBackupFile])
|
||||||
|
{
|
||||||
|
[fileManager removeFileAtPath:backupFilename handler:nil];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
- (IBAction)saveDocument:(id)sender
|
- (IBAction)saveDocument:(id)sender
|
||||||
{
|
{
|
||||||
[self _doSaveAs:NSSaveOperation];
|
[self _doSaveAs:NSSaveOperation];
|
||||||
}
|
}
|
||||||
|
|
||||||
- (IBAction)saveDocumentAs:(id)sender
|
- (IBAction)saveDocumentAs:(id)sender
|
||||||
{
|
{
|
||||||
[self _doSaveAs:NSSaveAsOperation];
|
[self _doSaveAs:NSSaveAsOperation];
|
||||||
}
|
}
|
||||||
|
|
||||||
- (IBAction)saveDocumentTo:(id)sender
|
- (IBAction)saveDocumentTo:(id)sender
|
||||||
{
|
{
|
||||||
[self _doSaveAs:NSSaveToOperation];
|
[self _doSaveAs:NSSaveToOperation];
|
||||||
}
|
}
|
||||||
|
|
||||||
- (IBAction)revertDocumentToSaved:(id)sender
|
- (IBAction)revertDocumentToSaved:(id)sender
|
||||||
{
|
{
|
||||||
int result;
|
int result;
|
||||||
|
|
||||||
//FIXME -- localize.
|
//FIXME -- localize.
|
||||||
result = NSRunAlertPanel(@"Revert",
|
result = NSRunAlertPanel(@"Revert",
|
||||||
@"%@ has been edited. Are you sure you want to undo changes?",
|
@"%@ has been edited. Are you sure you want to undo changes?",
|
||||||
@"Revert", @"Cancel", nil, [self displayName]);
|
@"Revert", @"Cancel", nil, [self displayName]);
|
||||||
|
|
||||||
if (result == NSAlertDefaultReturn &&
|
if (result == NSAlertDefaultReturn &&
|
||||||
[self revertToSavedFromFile:[self fileName] ofType:[self fileType]])
|
[self revertToSavedFromFile:[self fileName] ofType:[self fileType]])
|
||||||
{
|
{
|
||||||
[self updateChangeCount:NSChangeCleared];
|
[self updateChangeCount:NSChangeCleared];
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
- (void)close
|
- (void)close
|
||||||
{
|
{
|
||||||
// We have an _docFlags.inClose flag, but I don't think we need to use it.
|
// We have an _docFlags.inClose flag, but I don't think we need to use it.
|
||||||
[_windowControllers makeObjectsPerformSelector:@selector(close)];
|
[_windowControllers makeObjectsPerformSelector:@selector(close)];
|
||||||
[[NSDocumentController sharedDocumentController] _removeDocument:self];
|
[[NSDocumentController sharedDocumentController] _removeDocument:self];
|
||||||
}
|
}
|
||||||
|
|
||||||
- (void)windowControllerWillLoadNib:(NSWindowController *)windowController {}
|
- (void)windowControllerWillLoadNib:(NSWindowController *)windowController {}
|
||||||
|
@ -572,82 +577,86 @@
|
||||||
|
|
||||||
- (NSUndoManager *)undoManager
|
- (NSUndoManager *)undoManager
|
||||||
{
|
{
|
||||||
if (_undoManager == nil && [self hasUndoManager])
|
if (_undoManager == nil && [self hasUndoManager])
|
||||||
{
|
{
|
||||||
[self setUndoManager:[[[NSUndoManager alloc] init] autorelease]];
|
[self setUndoManager:[[[NSUndoManager alloc] init] autorelease]];
|
||||||
}
|
}
|
||||||
|
|
||||||
return _undoManager;
|
return _undoManager;
|
||||||
}
|
}
|
||||||
|
|
||||||
- (void)setUndoManager:(NSUndoManager *)undoManager
|
- (void)setUndoManager:(NSUndoManager *)undoManager
|
||||||
{
|
{
|
||||||
if (undoManager != _undoManager)
|
if (undoManager != _undoManager)
|
||||||
{
|
{
|
||||||
NSNotificationCenter *center = [NSNotificationCenter defaultCenter];
|
NSNotificationCenter *center = [NSNotificationCenter defaultCenter];
|
||||||
|
|
||||||
if (_undoManager)
|
if (_undoManager)
|
||||||
{
|
{
|
||||||
[center removeObserver:self
|
[center removeObserver:self
|
||||||
name:NSUndoManagerWillCloseUndoGroupNotification
|
name:NSUndoManagerWillCloseUndoGroupNotification
|
||||||
object:_undoManager];
|
object:_undoManager];
|
||||||
[center removeObserver:self
|
[center removeObserver:self
|
||||||
name:NSUndoManagerDidUndoChangeNotification
|
name:NSUndoManagerDidUndoChangeNotification
|
||||||
object:_undoManager];
|
object:_undoManager];
|
||||||
[center removeObserver:self
|
[center removeObserver:self
|
||||||
name:NSUndoManagerDidRedoChangeNotification
|
name:NSUndoManagerDidRedoChangeNotification
|
||||||
object:_undoManager];
|
object:_undoManager];
|
||||||
}
|
}
|
||||||
|
|
||||||
[(NSObject*)undoManager retain];
|
[(NSObject*)undoManager retain];
|
||||||
[(NSObject*)_undoManager release];
|
[(NSObject*)_undoManager release];
|
||||||
_undoManager = undoManager;
|
_undoManager = undoManager;
|
||||||
|
|
||||||
if (_undoManager == nil)
|
if (_undoManager == nil)
|
||||||
{
|
{
|
||||||
[self setHasUndoManager:NO];
|
[self setHasUndoManager:NO];
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
[center addObserver:self
|
[center addObserver:self
|
||||||
selector:@selector(_changeWasDone:)
|
selector:@selector(_changeWasDone:)
|
||||||
name:NSUndoManagerWillCloseUndoGroupNotification
|
name:NSUndoManagerWillCloseUndoGroupNotification
|
||||||
object:_undoManager];
|
object:_undoManager];
|
||||||
[center addObserver:self
|
[center addObserver:self
|
||||||
selector:@selector(_changeWasUndone:)
|
selector:@selector(_changeWasUndone:)
|
||||||
name:NSUndoManagerDidUndoChangeNotification
|
name:NSUndoManagerDidUndoChangeNotification
|
||||||
object:_undoManager];
|
object:_undoManager];
|
||||||
[[NSNotificationCenter defaultCenter]
|
[[NSNotificationCenter defaultCenter]
|
||||||
addObserver:self
|
addObserver:self
|
||||||
selector:@selector(_changeWasRedone:)
|
selector:@selector(_changeWasRedone:)
|
||||||
name:NSUndoManagerDidRedoChangeNotification
|
name:NSUndoManagerDidRedoChangeNotification
|
||||||
object:_undoManager];
|
object:_undoManager];
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
- (BOOL)hasUndoManager
|
- (BOOL)hasUndoManager
|
||||||
{
|
{
|
||||||
return _docFlags.hasUndoManager;
|
return _docFlags.hasUndoManager;
|
||||||
}
|
}
|
||||||
|
|
||||||
- (void)setHasUndoManager:(BOOL)flag
|
- (void)setHasUndoManager:(BOOL)flag
|
||||||
{
|
{
|
||||||
if (_undoManager && !flag)
|
if (_undoManager && !flag)
|
||||||
[self setUndoManager:nil];
|
[self setUndoManager:nil];
|
||||||
|
|
||||||
_docFlags.hasUndoManager = flag;
|
_docFlags.hasUndoManager = flag;
|
||||||
}
|
}
|
||||||
|
|
||||||
- (void)_changeWasDone:(NSNotification *)notification
|
- (void)_changeWasDone:(NSNotification *)notification
|
||||||
{ [self updateChangeCount:NSChangeDone];
|
{
|
||||||
|
[self updateChangeCount:NSChangeDone];
|
||||||
}
|
}
|
||||||
|
|
||||||
- (void)_changeWasUndone:(NSNotification *)notification
|
- (void)_changeWasUndone:(NSNotification *)notification
|
||||||
{ [self updateChangeCount:NSChangeUndone];
|
{
|
||||||
|
[self updateChangeCount:NSChangeUndone];
|
||||||
}
|
}
|
||||||
|
|
||||||
- (void)_changeWasRedone:(NSNotification *)notification
|
- (void)_changeWasRedone:(NSNotification *)notification
|
||||||
{ [self updateChangeCount:NSChangeDone];
|
{
|
||||||
|
[self updateChangeCount:NSChangeDone];
|
||||||
}
|
}
|
||||||
|
|
||||||
@end
|
@end
|
||||||
|
|
||||||
|
|
|
@ -22,15 +22,16 @@ static NSString *NSDocumentClassKey = @"NSDocumentClass";
|
||||||
|
|
||||||
static NSDictionary *TypeInfoForName(NSArray *types, NSString *typeName)
|
static NSDictionary *TypeInfoForName(NSArray *types, NSString *typeName)
|
||||||
{
|
{
|
||||||
int i, count = [types count];
|
int i, count = [types count];
|
||||||
for (i=0; i<count;i++)
|
for (i=0; i<count;i++)
|
||||||
{
|
{
|
||||||
NSDictionary *dict = [types objectAtIndex:i];
|
NSDictionary *dict = [types objectAtIndex:i];
|
||||||
if ([[dict objectForKey:NSNameKey] isEqualToString:typeName])
|
|
||||||
return dict;
|
|
||||||
}
|
|
||||||
|
|
||||||
return nil;
|
if ([[dict objectForKey:NSNameKey] isEqualToString:typeName])
|
||||||
|
return dict;
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil;
|
||||||
}
|
}
|
||||||
|
|
||||||
@implementation NSDocumentController
|
@implementation NSDocumentController
|
||||||
|
@ -41,281 +42,291 @@ static NSDictionary *TypeInfoForName(NSArray *types, NSString *typeName)
|
||||||
|
|
||||||
+ (id)documentController //private
|
+ (id)documentController //private
|
||||||
{
|
{
|
||||||
return [self sharedDocumentController];
|
return [self sharedDocumentController];
|
||||||
}
|
}
|
||||||
|
|
||||||
+ (id)allocWithZone:(NSZone *)zone
|
+ (id)allocWithZone:(NSZone *)zone
|
||||||
{
|
{
|
||||||
return [self sharedDocumentController];
|
return [self sharedDocumentController];
|
||||||
}
|
}
|
||||||
|
|
||||||
+ (id)sharedDocumentController
|
+ (id)sharedDocumentController
|
||||||
{
|
{
|
||||||
static id instance = nil;
|
static id instance = nil;
|
||||||
|
|
||||||
if (instance == nil) instance = [[super allocWithZone:NULL] init];
|
if (instance == nil)
|
||||||
return instance;
|
instance = [[super allocWithZone:NULL] init];
|
||||||
|
|
||||||
|
return instance;
|
||||||
}
|
}
|
||||||
|
|
||||||
- init
|
- init
|
||||||
{
|
{
|
||||||
NSDictionary *customDict = [[NSBundle mainBundle] infoDictionary];
|
NSDictionary *customDict = [[NSBundle mainBundle] infoDictionary];
|
||||||
|
|
||||||
_types = [[customDict objectForKey:NSTypesKey] retain];
|
_types = [[customDict objectForKey:NSTypesKey] retain];
|
||||||
_documents = [[NSMutableArray alloc] init];
|
_documents = [[NSMutableArray alloc] init];
|
||||||
[self setShouldCreateUI:YES];
|
[self setShouldCreateUI:YES];
|
||||||
|
|
||||||
[[[NSWorkspace sharedWorkspace] notificationCenter]
|
[[[NSWorkspace sharedWorkspace] notificationCenter]
|
||||||
addObserver:self
|
addObserver:self
|
||||||
selector:@selector(_workspaceWillPowerOff:)
|
selector:@selector(_workspaceWillPowerOff:)
|
||||||
name:NSWorkspaceWillPowerOffNotification
|
name:NSWorkspaceWillPowerOffNotification
|
||||||
object:nil];
|
object:nil];
|
||||||
|
|
||||||
return self;
|
return self;
|
||||||
}
|
}
|
||||||
|
|
||||||
- (void)dealloc
|
- (void)dealloc
|
||||||
{
|
{
|
||||||
[[[NSWorkspace sharedWorkspace] notificationCenter] removeObserver:self];
|
[[[NSWorkspace sharedWorkspace] notificationCenter] removeObserver:self];
|
||||||
[_documents release];
|
[_documents release];
|
||||||
[_types release];
|
[_types release];
|
||||||
[super dealloc];
|
[super dealloc];
|
||||||
}
|
}
|
||||||
|
|
||||||
- (BOOL)shouldCreateUI
|
- (BOOL)shouldCreateUI
|
||||||
{
|
{
|
||||||
return _controllerFlags.shouldCreateUI;
|
return _controllerFlags.shouldCreateUI;
|
||||||
}
|
}
|
||||||
|
|
||||||
- (void)setShouldCreateUI:(BOOL)flag
|
- (void)setShouldCreateUI:(BOOL)flag
|
||||||
{
|
{
|
||||||
_controllerFlags.shouldCreateUI = flag;
|
_controllerFlags.shouldCreateUI = flag;
|
||||||
}
|
}
|
||||||
|
|
||||||
- (id)makeUntitledDocumentOfType:(NSString *)type
|
- (id)makeUntitledDocumentOfType:(NSString *)type
|
||||||
{
|
{
|
||||||
Class documentClass = [self documentClassForType:type];
|
Class documentClass = [self documentClassForType:type];
|
||||||
return [[[documentClass alloc] init] autorelease];
|
return [[[documentClass alloc] init] autorelease];
|
||||||
}
|
}
|
||||||
|
|
||||||
- (id)makeDocumentWithContentsOfFile:(NSString *)fileName ofType:(NSString *)type
|
- (id)makeDocumentWithContentsOfFile:(NSString *)fileName ofType:(NSString *)type
|
||||||
{
|
{
|
||||||
Class documentClass = [self documentClassForType:type];
|
Class documentClass = [self documentClassForType:type];
|
||||||
return [[[documentClass alloc] initWithContentsOfFile:fileName ofType:type] autorelease];
|
return [[[documentClass alloc] initWithContentsOfFile:fileName ofType:type] autorelease];
|
||||||
}
|
}
|
||||||
|
|
||||||
- (id)makeDocumentWithContentsOfURL:(NSURL *)url ofType:(NSString *)type
|
- (id)makeDocumentWithContentsOfURL:(NSURL *)url ofType:(NSString *)type
|
||||||
{
|
{
|
||||||
Class documentClass = [self documentClassForType:type];
|
Class documentClass = [self documentClassForType:type];
|
||||||
return [[[documentClass alloc] initWithContentsOfURL:url ofType:type] autorelease];
|
return [[[documentClass alloc] initWithContentsOfURL:url ofType:type] autorelease];
|
||||||
}
|
}
|
||||||
|
|
||||||
- _defaultType
|
- _defaultType
|
||||||
{
|
{
|
||||||
if ([_types count] == 0) return nil; // raise exception?
|
if ([_types count] == 0)
|
||||||
return [[_types objectAtIndex:0] objectForKey:NSNameKey];
|
return nil; // raise exception?
|
||||||
|
|
||||||
|
return [[_types objectAtIndex:0] objectForKey:NSNameKey];
|
||||||
}
|
}
|
||||||
|
|
||||||
/* These next two should really have been public. */
|
/* These next two should really have been public. */
|
||||||
- (void)_addDocument:(NSDocument *)document
|
- (void)_addDocument:(NSDocument *)document
|
||||||
{
|
{
|
||||||
[_documents addObject:document];
|
[_documents addObject:document];
|
||||||
}
|
}
|
||||||
|
|
||||||
- (void)_removeDocument:(NSDocument *)document
|
- (void)_removeDocument:(NSDocument *)document
|
||||||
{
|
{
|
||||||
[_documents removeObject:document];
|
[_documents removeObject:document];
|
||||||
}
|
}
|
||||||
|
|
||||||
- (id)openUntitledDocumentOfType:(NSString*)type display:(BOOL)display
|
- (id)openUntitledDocumentOfType:(NSString*)type display:(BOOL)display
|
||||||
{
|
{
|
||||||
NSDocument *document = [self makeUntitledDocumentOfType:type];
|
NSDocument *document = [self makeUntitledDocumentOfType:type];
|
||||||
|
|
||||||
if (document == nil) return nil;
|
if (document == nil)
|
||||||
|
return nil;
|
||||||
|
|
||||||
[self _addDocument:document];
|
[self _addDocument:document];
|
||||||
if ([self shouldCreateUI])
|
if ([self shouldCreateUI])
|
||||||
{
|
{
|
||||||
[document makeWindowControllers];
|
[document makeWindowControllers];
|
||||||
if (display)
|
if (display)
|
||||||
[document showWindows];
|
[document showWindows];
|
||||||
}
|
}
|
||||||
|
|
||||||
return document;
|
return document;
|
||||||
}
|
}
|
||||||
|
|
||||||
- (id)openDocumentWithContentsOfFile:(NSString *)fileName display:(BOOL)display
|
- (id)openDocumentWithContentsOfFile:(NSString *)fileName display:(BOOL)display
|
||||||
{
|
{
|
||||||
NSDocument *document = [self documentForFileName:fileName];
|
NSDocument *document = [self documentForFileName:fileName];
|
||||||
|
|
||||||
if (document == nil)
|
if (document == nil)
|
||||||
{
|
{
|
||||||
NSString *type = [self typeFromFileExtension:[fileName pathExtension]];
|
NSString *type = [self typeFromFileExtension:[fileName pathExtension]];
|
||||||
|
|
||||||
if ((document = [self makeDocumentWithContentsOfFile:fileName ofType:type]))
|
if ((document = [self makeDocumentWithContentsOfFile:fileName ofType:type]))
|
||||||
[self _addDocument:document];
|
[self _addDocument:document];
|
||||||
if ([self shouldCreateUI])
|
|
||||||
[document makeWindowControllers];
|
if ([self shouldCreateUI])
|
||||||
|
[document makeWindowControllers];
|
||||||
}
|
}
|
||||||
|
|
||||||
if (display && [self shouldCreateUI])
|
if (display && [self shouldCreateUI])
|
||||||
{
|
{
|
||||||
[document showWindows];
|
[document showWindows];
|
||||||
}
|
}
|
||||||
|
|
||||||
return document;
|
return document;
|
||||||
}
|
}
|
||||||
|
|
||||||
- (id)openDocumentWithContentsOfURL:(NSURL *)url display:(BOOL)display
|
- (id)openDocumentWithContentsOfURL:(NSURL *)url display:(BOOL)display
|
||||||
{
|
{
|
||||||
// Should we only do this if [url isFileURL] is YES?
|
// Should we only do this if [url isFileURL] is YES?
|
||||||
NSDocument *document = [self documentForFileName:[url path]];
|
NSDocument *document = [self documentForFileName:[url path]];
|
||||||
|
|
||||||
if (document == nil)
|
if (document == nil)
|
||||||
{
|
{
|
||||||
NSString *type = [self typeFromFileExtension:[[url path] pathExtension]];
|
NSString *type = [self typeFromFileExtension:[[url path] pathExtension]];
|
||||||
|
|
||||||
document = [self makeDocumentWithContentsOfURL:url ofType:type];
|
document = [self makeDocumentWithContentsOfURL:url ofType:type];
|
||||||
if (document == nil) return nil;
|
|
||||||
|
|
||||||
[self _addDocument:document];
|
if (document == nil)
|
||||||
if ([self shouldCreateUI])
|
return nil;
|
||||||
|
|
||||||
|
[self _addDocument:document];
|
||||||
|
|
||||||
|
if ([self shouldCreateUI])
|
||||||
{
|
{
|
||||||
[document makeWindowControllers];
|
[document makeWindowControllers];
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (display && [self shouldCreateUI])
|
if (display && [self shouldCreateUI])
|
||||||
{
|
{
|
||||||
[document showWindows];
|
[document showWindows];
|
||||||
}
|
}
|
||||||
|
|
||||||
return document;
|
return document;
|
||||||
}
|
}
|
||||||
|
|
||||||
- _setupOpenPanel
|
- _setupOpenPanel
|
||||||
{
|
{
|
||||||
NSOpenPanel *openPanel = [NSOpenPanel openPanel];
|
NSOpenPanel *openPanel = [NSOpenPanel openPanel];
|
||||||
[openPanel setDirectory:[self currentDirectory]];
|
[openPanel setDirectory:[self currentDirectory]];
|
||||||
[openPanel setAllowsMultipleSelection:YES];
|
[openPanel setAllowsMultipleSelection:YES];
|
||||||
return openPanel;
|
return openPanel;
|
||||||
}
|
}
|
||||||
|
|
||||||
- (int)runModalOpenPanel:(NSOpenPanel *)openPanel forTypes:(NSArray *)openableFileExtensions
|
- (int)runModalOpenPanel:(NSOpenPanel *)openPanel forTypes:(NSArray *)openableFileExtensions
|
||||||
{
|
{
|
||||||
return [openPanel runModalForTypes:openableFileExtensions];
|
return [openPanel runModalForTypes:openableFileExtensions];
|
||||||
}
|
}
|
||||||
|
|
||||||
- (NSArray *)_openableFileExtensions
|
- (NSArray *)_openableFileExtensions
|
||||||
{
|
{
|
||||||
int i, count = [_types count];
|
int i, count = [_types count];
|
||||||
NSMutableArray *array = [NSMutableArray arrayWithCapacity:count];
|
NSMutableArray *array = [NSMutableArray arrayWithCapacity:count];
|
||||||
|
|
||||||
for (i=0; i<count; i++)
|
for (i=0; i<count; i++)
|
||||||
{
|
{
|
||||||
NSDictionary *typeInfo = [_types objectAtIndex:i];
|
NSDictionary *typeInfo = [_types objectAtIndex:i];
|
||||||
[array addObjectsFromArray:[typeInfo objectForKey:NSUnixExtensionsKey]];
|
[array addObjectsFromArray:[typeInfo objectForKey:NSUnixExtensionsKey]];
|
||||||
[array addObjectsFromArray:[typeInfo objectForKey:NSDOSExtensionsKey]];
|
[array addObjectsFromArray:[typeInfo objectForKey:NSDOSExtensionsKey]];
|
||||||
}
|
}
|
||||||
|
|
||||||
return array;
|
return array;
|
||||||
}
|
}
|
||||||
|
|
||||||
- (NSArray *)fileNamesFromRunningOpenPanel
|
- (NSArray *)fileNamesFromRunningOpenPanel
|
||||||
{
|
{
|
||||||
NSArray *types = [self _openableFileExtensions];
|
NSArray *types = [self _openableFileExtensions];
|
||||||
NSOpenPanel *openPanel = [self _setupOpenPanel];
|
NSOpenPanel *openPanel = [self _setupOpenPanel];
|
||||||
|
|
||||||
if ([self runModalOpenPanel:openPanel forTypes:types])
|
if ([self runModalOpenPanel:openPanel forTypes:types])
|
||||||
{
|
{
|
||||||
return [openPanel filenames];
|
return [openPanel filenames];
|
||||||
}
|
}
|
||||||
|
|
||||||
return nil;
|
return nil;
|
||||||
}
|
}
|
||||||
|
|
||||||
- (NSArray *)URLsFromRunningOpenPanel
|
- (NSArray *)URLsFromRunningOpenPanel
|
||||||
{
|
{
|
||||||
NSArray *types = [self _openableFileExtensions];
|
NSArray *types = [self _openableFileExtensions];
|
||||||
NSOpenPanel *openPanel = [self _setupOpenPanel];
|
NSOpenPanel *openPanel = [self _setupOpenPanel];
|
||||||
|
|
||||||
if ([self runModalOpenPanel:openPanel forTypes:types])
|
if ([self runModalOpenPanel:openPanel forTypes:types])
|
||||||
{
|
{
|
||||||
return [openPanel URLs];
|
return [openPanel URLs];
|
||||||
}
|
}
|
||||||
|
|
||||||
return nil;
|
return nil;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
- (IBAction)saveAllDocuments:(id)sender
|
- (IBAction)saveAllDocuments:(id)sender
|
||||||
{
|
{
|
||||||
NSDocument *document;
|
NSDocument *document;
|
||||||
NSEnumerator *docEnum = [_documents objectEnumerator];
|
NSEnumerator *docEnum = [_documents objectEnumerator];
|
||||||
|
|
||||||
while ((document = [docEnum nextObject]))
|
while ((document = [docEnum nextObject]))
|
||||||
|
{
|
||||||
|
if ([document isDocumentEdited]) //maybe we should save regardless...
|
||||||
{
|
{
|
||||||
if ([document isDocumentEdited]) //maybe we should save regardless...
|
[document saveDocument:sender];
|
||||||
{
|
|
||||||
[document saveDocument:sender];
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
- (IBAction)openDocument:(id)sender
|
- (IBAction)openDocument:(id)sender
|
||||||
{
|
{
|
||||||
NSEnumerator *fileEnum = [[self fileNamesFromRunningOpenPanel] objectEnumerator];
|
NSEnumerator *fileEnum = [[self fileNamesFromRunningOpenPanel] objectEnumerator];
|
||||||
NSString *filename;
|
NSString *filename;
|
||||||
|
|
||||||
while ((filename = [fileEnum nextObject]))
|
while ((filename = [fileEnum nextObject]))
|
||||||
{
|
{
|
||||||
[self openDocumentWithContentsOfFile:filename display:YES];
|
[self openDocumentWithContentsOfFile:filename display:YES];
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
- (IBAction)newDocument:(id)sender
|
- (IBAction)newDocument:(id)sender
|
||||||
{
|
{
|
||||||
[self openUntitledDocumentOfType:[self _defaultType] display:YES];
|
[self openUntitledDocumentOfType:[self _defaultType] display:YES];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
- (BOOL)closeAllDocuments
|
- (BOOL)closeAllDocuments
|
||||||
{
|
{
|
||||||
NSDocument *document;
|
NSDocument *document;
|
||||||
NSEnumerator *docEnum = [_documents objectEnumerator];
|
NSEnumerator *docEnum = [_documents objectEnumerator];
|
||||||
|
|
||||||
while ((document = [docEnum nextObject]))
|
while ((document = [docEnum nextObject]))
|
||||||
{
|
{
|
||||||
if (![document canCloseDocument]) return NO;
|
if (![document canCloseDocument]) return NO;
|
||||||
[document close];
|
[document close];
|
||||||
[self _removeDocument:document];
|
[self _removeDocument:document];
|
||||||
}
|
}
|
||||||
|
|
||||||
return YES;
|
return YES;
|
||||||
}
|
}
|
||||||
|
|
||||||
- (BOOL)reviewUnsavedDocumentsWithAlertTitle:(NSString *)title cancellable:(BOOL)cancellable
|
- (BOOL)reviewUnsavedDocumentsWithAlertTitle:(NSString *)title cancellable:(BOOL)cancellable
|
||||||
{
|
{
|
||||||
//FIXME -- localize.
|
//FIXME -- localize.
|
||||||
NSString *cancelString = (cancellable)? @"Cancel" : nil;
|
NSString *cancelString = (cancellable)? @"Cancel" : nil;
|
||||||
int result;
|
int result;
|
||||||
|
|
||||||
if (![self hasEditedDocuments]) return YES;
|
if (![self hasEditedDocuments]) return YES;
|
||||||
|
|
||||||
result = NSRunAlertPanel(title, @"You have unsaved documents.",
|
result = NSRunAlertPanel(title, @"You have unsaved documents.",
|
||||||
@"Review Unsaved", cancelString, @"Quit Anyways");
|
@"Review Unsaved", cancelString, @"Quit Anyways");
|
||||||
|
|
||||||
#define ReviewUnsaved NSAlertDefaultReturn
|
#define ReviewUnsaved NSAlertDefaultReturn
|
||||||
#define Cancel NSAlertAlternateReturn
|
#define Cancel NSAlertAlternateReturn
|
||||||
#define QuitAnyways NSAlertOtherReturn
|
#define QuitAnyways NSAlertOtherReturn
|
||||||
|
|
||||||
switch (result)
|
switch (result)
|
||||||
{
|
{
|
||||||
case ReviewUnsaved: return [self closeAllDocuments];
|
case ReviewUnsaved: return [self closeAllDocuments];
|
||||||
case QuitAnyways: return YES;
|
case QuitAnyways: return YES;
|
||||||
case Cancel:
|
case Cancel:
|
||||||
default: return NO;
|
default: return NO;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#ifdef OPENSTEP_ONLY
|
#ifdef OPENSTEP_ONLY
|
||||||
|
@ -326,148 +337,157 @@ static NSDictionary *TypeInfoForName(NSArray *types, NSString *typeName)
|
||||||
*/
|
*/
|
||||||
- (BOOL)application:(NSApplication *)sender openFile:(NSString *)filename
|
- (BOOL)application:(NSApplication *)sender openFile:(NSString *)filename
|
||||||
{
|
{
|
||||||
return [self openDocumentWithContentsOfFile:filename display:YES] ? YES : NO;
|
return [self openDocumentWithContentsOfFile:filename display:YES] ? YES : NO;
|
||||||
}
|
}
|
||||||
|
|
||||||
- (BOOL)application:(NSApplication *)sender openTempFile:(NSString *)filename;
|
- (BOOL)application:(NSApplication *)sender openTempFile:(NSString *)filename;
|
||||||
{
|
{
|
||||||
return [self openDocumentWithContentsOfFile:filename display:YES] ? YES : NO;
|
return [self openDocumentWithContentsOfFile:filename display:YES] ? YES : NO;
|
||||||
}
|
}
|
||||||
|
|
||||||
- (BOOL)applicationOpenUntitledFile:(NSApplication *)sender
|
- (BOOL)applicationOpenUntitledFile:(NSApplication *)sender
|
||||||
{
|
{
|
||||||
return [self openUntitledDocumentOfType:[self _defaultType] display:YES] ? YES : NO;
|
return [self openUntitledDocumentOfType:[self _defaultType] display:YES] ? YES : NO;
|
||||||
}
|
}
|
||||||
|
|
||||||
- (BOOL)application:(id)sender openFileWithoutUI:(NSString *)filename
|
- (BOOL)application:(id)sender openFileWithoutUI:(NSString *)filename
|
||||||
{
|
{
|
||||||
return [self openDocumentWithContentsOfFile:filename display:NO] ? YES : NO;
|
return [self openDocumentWithContentsOfFile:filename display:NO] ? YES : NO;
|
||||||
}
|
}
|
||||||
|
|
||||||
- (BOOL)applicationShouldTerminate:(NSApplication *)sender
|
- (BOOL)applicationShouldTerminate:(NSApplication *)sender
|
||||||
{
|
{
|
||||||
return [self reviewUnsavedDocumentsWithAlertTitle:@"Quit" cancellable:YES];
|
return [self reviewUnsavedDocumentsWithAlertTitle:@"Quit" cancellable:YES];
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
- (void)_workspaceWillPowerOff:(NSNotification *)notification
|
- (void)_workspaceWillPowerOff:(NSNotification *)notification
|
||||||
{
|
{
|
||||||
// FIXME -- localize.
|
// FIXME -- localize.
|
||||||
[self reviewUnsavedDocumentsWithAlertTitle:@"Power" cancellable:NO];
|
[self reviewUnsavedDocumentsWithAlertTitle:@"Power" cancellable:NO];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
- (NSArray *)documents
|
- (NSArray *)documents
|
||||||
{
|
{
|
||||||
return _documents;
|
return _documents;
|
||||||
}
|
}
|
||||||
|
|
||||||
- (BOOL)hasEditedDocuments;
|
- (BOOL)hasEditedDocuments;
|
||||||
{
|
{
|
||||||
int i, count = [_documents count];
|
int i, count = [_documents count];
|
||||||
|
|
||||||
for (i=0; i<count; i++)
|
for (i=0; i<count; i++)
|
||||||
{
|
{
|
||||||
if ([[_documents objectAtIndex:i] isDocumentEdited]) return YES;
|
if ([[_documents objectAtIndex:i] isDocumentEdited])
|
||||||
}
|
return YES;
|
||||||
|
}
|
||||||
|
|
||||||
return NO;
|
return NO;
|
||||||
}
|
}
|
||||||
|
|
||||||
- (id)currentDocument
|
- (id)currentDocument
|
||||||
{
|
{
|
||||||
return [self documentForWindow:[[NSApplication sharedApplication] mainWindow]];
|
return [self documentForWindow:[[NSApplication sharedApplication] mainWindow]];
|
||||||
}
|
}
|
||||||
|
|
||||||
- (NSString *)currentDirectory
|
- (NSString *)currentDirectory
|
||||||
{
|
{
|
||||||
NSFileManager *manager = [NSFileManager defaultManager];
|
NSFileManager *manager = [NSFileManager defaultManager];
|
||||||
NSDocument *currentDocument = [self currentDocument];
|
NSDocument *currentDocument = [self currentDocument];
|
||||||
NSString *directory = [[currentDocument fileName] stringByDeletingLastPathComponent];
|
NSString *directory = [[currentDocument fileName] stringByDeletingLastPathComponent];
|
||||||
BOOL isDir = NO;
|
BOOL isDir = NO;
|
||||||
|
|
||||||
if (directory &&
|
if (directory &&
|
||||||
[manager fileExistsAtPath:directory isDirectory:&isDir] && isDir) return directory;
|
[manager fileExistsAtPath:directory isDirectory:&isDir] && isDir)
|
||||||
//FIXME -- need to remember last saved directory, and return that here.
|
return directory;
|
||||||
//Only return NSHomeDirectory if nothing's been saved yet.
|
|
||||||
return NSHomeDirectory();
|
//FIXME -- need to remember last saved directory, and return that here.
|
||||||
|
//Only return NSHomeDirectory if nothing's been saved yet.
|
||||||
|
return NSHomeDirectory();
|
||||||
}
|
}
|
||||||
|
|
||||||
- (id)documentForWindow:(NSWindow *)window
|
- (id)documentForWindow:(NSWindow *)window
|
||||||
{
|
{
|
||||||
id document;
|
id document;
|
||||||
|
|
||||||
if (window == nil) return nil;
|
if (window == nil)
|
||||||
if (![[window windowController] isKindOfClass:[NSWindowController class]]) return nil;
|
return nil;
|
||||||
|
|
||||||
document = [[window windowController] document];
|
if (![[window windowController] isKindOfClass:[NSWindowController class]])
|
||||||
if (![document isKindOfClass:[NSDocument class]]) return nil;
|
return nil;
|
||||||
return document;
|
|
||||||
|
document = [[window windowController] document];
|
||||||
|
|
||||||
|
if (![document isKindOfClass:[NSDocument class]])
|
||||||
|
return nil;
|
||||||
|
|
||||||
|
return document;
|
||||||
}
|
}
|
||||||
|
|
||||||
- (id)documentForFileName:(NSString *)fileName
|
- (id)documentForFileName:(NSString *)fileName
|
||||||
{
|
{
|
||||||
int i, count = [_documents count];
|
int i, count = [_documents count];
|
||||||
|
|
||||||
for (i=0; i<count; i++)
|
for (i=0; i<count; i++)
|
||||||
{
|
{
|
||||||
NSDocument *document = [_documents objectAtIndex:i];
|
NSDocument *document = [_documents objectAtIndex:i];
|
||||||
|
|
||||||
if ([[document fileName] isEqualToString:fileName])
|
if ([[document fileName] isEqualToString:fileName])
|
||||||
return document;
|
return document;
|
||||||
}
|
}
|
||||||
|
|
||||||
return nil;
|
return nil;
|
||||||
}
|
}
|
||||||
|
|
||||||
- (BOOL)validateMenuItem:(NSMenuItem *)anItem
|
- (BOOL)validateMenuItem:(NSMenuItem *)anItem
|
||||||
{
|
{
|
||||||
if ([anItem action] == @selector(saveAllDocuments:))
|
if ([anItem action] == @selector(saveAllDocuments:))
|
||||||
return [self hasEditedDocuments];
|
return [self hasEditedDocuments];
|
||||||
return YES;
|
return YES;
|
||||||
}
|
}
|
||||||
|
|
||||||
- (NSString *)displayNameForType:(NSString *)type
|
- (NSString *)displayNameForType:(NSString *)type
|
||||||
{
|
{
|
||||||
NSString *name = [TYPE_INFO(type) objectForKey:NSHumanReadableNameKey];
|
NSString *name = [TYPE_INFO(type) objectForKey:NSHumanReadableNameKey];
|
||||||
|
|
||||||
return name? name : type;
|
return name? name : type;
|
||||||
}
|
}
|
||||||
|
|
||||||
- (NSString *)typeFromFileExtension:(NSString *)fileExtension
|
- (NSString *)typeFromFileExtension:(NSString *)fileExtension
|
||||||
{
|
{
|
||||||
int i, count = [_types count];
|
int i, count = [_types count];
|
||||||
|
|
||||||
for (i=0; i<count;i++)
|
for (i=0; i<count;i++)
|
||||||
|
{
|
||||||
|
NSDictionary *typeInfo = [_types objectAtIndex:i];
|
||||||
|
|
||||||
|
if ([[typeInfo objectForKey:NSUnixExtensionsKey] containsObject:fileExtension] ||
|
||||||
|
[[typeInfo objectForKey:NSDOSExtensionsKey] containsObject:fileExtension])
|
||||||
{
|
{
|
||||||
NSDictionary *typeInfo = [_types objectAtIndex:i];
|
return [typeInfo objectForKey:NSNameKey];
|
||||||
|
|
||||||
if ([[typeInfo objectForKey:NSUnixExtensionsKey] containsObject:fileExtension] ||
|
|
||||||
[[typeInfo objectForKey:NSDOSExtensionsKey] containsObject:fileExtension])
|
|
||||||
{
|
|
||||||
return [typeInfo objectForKey:NSNameKey];
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
return nil;
|
return nil;
|
||||||
}
|
}
|
||||||
|
|
||||||
- (NSArray *)fileExtensionsFromType:(NSString *)type
|
- (NSArray *)fileExtensionsFromType:(NSString *)type
|
||||||
{
|
{
|
||||||
NSDictionary *typeInfo = TYPE_INFO(type);
|
NSDictionary *typeInfo = TYPE_INFO(type);
|
||||||
NSArray *unixExtensions = [typeInfo objectForKey:NSUnixExtensionsKey];
|
NSArray *unixExtensions = [typeInfo objectForKey:NSUnixExtensionsKey];
|
||||||
NSArray *dosExtensions = [typeInfo objectForKey:NSDOSExtensionsKey];
|
NSArray *dosExtensions = [typeInfo objectForKey:NSDOSExtensionsKey];
|
||||||
|
|
||||||
if (!dosExtensions) return unixExtensions;
|
if (!dosExtensions) return unixExtensions;
|
||||||
if (!unixExtensions) return dosExtensions;
|
if (!unixExtensions) return dosExtensions;
|
||||||
return [unixExtensions arrayByAddingObjectsFromArray:dosExtensions];
|
return [unixExtensions arrayByAddingObjectsFromArray:dosExtensions];
|
||||||
}
|
}
|
||||||
|
|
||||||
- (Class)documentClassForType:(NSString *)type
|
- (Class)documentClassForType:(NSString *)type
|
||||||
{
|
{
|
||||||
NSString *className = [TYPE_INFO(type) objectForKey:NSDocumentClassKey];
|
NSString *className = [TYPE_INFO(type) objectForKey:NSDocumentClassKey];
|
||||||
|
|
||||||
return className? NSClassFromString(className) : Nil;
|
return className? NSClassFromString(className) : Nil;
|
||||||
}
|
}
|
||||||
|
|
||||||
static NSString *NSEditorRole = @"Editor";
|
static NSString *NSEditorRole = @"Editor";
|
||||||
|
@ -476,52 +496,52 @@ static NSString *NSNoRole = @"None";
|
||||||
|
|
||||||
- (NSArray *)_editorAndViewerTypesForClass:(Class)documentClass
|
- (NSArray *)_editorAndViewerTypesForClass:(Class)documentClass
|
||||||
{
|
{
|
||||||
int i, count = [_types count];
|
int i, count = [_types count];
|
||||||
NSMutableArray *types = [NSMutableArray arrayWithCapacity:count];
|
NSMutableArray *types = [NSMutableArray arrayWithCapacity:count];
|
||||||
NSString *docClassName = NSStringFromClass(documentClass);
|
NSString *docClassName = NSStringFromClass(documentClass);
|
||||||
|
|
||||||
for (i=0; i<count; i++)
|
for (i=0; i<count; i++)
|
||||||
|
{
|
||||||
|
NSDictionary *typeInfo = [_types objectAtIndex:i];
|
||||||
|
NSString *className = [typeInfo objectForKey:NSDocumentClassKey];
|
||||||
|
NSString *role = [typeInfo objectForKey:NSRoleKey];
|
||||||
|
|
||||||
|
if ([docClassName isEqualToString:className] &&
|
||||||
|
(role == nil || [role isEqual:NSEditorRole] || [role isEqual:NSViewerRole]))
|
||||||
{
|
{
|
||||||
NSDictionary *typeInfo = [_types objectAtIndex:i];
|
[types addObject:[typeInfo objectForKey:NSNameKey]];
|
||||||
NSString *className = [typeInfo objectForKey:NSDocumentClassKey];
|
|
||||||
NSString *role = [typeInfo objectForKey:NSRoleKey];
|
|
||||||
|
|
||||||
if ([docClassName isEqualToString:className] &&
|
|
||||||
(role == nil || [role isEqual:NSEditorRole] || [role isEqual:NSViewerRole]))
|
|
||||||
{
|
|
||||||
[types addObject:[typeInfo objectForKey:NSNameKey]];
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
return types;
|
return types;
|
||||||
}
|
}
|
||||||
|
|
||||||
- (NSArray *)_editorTypesForClass:(Class)documentClass
|
- (NSArray *)_editorTypesForClass:(Class)documentClass
|
||||||
{
|
{
|
||||||
int i, count = [_types count];
|
int i, count = [_types count];
|
||||||
NSMutableArray *types = [NSMutableArray arrayWithCapacity:count];
|
NSMutableArray *types = [NSMutableArray arrayWithCapacity:count];
|
||||||
NSString *docClassName = NSStringFromClass(documentClass);
|
NSString *docClassName = NSStringFromClass(documentClass);
|
||||||
|
|
||||||
for (i=0; i<count; i++)
|
for (i=0; i<count; i++)
|
||||||
|
{
|
||||||
|
NSDictionary *typeInfo = [_types objectAtIndex:i];
|
||||||
|
NSString *className = [typeInfo objectForKey:NSDocumentClassKey];
|
||||||
|
NSString *role = [typeInfo objectForKey:NSRoleKey];
|
||||||
|
|
||||||
|
if ([docClassName isEqualToString:className] &&
|
||||||
|
(role == nil || [role isEqual:NSEditorRole]))
|
||||||
{
|
{
|
||||||
NSDictionary *typeInfo = [_types objectAtIndex:i];
|
[types addObject:[typeInfo objectForKey:NSNameKey]];
|
||||||
NSString *className = [typeInfo objectForKey:NSDocumentClassKey];
|
|
||||||
NSString *role = [typeInfo objectForKey:NSRoleKey];
|
|
||||||
|
|
||||||
if ([docClassName isEqualToString:className] &&
|
|
||||||
(role == nil || [role isEqual:NSEditorRole]))
|
|
||||||
{
|
|
||||||
[types addObject:[typeInfo objectForKey:NSNameKey]];
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
return types;
|
return types;
|
||||||
}
|
}
|
||||||
|
|
||||||
- (NSArray *)_exportableTypesForClass:(Class)documentClass
|
- (NSArray *)_exportableTypesForClass:(Class)documentClass
|
||||||
{
|
{
|
||||||
// Dunno what this method is for; maybe looks for filter types
|
// Dunno what this method is for; maybe looks for filter types
|
||||||
return [self _editorTypesForClass:documentClass];
|
return [self _editorTypesForClass:documentClass];
|
||||||
}
|
}
|
||||||
|
|
||||||
@end
|
@end
|
||||||
|
|
|
@ -105,7 +105,6 @@ static NSString* NSMenuLocationsKey = @"NSMenuLocations";
|
||||||
menu_follow_transient = NO;
|
menu_follow_transient = NO;
|
||||||
menu_is_beholdenToPopUpButton = YES;
|
menu_is_beholdenToPopUpButton = YES;
|
||||||
ASSIGN(menu_popb, popb);
|
ASSIGN(menu_popb, popb);
|
||||||
// menu_popb = popb;
|
|
||||||
|
|
||||||
menu_changed = YES;
|
menu_changed = YES;
|
||||||
/* According to the spec, menus do autoenable by default */
|
/* According to the spec, menus do autoenable by default */
|
||||||
|
@ -171,8 +170,8 @@ static NSString* NSMenuLocationsKey = @"NSMenuLocations";
|
||||||
defer: NO];
|
defer: NO];
|
||||||
|
|
||||||
titleView = [NSMenuWindowTitleView new];
|
titleView = [NSMenuWindowTitleView new];
|
||||||
[titleView setFrameOrigin: NSMakePoint(0, winRect.size.height-21)];
|
[titleView setFrameOrigin: NSMakePoint(0, winRect.size.height-22)];
|
||||||
[titleView setFrameSize: NSMakeSize (winRect.size.width, 21)];
|
[titleView setFrameSize: NSMakeSize (winRect.size.width, 22)];
|
||||||
[[aWindow contentView] addSubview:menu_view];
|
[[aWindow contentView] addSubview:menu_view];
|
||||||
[[aWindow contentView] addSubview:titleView];
|
[[aWindow contentView] addSubview:titleView];
|
||||||
[titleView setMenu: self];
|
[titleView setMenu: self];
|
||||||
|
@ -515,7 +514,7 @@ static NSString* NSMenuLocationsKey = @"NSMenuLocations";
|
||||||
NSPoint subOrigin = [win_link convertBaseToScreen: NSMakePoint(aRect.origin.x, aRect.origin.y)];
|
NSPoint subOrigin = [win_link convertBaseToScreen: NSMakePoint(aRect.origin.x, aRect.origin.y)];
|
||||||
|
|
||||||
return NSMakePoint (frame.origin.x + frame.size.width + 1,
|
return NSMakePoint (frame.origin.x + frame.size.width + 1,
|
||||||
subOrigin.y - (submenuFrame.size.height - 41));
|
subOrigin.y - (submenuFrame.size.height - 42));
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
|
@ -727,20 +726,17 @@ static NSString* NSMenuLocationsKey = @"NSMenuLocations";
|
||||||
|
|
||||||
if (!menu_is_beholdenToPopUpButton)
|
if (!menu_is_beholdenToPopUpButton)
|
||||||
{
|
{
|
||||||
size.height += 21;
|
size.height += 22;
|
||||||
[aWindow setContentSize: size];
|
[aWindow setContentSize: size];
|
||||||
[bWindow setContentSize: size];
|
[bWindow setContentSize: size];
|
||||||
[menu_view setFrameOrigin: NSMakePoint(0, 0)];
|
[menu_view setFrameOrigin: NSMakePoint(0, 0)];
|
||||||
[titleView setFrame: NSMakeRect(0,size.height-21,size.width,21)];
|
[titleView setFrame: NSMakeRect(0,size.height-22,size.width,22)];
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
[aWindow setContentSize: size];
|
[aWindow setContentSize: size];
|
||||||
}
|
}
|
||||||
|
|
||||||
// FIXME, popup sets itself up.
|
|
||||||
// [menu_view setNeedsDisplay:YES];
|
|
||||||
|
|
||||||
[aWindow display];
|
[aWindow display];
|
||||||
|
|
||||||
menu_changed = NO;
|
menu_changed = NO;
|
||||||
|
@ -862,11 +858,9 @@ NSArray* array;
|
||||||
if (menu_changed)
|
if (menu_changed)
|
||||||
[self sizeToFit];
|
[self sizeToFit];
|
||||||
|
|
||||||
if (menu_supermenu && ![self isTornOff]) // query super menu for
|
if (menu_supermenu && ![self isTornOff]) // query super menu for
|
||||||
{ // position
|
{ // position
|
||||||
NSPoint location = [menu_supermenu locationForSubmenu: self];
|
[aWindow setFrameOrigin:[menu_supermenu locationForSubmenu: self]];
|
||||||
|
|
||||||
[aWindow setFrameOrigin: location];
|
|
||||||
menu_supermenu->menu_attached_menu = self;
|
menu_supermenu->menu_attached_menu = self;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
|
|
|
@ -130,65 +130,94 @@ static float GSMenuBarHeight = 25.0; // a guess.
|
||||||
|
|
||||||
[self lockFocus];
|
[self lockFocus];
|
||||||
|
|
||||||
if (index == -1) {
|
if (index == -1)
|
||||||
if (menuv_highlightedItemIndex != -1) {
|
{
|
||||||
anItem = [menuv_items_link objectAtIndex: menuv_highlightedItemIndex];
|
if (menuv_highlightedItemIndex != -1)
|
||||||
|
{
|
||||||
|
NSRect aRect = [self rectOfItemAtIndex: menuv_highlightedItemIndex];
|
||||||
|
|
||||||
[anItem highlight: NO
|
anItem = [menuv_items_link objectAtIndex: menuv_highlightedItemIndex];
|
||||||
withFrame: [self rectOfItemAtIndex: menuv_highlightedItemIndex]
|
|
||||||
inView: self];
|
|
||||||
|
|
||||||
if ([anItem hasSubmenu] && ![[anItem target] isTornOff])
|
[anItem highlight: NO
|
||||||
[[anItem target] close];
|
withFrame: aRect
|
||||||
else if ([anItem hasSubmenu] && [[anItem target] isTornOff])
|
inView: self];
|
||||||
[[anItem target] closeTransient];
|
|
||||||
|
|
||||||
[anItem setState: 0];
|
[self setNeedsDisplayInRect: aRect];
|
||||||
menuv_highlightedItemIndex = -1;
|
|
||||||
|
[window flushWindow];
|
||||||
|
|
||||||
|
if ([anItem hasSubmenu]
|
||||||
|
&& ![[anItem target] isTornOff])
|
||||||
|
[[anItem target] close];
|
||||||
|
else if ([anItem hasSubmenu]
|
||||||
|
&& [[anItem target] isTornOff])
|
||||||
|
[[anItem target] closeTransient];
|
||||||
|
|
||||||
|
[anItem setState: 0];
|
||||||
|
menuv_highlightedItemIndex = -1;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
} else if (index >= 0) {
|
else if (index >= 0)
|
||||||
if ( menuv_highlightedItemIndex != -1 ) {
|
{
|
||||||
|
if ( menuv_highlightedItemIndex != -1)
|
||||||
|
{
|
||||||
|
NSRect aRect = [self rectOfItemAtIndex: menuv_highlightedItemIndex];
|
||||||
|
|
||||||
anItem = [menuv_items_link objectAtIndex: menuv_highlightedItemIndex];
|
anItem = [menuv_items_link objectAtIndex: menuv_highlightedItemIndex];
|
||||||
|
|
||||||
[anItem highlight: NO
|
[anItem highlight: NO
|
||||||
withFrame: [self rectOfItemAtIndex: menuv_highlightedItemIndex]
|
withFrame: aRect
|
||||||
inView: self];
|
inView: self];
|
||||||
|
|
||||||
if ([anItem hasSubmenu] && ![[anItem target] isTornOff])
|
[self setNeedsDisplayInRect: aRect];
|
||||||
[[anItem target] close];
|
|
||||||
else if ([anItem hasSubmenu] && [[anItem target] isTornOff])
|
|
||||||
[[anItem target] closeTransient];
|
|
||||||
|
|
||||||
[anItem setState: 0];
|
[window flushWindow];
|
||||||
}
|
|
||||||
|
|
||||||
if (menuv_highlightedItemIndex != index) {
|
if ([anItem hasSubmenu]
|
||||||
|
&& ![[anItem target] isTornOff])
|
||||||
|
[[anItem target] close];
|
||||||
|
else if ([anItem hasSubmenu]
|
||||||
|
&& [[anItem target] isTornOff])
|
||||||
|
[[anItem target] closeTransient];
|
||||||
|
|
||||||
anItem = [menuv_items_link objectAtIndex: index];
|
[anItem setState: 0];
|
||||||
|
|
||||||
if ([anItem isEnabled])
|
|
||||||
{
|
|
||||||
[anItem highlight: YES
|
|
||||||
withFrame: [self rectOfItemAtIndex: index]
|
|
||||||
inView: self];
|
|
||||||
|
|
||||||
[anItem setState: 1];
|
|
||||||
|
|
||||||
if ([anItem hasSubmenu] && ![[anItem target] isTornOff])
|
|
||||||
[[anItem target] display];
|
|
||||||
else if ([anItem hasSubmenu] && [[anItem target] isTornOff])
|
|
||||||
[[anItem target] displayTransient];
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// set ivar to new index
|
if (index != menuv_highlightedItemIndex)
|
||||||
menuv_highlightedItemIndex = index;
|
{
|
||||||
} else {
|
anItem = [menuv_items_link objectAtIndex: index];
|
||||||
menuv_highlightedItemIndex = -1;
|
|
||||||
|
if ([anItem isEnabled])
|
||||||
|
{
|
||||||
|
NSRect aRect = [self rectOfItemAtIndex: index];
|
||||||
|
|
||||||
|
[anItem highlight: YES
|
||||||
|
withFrame: aRect
|
||||||
|
inView: self];
|
||||||
|
[self setNeedsDisplayInRect: aRect];
|
||||||
|
|
||||||
|
[window flushWindow];
|
||||||
|
|
||||||
|
if ([anItem hasSubmenu]
|
||||||
|
&& ![[anItem target] isTornOff])
|
||||||
|
[[anItem target] display];
|
||||||
|
else if ([anItem hasSubmenu]
|
||||||
|
&& [[anItem target] isTornOff])
|
||||||
|
[[anItem target] displayTransient];
|
||||||
|
|
||||||
|
[anItem setState: 1];
|
||||||
|
}
|
||||||
|
|
||||||
|
// set ivar to new index
|
||||||
|
menuv_highlightedItemIndex = index;
|
||||||
|
}
|
||||||
|
else if (menuv_highlightedItemIndex == index)
|
||||||
|
{
|
||||||
|
menuv_highlightedItemIndex = -1;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
[self unlockFocus];
|
[self unlockFocus];
|
||||||
[window flushWindow];
|
|
||||||
}
|
}
|
||||||
|
|
||||||
- (int)highlightedItemIndex
|
- (int)highlightedItemIndex
|
||||||
|
@ -522,16 +551,18 @@ static float GSMenuBarHeight = 25.0; // a guess.
|
||||||
{ // Recursive menu close & deselect.
|
{ // Recursive menu close & deselect.
|
||||||
if ([aMenu supermenu] && aMenu != menuv_menu)
|
if ([aMenu supermenu] && aMenu != menuv_menu)
|
||||||
{
|
{
|
||||||
[[[aMenu supermenu] menuView] setHighlightedItemIndex: -1];
|
[[aMenu menuView] setHighlightedItemIndex: -1];
|
||||||
aMenu = [aMenu supermenu];
|
aMenu = [aMenu supermenu];
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
finished = YES;
|
finished = YES;
|
||||||
|
|
||||||
[window flushWindow];
|
// [window flushWindow];
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
[self setHighlightedItemIndex: index];
|
[self setHighlightedItemIndex: index];
|
||||||
|
|
||||||
lastIndex = index;
|
lastIndex = index;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -726,8 +757,6 @@ cell do the following */
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
finished = YES;
|
finished = YES;
|
||||||
|
|
||||||
[window flushWindow];
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
|
@ -770,7 +799,7 @@ cell do the following */
|
||||||
{ // Recursive menu close & deselect.
|
{ // Recursive menu close & deselect.
|
||||||
if ([aMenu supermenu])
|
if ([aMenu supermenu])
|
||||||
{
|
{
|
||||||
[[[aMenu supermenu] menuView] setHighlightedItemIndex: -1];
|
[[aMenu menuView] setHighlightedItemIndex: -1];
|
||||||
aMenu = [aMenu supermenu];
|
aMenu = [aMenu supermenu];
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
|
|
|
@ -27,7 +27,6 @@
|
||||||
#include <AppKit/NSCStringText.h>
|
#include <AppKit/NSCStringText.h>
|
||||||
#include <AppKit/NSEvent.h>
|
#include <AppKit/NSEvent.h>
|
||||||
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Dummy definitions provided here to avoid errors when not linking with
|
* Dummy definitions provided here to avoid errors when not linking with
|
||||||
* a back end.
|
* a back end.
|
||||||
|
@ -302,4 +301,3 @@ float NSLinkFrameThickness(void)
|
||||||
{
|
{
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue