VocalEasel/Sources/VLDocument.mm

595 lines
14 KiB
Plaintext
Raw Normal View History

2006-09-11 02:49:56 +00:00
//
2007-04-27 06:41:34 +00:00
// File: VLDocument.mm - VocalEasel document
2006-09-11 02:49:56 +00:00
//
2007-04-27 06:41:34 +00:00
// Author(s):
//
// (MN) Matthias Neeracher
//
2008-04-12 21:33:43 +00:00
// Copyright © 2005-2008 Matthias Neeracher
2006-09-11 02:49:56 +00:00
//
#import "VLDocument.h"
2006-10-22 07:16:29 +00:00
#import "VLXMLDocument.h"
#import "VLLilypondDocument.h"
2006-11-04 08:15:34 +00:00
#import "VLMMADocument.h"
2006-11-12 11:37:36 +00:00
#import "VLMIDIDocument.h"
2007-04-18 08:25:43 +00:00
#import "VLPDFDocument.h"
#import "VLPListDocument.h"
#import "VLPDFWindow.h"
#import "VLLogWindow.h"
#import "VLSheetWindow.h"
2006-11-13 04:26:09 +00:00
#import "VLSoundOut.h"
2006-09-11 02:49:56 +00:00
2007-04-18 08:25:43 +00:00
#import <Quartz/Quartz.h>
@interface PDFDocument (PDFKitSecretsIKnow)
- (NSPrintOperation *)getPrintOperationForPrintInfo:(NSPrintInfo *)printInfo autoRotate:(BOOL)doRotate;
@end
2006-12-04 07:04:24 +00:00
@interface VLSongWrapper : NSObject {
VLSong * wrappedSong;
}
+ (VLSongWrapper *)wrapperWithSong:(VLSong *)song;
- (VLSong *)song;
@end
@implementation VLSongWrapper
- (id)initWithSong:(VLSong *)song
{
if (self = [super init])
wrappedSong = new VLSong(*song);
return self;
}
- (void) dealloc
{
delete wrappedSong;
[super dealloc];
}
+ (VLSongWrapper *)wrapperWithSong:(VLSong *)song
{
return [[[VLSongWrapper alloc] initWithSong:song] autorelease];
}
- (VLSong *)song
{
return wrappedSong;
}
@end
@implementation VLDocument
2006-10-03 17:52:54 +00:00
- (id)init
2006-10-03 17:52:54 +00:00
{
self = [super init];
if (self) {
song = new VLSong;
lilypondTemplate = @"default";
songTitle = @"";
songLyricist = @"";
songComposer = @"";
songArranger = @"";
2006-11-04 08:15:34 +00:00
songGroove = @"Swing";
2008-01-26 16:38:30 +00:00
songTempo = [[NSNumber alloc] initWithInt:120];
chordSize = 6.0f;
lyricSize = 0.0f;
staffSize = 20.0f;
playElements = kVLPlayAccompaniment|kVLPlayMelody|kVLPlayCountIn;
sheetWin = nil;
pdfWin = nil;
logWin = nil;
2006-11-10 08:09:18 +00:00
tmpPath = nil;
vcsWrapper = nil;
repeatVolta = 2;
2007-01-21 11:34:40 +00:00
brandNew = true;
2007-04-16 05:35:52 +00:00
observers = [[NSMutableArray alloc] init];
validTmpFiles = [[NSMutableDictionary alloc] initWithCapacity:10];
2006-12-04 07:04:24 +00:00
[self setHasUndoManager:YES];
undo =
[[VLKeyValueUndo alloc] initWithOwner:self
keysAndNames: [NSDictionary dictionaryWithObjectsAndKeys:
@"", @"songTitle",
@"", @"songLyricist",
@"", @"songComposer",
@"", @"songArranger",
@"", @"songGroove",
@"", @"songTempo",
nil]];
2007-04-18 08:25:43 +00:00
printDoc = nil;
}
return self;
2006-10-03 17:52:54 +00:00
}
- (void)updateChangeCount:(NSDocumentChangeType)changeType
{
[validTmpFiles removeAllObjects];
[super updateChangeCount:changeType];
}
2007-04-16 05:35:52 +00:00
- (void) addObserver:(id)observer
{
[observers addObject:observer];
}
- (void) close
{
[observers makeObjectsPerformSelector:@selector(removeObservers:) withObject:self];
2007-04-20 05:33:58 +00:00
[observers removeAllObjects];
2007-04-16 05:35:52 +00:00
[super close];
}
- (void) dealloc
2006-10-03 17:52:54 +00:00
{
delete song;
2006-10-03 17:52:54 +00:00
[lilypondTemplate release];
[songTitle release];
[songLyricist release];
[songComposer release];
[songArranger release];
[vcsWrapper release];
2006-12-04 07:04:24 +00:00
[undo release];
2007-04-16 05:35:52 +00:00
[observers release];
2006-11-10 08:09:18 +00:00
if (tmpPath) {
[[NSFileManager defaultManager] removeFileAtPath:tmpPath handler:nil];
[tmpPath release];
}
[super dealloc];
2006-10-03 17:52:54 +00:00
}
- (void)removeWindowController:(NSWindowController *)win
{
if (win == logWin)
logWin = nil;
else if (win == pdfWin)
pdfWin = nil;
2007-01-21 11:34:40 +00:00
else if (win == sheetWin)
sheetWin = nil;
[super removeWindowController:win];
}
- (VLLogWindow *)logWin
{
if (!logWin) {
logWin = [[VLLogWindow alloc] initWithWindowNibName: @"VLLogWindow"];
[self addWindowController: logWin];
[logWin release];
}
return logWin;
}
- (VLPDFWindow *)pdfWin
{
if (!pdfWin) {
pdfWin = [[VLPDFWindow alloc] initWithWindowNibName: @"VLPDFWindow"];
[self addWindowController: pdfWin];
[pdfWin release];
}
return pdfWin;
}
- (void)makeWindowControllers
2006-09-11 02:49:56 +00:00
{
sheetWin = [[VLSheetWindow alloc] initWithWindowNibName: @"VLDocument"];
[self addWindowController: sheetWin];
[sheetWin setShouldCloseDocument:YES];
[sheetWin release];
2006-09-11 02:49:56 +00:00
}
- (void)showWindows
2006-09-11 02:49:56 +00:00
{
[sheetWin showWindow: self];
if ([pdfWin isWindowLoaded])
[pdfWin showWindow: self];
if ([logWin isWindowLoaded])
[logWin showWindow: self];
2006-09-11 02:49:56 +00:00
}
- (VLSong *) song
{
return song;
}
2008-03-30 19:13:17 +00:00
- (void) setSongTitle:(NSString *)newTitle
{
if (newTitle != songTitle) {
[songTitle release];
songTitle = [newTitle retain];
}
[[self windowControllers] makeObjectsPerformSelector:
@selector(synchronizeWindowTitleWithDocumentName)];
}
2006-09-11 02:49:56 +00:00
- (NSNumber *) songKey
{
const VLProperties & prop = song->fProperties.front();
return [NSNumber numberWithInt: (prop.fKey << 8) | (prop.fMode & 0xFF)];
}
- (void) setKey:(int)key transpose:(BOOL)transpose inSections:(NSRange)sections
2006-09-11 02:49:56 +00:00
{
2006-12-04 07:04:24 +00:00
[self willChangeSong];
2007-05-06 05:07:39 +00:00
[self willChangeValueForKey:@"songKey"];
while (sections.length-- > 0)
song->ChangeKey(sections.location++, key>>8, key & 0xFF, transpose);
2007-05-06 05:07:39 +00:00
[self didChangeValueForKey:@"songKey"];
[self didChangeSong];
2006-09-11 02:49:56 +00:00
}
- (NSNumber *) songTime
{
const VLProperties & prop = song->fProperties.front();
return [NSNumber numberWithInt: (prop.fTime.fNum << 8) | prop.fTime.fDenom];
}
- (void) setTimeNum:(int)num denom:(int)denom inSections:(NSRange)sections
2006-09-11 02:49:56 +00:00
{
2006-12-04 07:04:24 +00:00
[self willChangeSong];
2007-05-06 05:07:39 +00:00
[self willChangeValueForKey:@"songTime"];
while (sections.length-- > 0)
song->ChangeTime(sections.location++, VLFraction(num, denom));
2007-05-06 05:07:39 +00:00
[self didChangeValueForKey:@"songTime"];
[self didChangeSong];
2006-09-11 02:49:56 +00:00
}
- (NSNumber *) songDivisions
{
const VLProperties & prop = song->fProperties.front();
return [NSNumber numberWithInt: prop.fDivisions];
}
- (void) setDivisions:(int)divisions inSections:(NSRange)sections
2006-09-11 02:49:56 +00:00
{
2006-12-04 07:04:24 +00:00
[self willChangeSong];
2007-05-06 05:07:39 +00:00
[self willChangeValueForKey:@"songDivisions"];
while (sections.length-- > 0)
song->ChangeDivisions(sections.location++, divisions);
2007-05-06 05:07:39 +00:00
[self didChangeValueForKey:@"songDivisions"];
[self didChangeSong];
2006-09-11 02:49:56 +00:00
}
2008-01-23 01:20:09 +00:00
- (void) setGroove:(NSString *)groove inSections:(NSRange)sections
{
const char * grv = [groove UTF8String];
[self willChangeSong];
[self willChangeValueForKey:@"songGroove"];
while (sections.length-- > 0)
song->fProperties[sections.location++].fGroove = grv;
[self didChangeValueForKey:@"songGroove"];
[self didChangeSong];
}
2008-04-12 21:33:43 +00:00
- (void) changeOctave:(BOOL)up inSections:(NSRange)sections
{
[self willChangeSong];
while (sections.length-- > 0)
song->ChangeOctave(sections.location++, up);
[self didChangeSong];
}
2008-01-26 16:38:30 +00:00
- (void) setChordSize:(float)size
{
[[[self undoManager] prepareWithInvocationTarget:self] setChordSize:chordSize];
chordSize = size;
2008-01-27 21:59:51 +00:00
[validTmpFiles removeObjectForKey:@"ly"];
[validTmpFiles removeObjectForKey:@"pdf"];
2008-01-26 16:38:30 +00:00
}
- (void) setLyricSize:(float)size
{
[[[self undoManager] prepareWithInvocationTarget:self] setLyricSize:lyricSize];
lyricSize = size;
2008-01-27 21:59:51 +00:00
[validTmpFiles removeObjectForKey:@"ly"];
[validTmpFiles removeObjectForKey:@"pdf"];
2008-01-26 16:38:30 +00:00
}
- (void) setStaffSize:(float)size
{
[[[self undoManager] prepareWithInvocationTarget:self] setStaffSize:staffSize];
staffSize = size;
2008-01-27 21:59:51 +00:00
[validTmpFiles removeObjectForKey:@"ly"];
[validTmpFiles removeObjectForKey:@"pdf"];
2008-01-26 16:38:30 +00:00
}
- (int) repeatVolta
{
return repeatVolta;
}
- (IBAction) togglePlayElements:(id)sender
{
playElements ^= [sender tag];
[validTmpFiles removeObjectForKey:@"mma"];
[validTmpFiles removeObjectForKey:@"mid"];
}
- (BOOL) validateMenuItem:(NSMenuItem *)menuItem
{
if ([menuItem action] == @selector(togglePlayElements:))
if (int tag = [menuItem tag])
[menuItem setState:(playElements & tag) != 0];
return YES;
}
2007-01-21 11:34:40 +00:00
- (bool) brandNew
{
return brandNew && ![self isDocumentEdited];
}
- (void) setRepeatVolta:(int)volta
{
repeatVolta = volta;
}
2006-11-10 08:09:18 +00:00
- (NSString *) tmpPath
{
if (!tmpPath) {
tmpPath = [[NSString alloc] initWithFormat:@"/var/tmp/VocalEasel.%08x",
self];
[[NSFileManager defaultManager] createDirectoryAtPath:tmpPath attributes:nil];
}
return tmpPath;
}
- (NSString *) workPath
{
if ([self fileURL]) // Prefer our wrapper directory
return [[self fileURL] path];
else
return [self tmpPath];
}
- (NSString *) baseName
{
return [[[self workPath] lastPathComponent] stringByDeletingPathExtension];
}
- (NSURL *) fileURLWithExtension:(NSString*)extension
{
return [NSURL fileURLWithPath:
[[[self workPath] stringByAppendingPathComponent:[self baseName]]
stringByAppendingPathExtension:extension]];
}
- (BOOL)saveToURL:(NSURL *)absoluteURL ofType:(NSString *)typeName forSaveOperation:(NSSaveOperationType)saveOperation error:(NSError **)outError
{
NSFileWrapper * preservedVCSWrapper = nil;
switch (saveOperation) {
case NSSaveToOperation:
case NSAutosaveOperation:
preservedVCSWrapper = vcsWrapper;
[preservedVCSWrapper retain];
// Fall through
case NSSaveAsOperation:
[vcsWrapper release];
vcsWrapper = nil;
// Fall through
case NSSaveOperation:
break;
}
BOOL res = [super saveToURL:absoluteURL ofType:typeName
forSaveOperation:saveOperation error:outError];
if (!vcsWrapper)
vcsWrapper = preservedVCSWrapper;
if ([typeName isEqual:@"VLNativeType"])
[validTmpFiles removeAllObjects];
return res;
}
- (NSFileWrapper *)fileWrapperOfType:(NSString *)typeName error:(NSError **)outError
2006-10-22 07:16:29 +00:00
{
2006-10-23 07:42:53 +00:00
if ([typeName isEqual:@"VLNativeType"]) {
return [self XMLFileWrapperWithError:outError flat:NO];
} else if ([typeName isEqual:@"VLMusicXMLType"]) {
return [self XMLFileWrapperWithError:outError flat:YES];
2006-10-23 07:42:53 +00:00
} else if ([typeName isEqual:@"VLLilypondType"]) {
return [self lilypondFileWrapperWithError:outError];
2006-11-04 08:15:34 +00:00
} else if ([typeName isEqual:@"VLMMAType"]) {
return [self mmaFileWrapperWithError:outError];
2006-11-12 11:37:36 +00:00
} else if ([typeName isEqual:@"VLMIDIType"]) {
return [self midiFileWrapperWithError:outError];
} else if ([typeName isEqual:@"VLPDFType"]) {
return [self pdfFileWrapperWithError:outError];
2006-10-23 07:42:53 +00:00
} else {
if (outError)
*outError = [NSError errorWithDomain:NSCocoaErrorDomain
code:NSPersistentStoreInvalidTypeError
userInfo:nil];
2006-10-22 07:16:29 +00:00
return nil;
2006-10-23 07:42:53 +00:00
}
2006-10-22 07:16:29 +00:00
}
- (BOOL)readFromFileWrapper:(NSFileWrapper *)wrapper ofType:(NSString *)typeName error:(NSError **)outError
2006-10-22 07:16:29 +00:00
{
2007-01-21 11:34:40 +00:00
brandNew = false;
//
// On opening a document, close all unchanged empty documents
//
NSEnumerator * docs = [[[NSDocumentController sharedDocumentController]
documents] objectEnumerator];
while (VLDocument * doc = [docs nextObject])
if ([doc brandNew])
[[doc windowControllers]
makeObjectsPerformSelector:@selector(close)];
2007-09-03 13:59:40 +00:00
if ([typeName isEqual:@"VLNativeType"] || [typeName isEqual:@"VLMusicXMLType"]) {
return [self readFromXMLFileWrapper:wrapper error:outError];
} else if ([typeName isEqual:@"VLLilypondType"] || [typeName isEqual:@"VLBIABType"]) {
if ([self readFromFileWrapper:wrapper withFilter:typeName error:outError]) {
[self setFileURL:nil];
return YES;
} else
return NO;
2006-10-23 07:42:53 +00:00
} else {
if (outError)
*outError = [NSError errorWithDomain:NSCocoaErrorDomain
code:NSPersistentStoreInvalidTypeError
userInfo:nil];
2006-10-22 07:16:29 +00:00
return NO;
2006-10-23 07:42:53 +00:00
}
2006-10-22 07:16:29 +00:00
}
- (void) changedFileWrapper
{
if (NSURL * url = [self fileURL])
if (NSDate * modDate =
[[[NSFileManager defaultManager] fileAttributesAtPath:[url path]
traverseLink:YES]
objectForKey:NSFileModificationDate])
[self setFileModificationDate:modDate];
}
- (void) createTmpFileWithExtension:(NSString*)ext ofType:(NSString*)type
{
if (![validTmpFiles objectForKey:ext]) {
NSError * err;
if ([self writeToURL:[self fileURLWithExtension:ext]
ofType:type error:&err]
) {
[validTmpFiles setObject:type forKey:ext];
[self changedFileWrapper];
}
}
}
2006-11-12 11:37:36 +00:00
- (NSTask *) taskWithLaunchPath:(NSString *)launch arguments:(NSArray *)args;
{
NSTask * task = [[NSTask alloc] init];
NSString * path = [self workPath];
NSPipe * pipe = [NSPipe pipe];
[task setCurrentDirectoryPath: path];
[task setStandardOutput: pipe];
[task setStandardError: pipe];
[task setArguments: args];
[task setLaunchPath: launch];
2007-04-18 05:28:06 +00:00
[[self logWin] window]; // Load but don't show
2006-11-12 11:37:36 +00:00
[NSThread detachNewThreadSelector:@selector(logFromFileHandle:) toTarget:logWin
withObject:[pipe fileHandleForReading]];
return task;
}
2006-11-13 04:26:09 +00:00
- (IBAction) play:(id)sender
{
[self createTmpFileWithExtension:@"mid" ofType:@"VLMIDIType"];
2006-11-13 04:26:09 +00:00
VLSoundOut::Instance()->PlayFile(
CFDataRef([NSData dataWithContentsOfURL:
[self fileURLWithExtension:@"mid"]]));
}
2008-01-24 01:29:18 +00:00
- (void) playWithGroove:(NSString *)groove inSections:(NSRange)sections
2007-04-23 05:46:37 +00:00
{
NSString * savedGroove = songGroove;
[validTmpFiles removeObjectForKey:@"mma"];
[validTmpFiles removeObjectForKey:@"mid"];
2008-01-24 01:29:18 +00:00
songGroove = groove;
previewRange = sections;
playElements |= kVLPlayGroovePreview;
2007-04-23 05:46:37 +00:00
[self play:groove];
2008-01-24 01:29:18 +00:00
playElements &= ~kVLPlayGroovePreview;
songGroove = savedGroove;
2007-04-23 05:46:37 +00:00
[validTmpFiles removeObjectForKey:@"mma"];
[validTmpFiles removeObjectForKey:@"mid"];
}
2007-01-02 07:01:55 +00:00
- (IBAction) stop:(id)sender
{
VLSoundOut::Instance()->Stop();
}
- (IBAction) playStop:(id)sender
{
if (VLSoundOut::Instance()->Playing()) {
[self stop:sender];
[sender setTitle:@"Play"];
} else {
[self play:sender];
[sender setTitle:@"Stop"];
}
}
- (IBAction) showOutput:(id)sender
{
[self createTmpFileWithExtension:@"pdf" ofType:@"VLPDFType"];
[[self pdfWin] showWindow:sender];
[pdfWin reloadPDF];
}
2007-04-18 08:25:43 +00:00
- (NSPrintOperation *)printOperationWithSettings:(NSDictionary *)printSettings
error:(NSError **)outError
{
[self createTmpFileWithExtension:@"pdf" ofType:@"VLPDFType"];
[printDoc autorelease];
printDoc = [[PDFDocument alloc] initWithURL:[self fileURLWithExtension:@"pdf"]];
NSPrintOperation *printOperation = [printDoc getPrintOperationForPrintInfo:[self printInfo] autoRotate:NO];
// Specify that the print operation can run in a separate thread. This will cause the print progress panel to appear as a sheet on the document window.
[printOperation setCanSpawnSeparateThread:YES];
// Set any print settings that might have been specified in a Print Document Apple event.
[[[printOperation printInfo] dictionary] addEntriesFromDictionary:printSettings];
return printOperation;
}
- (IBAction) showLog:(id)sender
{
[[self logWin] showWindow:sender];
}
2006-12-04 07:04:24 +00:00
- (void) willChangeSong
{
[self willChangeValueForKey:@"song"];
[[self undoManager] registerUndoWithTarget:self
selector:@selector(restoreSong:)
object:[VLSongWrapper wrapperWithSong:song]];
}
- (void) didChangeSong
{
[self didChangeValueForKey:@"song"];
[self updateChangeCount:NSChangeDone];
}
- (void) restoreSong:(VLSongWrapper *)savedSong
{
[self willChangeSong];
[self willChangeValueForKey:@"songKey"];
[self willChangeValueForKey:@"songTime"];
[self willChangeValueForKey:@"songDivisions"];
[self willChangeValueForKey:@"songGroove"];
2006-12-04 07:04:24 +00:00
song->swap(*[savedSong song]);
[self didChangeValueForKey:@"songKey"];
[self didChangeValueForKey:@"songTime"];
[self didChangeValueForKey:@"songDivisions"];
[self didChangeValueForKey:@"songGroove"];
2006-12-04 07:04:24 +00:00
[self didChangeSong];
}
2008-03-24 22:47:29 +00:00
- (NSString *) displayName
{
if ([songTitle isEqual:@""])
return [super displayName];
else
return songTitle;
}
2006-09-11 02:49:56 +00:00
@end