mirror of
https://github.com/microtherion/VocalEasel.git
synced 2025-01-13 13:43:59 +00:00
211 lines
6.7 KiB
Plaintext
211 lines
6.7 KiB
Plaintext
//
|
|
// File: VLLilypondDocument.mm - Export document in LilyPond format
|
|
//
|
|
// Author(s):
|
|
//
|
|
// (MN) Matthias Neeracher
|
|
//
|
|
// Copyright © 2006-2011 Matthias Neeracher
|
|
//
|
|
|
|
#import "VLLilypondDocument.h"
|
|
#import "VLLilypondWriter.h"
|
|
|
|
#import <algorithm>
|
|
|
|
@interface NSMutableString (VLLilypond)
|
|
|
|
- (void) substituteMacro:(NSString *)macro withValue:(NSString *)value;
|
|
- (void) purgeMacros;
|
|
|
|
@end
|
|
|
|
@implementation NSMutableString (VLLilypond)
|
|
|
|
- (void) substituteMacro:(NSString *)m withValue:(NSString *)value repeat:(BOOL)repeat
|
|
{
|
|
if ([value isEqual:@""])
|
|
return;
|
|
NSString * macro = [NSString stringWithFormat:@"<{%@}>", m];
|
|
NSRange range =
|
|
[value rangeOfCharacterFromSet:
|
|
[NSCharacterSet characterSetWithCharactersInString:@"\n"]];
|
|
BOOL hasEOL= range.location != NSNotFound;
|
|
unsigned from = 0;
|
|
|
|
for (range = [self rangeOfString:macro];
|
|
range.location != NSNotFound;
|
|
range = [self rangeOfString:macro options:0
|
|
range:NSMakeRange(from, [self length]-from)]
|
|
) {
|
|
if (hasEOL) {
|
|
//
|
|
// Multi line substitution, figure out a prefix
|
|
//
|
|
NSRange prefix, suffix;
|
|
NSRange line = [self lineRangeForRange:range];
|
|
suffix.location = range.location+range.length;
|
|
suffix.length = line.location+line.length-suffix.location;
|
|
prefix.location = line.location;
|
|
prefix.length = range.location-prefix.location;
|
|
NSString * pfxStr = [self substringWithRange:prefix];
|
|
NSString * nonBlank =
|
|
[pfxStr stringByTrimmingCharactersInSet:
|
|
[NSCharacterSet whitespaceCharacterSet]];
|
|
NSString * sfxStr =
|
|
[[self substringWithRange:suffix]
|
|
stringByTrimmingCharactersInSet:
|
|
[NSCharacterSet whitespaceCharacterSet]];
|
|
NSString * nl;
|
|
if ([nonBlank length]) {
|
|
NSRange nb = [pfxStr rangeOfString:nonBlank];
|
|
prefix.length = nb.location;
|
|
pfxStr =
|
|
[[self substringWithRange:prefix]
|
|
stringByAppendingString:@" "];
|
|
sfxStr = [NSString stringWithFormat:@"\n%@", pfxStr];
|
|
nl = @"\n";
|
|
} else {
|
|
range = line;
|
|
nl = @"";
|
|
}
|
|
NSArray * lines = [value componentsSeparatedByString:@"\n"];
|
|
value =
|
|
[NSString stringWithFormat:@"%@%@%@%@", nl, pfxStr,
|
|
[lines componentsJoinedByString:
|
|
[@"\n" stringByAppendingString:pfxStr]],
|
|
sfxStr];
|
|
}
|
|
from = range.location + [value length];
|
|
if (repeat) {
|
|
NSRange line = [self lineRangeForRange:range];
|
|
[self insertString:[self substringWithRange:line]
|
|
atIndex:line.location+line.length];
|
|
from = line.location+2*line.length+[value length]-range.length;
|
|
}
|
|
[self replaceCharactersInRange:range withString:value];
|
|
}
|
|
}
|
|
|
|
- (void)substituteMacro:(NSString*)macro withValue:(NSString*)value
|
|
{
|
|
[self substituteMacro:macro withValue:value repeat:NO];
|
|
}
|
|
|
|
- (void)substituteMacro:(NSString*)macro withValue:(NSString*)value escapingQuotes:(BOOL)escape
|
|
{
|
|
if (escape)
|
|
value = [value stringByReplacingOccurrencesOfString:@"\"" withString:@"\\\""];
|
|
[self substituteMacro:macro withValue:value repeat:NO];
|
|
}
|
|
|
|
- (void) purgeMacros
|
|
{
|
|
for (NSRange range = [self rangeOfString:@"<{"];
|
|
range.location != NSNotFound;
|
|
range = [self rangeOfString:@"<{"]
|
|
)
|
|
[self replaceCharactersInRange:[self lineRangeForRange:range]
|
|
withString: @""];
|
|
}
|
|
|
|
@end
|
|
|
|
static NSSize sPaperSizes[] = {
|
|
{842.0f, 1191.0f}, {595.0f, 842.0f}, {421.0f, 595.0f}, {298.0f, 421.0f},
|
|
{612.0f, 1008.0f}, {612.0f, 792.0f}, {792.0f, 1224.0f}
|
|
};
|
|
static const char * sPaperNames[] = {
|
|
"a3", "a4", "a5", "a6", "letter", "legal", "11x17", 0
|
|
};
|
|
|
|
@implementation VLDocument (Lilypond)
|
|
|
|
- (NSData *)lilypondDataWithError:(NSError **)outError
|
|
{
|
|
VLLilypondWriter writer;
|
|
writer.Visit(*song);
|
|
NSBundle * bndl = [NSBundle mainBundle];
|
|
NSString * tmpl =
|
|
[bndl pathForResource:lilypondTemplate
|
|
ofType:@"lyt" inDirectory:@"Templates"];
|
|
NSStringEncoding enc = NSUTF8StringEncoding;
|
|
NSMutableString * ly =
|
|
[NSMutableString stringWithContentsOfFile:tmpl encoding:enc error:outError];
|
|
NSPrintInfo * pi = [self printInfo];
|
|
NSSize sz = [pi paperSize];
|
|
int bestPaper = -1;
|
|
float bestDist = 1e10f;
|
|
|
|
if ([pi orientation] == NSLandscapeOrientation)
|
|
std::swap(sz.width, sz.height);
|
|
|
|
for (int paper = 0; sPaperNames[paper]; ++paper) {
|
|
float dist = hypotf(sz.width - sPaperSizes[paper].width,
|
|
sz.height- sPaperSizes[paper].height);
|
|
if (dist < bestDist) {
|
|
bestPaper = paper;
|
|
bestDist = dist;
|
|
}
|
|
}
|
|
|
|
NSString * paper = [NSString stringWithFormat:
|
|
[pi orientation] == NSLandscapeOrientation ? @"\"%s\" 'landscape" : @"\"%s\"",
|
|
sPaperNames[bestPaper]];
|
|
float scaling= [[[pi dictionary] objectForKey:NSPrintScalingFactor]
|
|
floatValue];
|
|
|
|
[ly substituteMacro:@"TITLE" withValue:songTitle escapingQuotes:YES];
|
|
[ly substituteMacro:@"POET" withValue:songLyricist escapingQuotes:YES];
|
|
[ly substituteMacro:@"COMPOSER" withValue:songComposer escapingQuotes:YES];
|
|
[ly substituteMacro:@"ARRANGER" withValue:songArranger escapingQuotes:YES];
|
|
[ly substituteMacro:@"VLVERSION" withValue:
|
|
[bndl objectForInfoDictionaryKey:@"CFBundleVersion"]];
|
|
[ly substituteMacro:@"PAPERSIZE" withValue:paper];
|
|
// [ly substituteMacro:@"FORMATTING" withValue:@"ragged-last-bottom = ##f"];
|
|
[ly substituteMacro:@"VLVERSION" withValue:
|
|
[bndl objectForInfoDictionaryKey:@"CFBundleVersion"]];
|
|
[ly substituteMacro:@"CHORDSIZE" withValue:
|
|
[NSString stringWithFormat:@"%f", chordSize]];
|
|
[ly substituteMacro:@"LYRICSIZE" withValue:
|
|
[NSString stringWithFormat:@"%f", lyricSize]];
|
|
[ly substituteMacro:@"STAFFSIZE" withValue:
|
|
[NSString stringWithFormat:@"%f", staffSize*scaling]];
|
|
[ly substituteMacro:@"TOPPADDING" withValue:
|
|
[NSString stringWithFormat:@"%f", topPadding]];
|
|
[ly substituteMacro:@"TITLEPADDING" withValue:
|
|
[NSString stringWithFormat:@"%f", titlePadding]];
|
|
[ly substituteMacro:@"STAFFPADDING" withValue:
|
|
[NSString stringWithFormat:@"%f", staffPadding]];
|
|
[ly substituteMacro:@"CHORDPADDING" withValue:
|
|
[NSString stringWithFormat:@"%f", chordPadding]];
|
|
[ly substituteMacro:@"LYRICPADDING" withValue:
|
|
[NSString stringWithFormat:@"%f", lyricPadding]];
|
|
[ly substituteMacro:@"CHORDS" withValue:
|
|
[NSString stringWithUTF8String:writer.Chords().c_str()]];
|
|
[ly substituteMacro:@"NOTES" withValue:
|
|
[NSString stringWithUTF8String:writer.Melody().c_str()]];
|
|
if (size_t stanzas = song->CountStanzas())
|
|
for (size_t s=0; s<stanzas; ++s) {
|
|
[ly substituteMacro:@"LYRICS" withValue:
|
|
[NSString stringWithUTF8String:writer.Lyrics(s).c_str()]
|
|
repeat: s<stanzas];
|
|
}
|
|
[ly purgeMacros];
|
|
return [ly dataUsingEncoding:enc];
|
|
}
|
|
|
|
- (NSFileWrapper *)lilypondFileWrapperWithError:(NSError **)outError
|
|
{
|
|
NSData * data = [self lilypondDataWithError:outError];
|
|
|
|
if (!data)
|
|
return nil;
|
|
else
|
|
return [[[NSFileWrapper alloc]
|
|
initRegularFileWithContents:data]
|
|
autorelease];
|
|
}
|
|
|
|
@end
|