libs-gui/Source/NSDocumentController.m
Fred Kiefer 31c1b346ac Commented out unused static variables.
Dummy implementation of new methods.
In [openDocumentWithContentsOfFile:display:] call
[noteNewRecentDocument:].
In [openDocumentWithContentsOfURL:display:] only call
[noteNewRecentDocumentURL:] when document is not nil.
Implemented [noteNewRecentDocument:].


git-svn-id: svn+ssh://svn.gna.org/svn/gnustep/libs/gui/trunk@10606 72102866-910b-0410-8b05-ffd578937521
2001-07-28 22:40:07 +00:00

704 lines
16 KiB
Objective-C

/*
NSDocumentController.m
The document controller class
Copyright (C) 1999 Free Software Foundation, Inc.
Author: Carl Lindberg <Carl.Lindberg@hbo.com>
Date: 1999
Modifications: Fred Kiefer <FredKiefer@gmx.de>
Date: June 2000
This file is part of the GNUstep GUI Library.
This library is free software; you can redistribute it and/or
modify it under the terms of the GNU Library General Public
License as published by the Free Software Foundation; either
version 2 of the License, or (at your option) any later version.
This library is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
Library General Public License for more details.
You should have received a copy of the GNU Library General Public
License along with this library; see the file COPYING.LIB.
If not, write to the Free Software Foundation,
59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
*/
#include <AppKit/NSDocumentController.h>
#include <AppKit/NSOpenPanel.h>
#include <AppKit/NSApplication.h>
#include <AppKit/NSMenuItem.h>
#include <AppKit/NSWorkspace.h>
#include <AppKit/NSDocumentFrameworkPrivate.h>
static NSString *NSTypesKey = @"NSTypes";
static NSString *NSNameKey = @"NSName";
static NSString *NSRoleKey = @"NSRole";
static NSString *NSHumanReadableNameKey = @"NSHumanReadableName";
static NSString *NSUnixExtensionsKey = @"NSUnixExtensions";
static NSString *NSDOSExtensionsKey = @"NSDOSExtensions";
//static NSString *NSMacOSTypesKey = @"NSMacOSTypes";
//static NSString *NSMIMETypesKey = @"NSMIMETypes";
static NSString *NSDocumentClassKey = @"NSDocumentClass";
#define TYPE_INFO(name) TypeInfoForName(_types, name)
static NSDictionary *TypeInfoForName (NSArray *types, NSString *typeName)
{
int i, count = [types count];
for (i = 0; i < count; i++)
{
NSDictionary *dict = [types objectAtIndex: i];
if ([[dict objectForKey: NSNameKey] isEqualToString: typeName])
{
return dict;
}
}
return nil;
}
@implementation NSDocumentController
+ (void)initialize
{
}
+ (id) documentController //private
{
return [self sharedDocumentController];
}
+ (id) allocWithZone: (NSZone *)zone
{
return [self sharedDocumentController];
}
+ (id) sharedDocumentController
{
static id instance = nil;
if (instance == nil)
{
instance = [[super allocWithZone:NULL] init];
}
return instance;
}
- init
{
NSDictionary *customDict = [[NSBundle mainBundle] infoDictionary];
ASSIGN (_types, [customDict objectForKey: NSTypesKey]);
_documents = [[NSMutableArray alloc] init];
// FIXME: Should fill this list from some stored values
_recentDocuments = [[NSMutableArray alloc] init];
[self setShouldCreateUI:YES];
[[[NSWorkspace sharedWorkspace] notificationCenter]
addObserver: self
selector: @selector(_workspaceWillPowerOff:)
name: NSWorkspaceWillPowerOffNotification
object: nil];
return self;
}
- (void) dealloc
{
[[[NSWorkspace sharedWorkspace] notificationCenter] removeObserver: self];
RELEASE (_documents);
RELEASE (_recentDocuments);
RELEASE (_types);
[super dealloc];
}
- (BOOL) shouldCreateUI
{
return _controllerFlags.shouldCreateUI;
}
- (void) setShouldCreateUI: (BOOL)flag
{
_controllerFlags.shouldCreateUI = flag;
}
- (id) makeUntitledDocumentOfType: (NSString *)type
{
Class documentClass = [self documentClassForType: type];
return AUTORELEASE ([[documentClass alloc] init]);
}
- (id) makeDocumentWithContentsOfFile: (NSString *)fileName
ofType: (NSString *)type
{
Class documentClass = [self documentClassForType:type];
return AUTORELEASE ([[documentClass alloc] initWithContentsOfFile: fileName
ofType: type]);
}
- (id) makeDocumentWithContentsOfURL: (NSURL *)url ofType: (NSString *)type
{
Class documentClass = [self documentClassForType: type];
return AUTORELEASE ([[documentClass alloc] initWithContentsOfURL: url
ofType: type]);
}
- _defaultType
{
if ([_types count] == 0)
{
return nil; // raise exception?
}
return [[_types objectAtIndex:0] objectForKey:NSNameKey];
}
- (void) addDocument: (NSDocument *)document
{
[_documents addObject: document];
}
- (void) removeDocument: (NSDocument *)document
{
[_documents removeObject: document];
}
- (id) openUntitledDocumentOfType: (NSString*)type display: (BOOL)display
{
NSDocument *document = [self makeUntitledDocumentOfType: type];
if (document == nil)
{
return nil;
}
[self addDocument: document];
if ([self shouldCreateUI])
{
[document makeWindowControllers];
if (display)
{
[document showWindows];
}
}
return document;
}
- (id) openDocumentWithContentsOfFile: (NSString *)fileName
display: (BOOL)display
{
NSDocument *document = [self documentForFileName: fileName];
if (document == nil)
{
NSString *type = [self typeFromFileExtension: [fileName pathExtension]];
if ((document = [self makeDocumentWithContentsOfFile: fileName
ofType: type]))
{
[self addDocument: document];
}
if ([self shouldCreateUI])
{
[document makeWindowControllers];
}
}
// remember this document as opened
[self noteNewRecentDocument: document];
if (display && [self shouldCreateUI])
{
[document showWindows];
}
return document;
}
- (id) openDocumentWithContentsOfURL: (NSURL *)url display: (BOOL)display
{
// Should we only do this if [url isFileURL] is YES?
NSDocument *document = [self documentForFileName: [url path]];
if (document == nil)
{
NSString *type = [self typeFromFileExtension:
[[url path] pathExtension]];
document = [self makeDocumentWithContentsOfURL: url ofType: type];
if (document == nil)
{
return nil;
}
[self addDocument: document];
if ([self shouldCreateUI])
{
[document makeWindowControllers];
}
}
// remember this document as opened
[self noteNewRecentDocumentURL: url];
if (display && [self shouldCreateUI])
{
[document showWindows];
}
return document;
}
- (NSOpenPanel *) _setupOpenPanel
{
NSOpenPanel *openPanel = [NSOpenPanel openPanel];
[openPanel setDirectory: [self currentDirectory]];
[openPanel setAllowsMultipleSelection: YES];
return openPanel;
}
- (int) runModalOpenPanel: (NSOpenPanel *)openPanel
forTypes: (NSArray *)openableFileExtensions
{
return [openPanel runModalForTypes:openableFileExtensions];
}
- (NSArray *) _openableFileExtensions
{
int i, count = [_types count];
NSMutableArray *array = [NSMutableArray arrayWithCapacity: count];
for (i = 0; i < count; i++)
{
NSDictionary *typeInfo = [_types objectAtIndex: i];
[array addObjectsFromArray: [typeInfo objectForKey: NSUnixExtensionsKey]];
[array addObjectsFromArray: [typeInfo objectForKey: NSDOSExtensionsKey]];
}
return array;
}
- (NSArray *) fileNamesFromRunningOpenPanel
{
NSArray *types = [self _openableFileExtensions];
NSOpenPanel *openPanel = [self _setupOpenPanel];
if ([self runModalOpenPanel: openPanel forTypes: types])
{
return [openPanel filenames];
}
return nil;
}
- (NSArray *) URLsFromRunningOpenPanel
{
NSArray *types = [self _openableFileExtensions];
NSOpenPanel *openPanel = [self _setupOpenPanel];
if ([self runModalOpenPanel: openPanel forTypes: types])
{
return [openPanel URLs];
}
return nil;
}
- (IBAction) saveAllDocuments: (id)sender
{
NSDocument *document;
NSEnumerator *docEnum = [_documents objectEnumerator];
while ((document = [docEnum nextObject]))
{
if ([document isDocumentEdited]) //maybe we should save regardless...
{
[document saveDocument: sender];
}
}
}
- (IBAction) openDocument: (id)sender
{
NSEnumerator *fileEnum;
NSString *filename;
fileEnum = [[self fileNamesFromRunningOpenPanel] objectEnumerator];
while ((filename = [fileEnum nextObject]))
{
[self openDocumentWithContentsOfFile: filename display: YES];
}
}
- (IBAction) newDocument: (id)sender
{
[self openUntitledDocumentOfType: [self _defaultType] display: YES];
}
- (BOOL) closeAllDocuments
{
NSDocument *document;
NSEnumerator *docEnum = [_documents objectEnumerator];
while ((document = [docEnum nextObject]))
{
if (![document canCloseDocument])
{
return NO;
}
[document close];
[self removeDocument: document];
}
return YES;
}
- (void)closeAllDocumentsWithDelegate:(id)delegate
didCloseAllSelector:(SEL)didAllCloseSelector
contextInfo:(void *)contextInfo
{
//FIXME
}
- (BOOL) reviewUnsavedDocumentsWithAlertTitle: (NSString *)title
cancellable: (BOOL)cancellable
{
//FIXME -- localize.
NSString *cancelString = (cancellable)? @"Cancel" : nil;
int result;
if (![self hasEditedDocuments])
{
return YES;
}
result = NSRunAlertPanel(title, @"You have unsaved documents.",
@"Review Unsaved", cancelString, @"Quit Anyways");
#define ReviewUnsaved NSAlertDefaultReturn
#define Cancel NSAlertAlternateReturn
#define QuitAnyways NSAlertOtherReturn
switch (result)
{
case ReviewUnsaved: return [self closeAllDocuments];
case QuitAnyways: return YES;
case Cancel:
default: return NO;
}
}
- (void)reviewUnsavedDocumentsWithAlertTitle:(NSString *)title
cancellable:(BOOL)cancellable
delegate:(id)delegate
didReviewAllSelector:(SEL)didReviewAllSelector
contextInfo:(void *)contextInfo
{
// FIXME
}
#ifdef OPENSTEP_ONLY
/*
* App delegate methods. Apple doesn't have these, but they put code
* into NSApplication to call the corresponding NSDocumentController
* methods if the app delegate didn't implement a given delegate method.
*/
- (BOOL) application:(NSApplication *)sender openFile: (NSString *)filename
{
return [self openDocumentWithContentsOfFile:filename display:YES] ? YES : NO;
}
- (BOOL) application:(NSApplication *)sender
openTempFile: (NSString *)filename;
{
return [self openDocumentWithContentsOfFile:filename display:YES] ? YES : NO;
}
- (BOOL) applicationOpenUntitledFile: (NSApplication *)sender
{
return [self openUntitledDocumentOfType: [self _defaultType]
display: YES] ? YES : NO;
}
- (BOOL) application:(id)sender openFileWithoutUI:(NSString *)filename
{
return [self openDocumentWithContentsOfFile: filename display: NO] ?
YES : NO;
}
- (BOOL) applicationShouldTerminate: (NSApplication *)sender
{
return [self reviewUnsavedDocumentsWithAlertTitle: @"Quit"
cancellable: YES];
}
#endif
- (void) _workspaceWillPowerOff: (NSNotification *)notification
{
// FIXME -- localize.
[self reviewUnsavedDocumentsWithAlertTitle: @"Power" cancellable: NO];
}
- (NSArray *) documents
{
return _documents;
}
- (BOOL) hasEditedDocuments;
{
int i, count = [_documents count];
for (i = 0; i < count; i++)
{
if ([[_documents objectAtIndex: i] isDocumentEdited])
{
return YES;
}
}
return NO;
}
- (id) currentDocument
{
return [self documentForWindow:
[[NSApplication sharedApplication] mainWindow]];
}
- (NSString *) currentDirectory
{
NSFileManager *manager = [NSFileManager defaultManager];
NSDocument *currentDocument = [self currentDocument];
NSString *directory = [[currentDocument fileName] stringByDeletingLastPathComponent];
BOOL isDir = NO;
if (directory &&
[manager fileExistsAtPath: directory isDirectory: &isDir] && isDir)
{
return directory;
}
//FIXME -- need to remember last opened directory, and return that here.
//Only return NSHomeDirectory if nothing's been opened yet.
return NSHomeDirectory ();
}
- (id) documentForWindow: (NSWindow *)window
{
id document;
if (window == nil)
{
return nil;
}
if (![[window windowController] isKindOfClass: [NSWindowController class]])
{
return nil;
}
document = [[window windowController] document];
if (![document isKindOfClass:[NSDocument class]])
{
return nil;
}
return document;
}
- (id) documentForFileName: (NSString *)fileName
{
int i, count = [_documents count];
for (i = 0; i < count; i++)
{
NSDocument *document = [_documents objectAtIndex: i];
if ([[document fileName] isEqualToString: fileName])
{
return document;
}
}
return nil;
}
- (BOOL) validateMenuItem: (NSMenuItem *)anItem
{
if ([anItem action] == @selector(saveAllDocuments:))
{
return [self hasEditedDocuments];
}
return YES;
}
- (BOOL)validateUserInterfaceItem:(id <NSValidatedUserInterfaceItem>)anItem
{
// FIXME
return YES;
}
- (NSString *) displayNameForType: (NSString *)type
{
NSString *name = [TYPE_INFO(type) objectForKey: NSHumanReadableNameKey];
return name? name : type;
}
- (NSString *) typeFromFileExtension: (NSString *)fileExtension
{
int i, count = [_types count];
for (i = 0; i < count;i++)
{
NSDictionary *typeInfo = [_types objectAtIndex: i];
if ([[typeInfo objectForKey:NSUnixExtensionsKey]
containsObject: fileExtension] ||
[[typeInfo objectForKey:NSDOSExtensionsKey]
containsObject: fileExtension])
{
return [typeInfo objectForKey: NSNameKey];
}
}
return nil;
}
- (NSArray *) fileExtensionsFromType: (NSString *)type
{
NSDictionary *typeInfo = TYPE_INFO(type);
NSArray *unixExtensions = [typeInfo objectForKey: NSUnixExtensionsKey];
NSArray *dosExtensions = [typeInfo objectForKey: NSDOSExtensionsKey];
if (!dosExtensions) return unixExtensions;
if (!unixExtensions) return dosExtensions;
return [unixExtensions arrayByAddingObjectsFromArray: dosExtensions];
}
- (Class) documentClassForType: (NSString *)type
{
NSString *className = [TYPE_INFO(type) objectForKey: NSDocumentClassKey];
return className? NSClassFromString(className) : Nil;
}
- (IBAction) clearRecentDocuments: (id)sender
{
[_recentDocuments removeAllObjects];
}
// The number of remembered recent documents
#define MAX_DOCS 5
- (void)noteNewRecentDocument:(NSDocument *)aDocument
{
NSString *fileName = [aDocument fileName];
NSURL *anURL = [NSURL fileURLWithPath: fileName];
if (anURL != nil)
[self noteNewRecentDocumentURL: anURL];
}
- (void) noteNewRecentDocumentURL: (NSURL *)anURL
{
unsigned index = [_recentDocuments indexOfObject: anURL];
if (index != NSNotFound)
{
// Always keep the current object at the end of the list
[_recentDocuments removeObjectAtIndex: index];
}
else if ([_recentDocuments count] > MAX_DOCS)
{
[_recentDocuments removeObjectAtIndex: 0];
}
[_recentDocuments addObject: anURL];
}
- (NSArray *) recentDocumentURLs
{
return _recentDocuments;
}
@end
@implementation NSDocumentController (Private)
static NSString *NSEditorRole = @"Editor";
static NSString *NSViewerRole = @"Viewer";
//static NSString *NSNoRole = @"None";
- (NSArray *) _editorAndViewerTypesForClass: (Class)documentClass
{
int i, count = [_types count];
NSMutableArray *types = [NSMutableArray arrayWithCapacity: count];
NSString *docClassName = NSStringFromClass (documentClass);
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]))
{
[types addObject: [typeInfo objectForKey: NSNameKey]];
}
}
return types;
}
- (NSArray *) _editorTypesForClass: (Class)documentClass
{
int i, count = [_types count];
NSMutableArray *types = [NSMutableArray arrayWithCapacity: count];
NSString *docClassName = NSStringFromClass (documentClass);
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]))
{
[types addObject: [typeInfo objectForKey: NSNameKey]];
}
}
return types;
}
- (NSArray *) _exportableTypesForClass: (Class)documentClass
{
// Dunno what this method is for; maybe looks for filter types
return [self _editorTypesForClass: documentClass];
}
@end