mirror of
https://github.com/gnustep/libs-gui.git
synced 2025-04-25 17:31:18 +00:00
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
704 lines
16 KiB
Objective-C
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
|
|
|