mirror of
https://github.com/microtherion/VocalEasel.git
synced 2024-12-22 11:14:00 +00:00
289 lines
7.1 KiB
Plaintext
289 lines
7.1 KiB
Plaintext
//
|
|
// File: VLSheetViewSelection.mm - Measure selection functionality
|
|
//
|
|
// Author(s):
|
|
//
|
|
// (MN) Matthias Neeracher
|
|
//
|
|
// Copyright © 2006-2007 Matthias Neeracher
|
|
//
|
|
|
|
#import "VLSheetView.h"
|
|
#import "VLSheetViewSelection.h"
|
|
#import "VLDocument.h"
|
|
|
|
//
|
|
// We're too lazy to properly serialize our private pasteboard format.
|
|
//
|
|
static VLSong sPasteboard;
|
|
|
|
@implementation VLSheetView (Selection)
|
|
|
|
- (void)editSelection
|
|
{
|
|
fSelStart = fSelEnd = fCursorMeasure;
|
|
[self setNeedsDisplay:YES];
|
|
}
|
|
|
|
- (void)adjustSelection:(NSEvent *)event
|
|
{
|
|
int prevMeasure = fCursorMeasure;
|
|
switch ([self findRegionForEvent:event]) {
|
|
case kRegionNote:
|
|
case kRegionChord:
|
|
case kRegionLyrics:
|
|
if (fCursorAt.fNum)
|
|
++fCursorMeasure;
|
|
//
|
|
// Fall through
|
|
//
|
|
case kRegionMeasure:
|
|
fCursorMeasure =
|
|
std::max(0, std::min<int>(fCursorMeasure, [self song]->CountMeasures()));
|
|
if (fCursorMeasure > fSelEnd) {
|
|
fSelEnd = fCursorMeasure;
|
|
[self setNeedsDisplay:YES];
|
|
} else if (fCursorMeasure < fSelStart) {
|
|
fSelStart = fCursorMeasure;
|
|
[self setNeedsDisplay:YES];
|
|
} else if (prevMeasure == fSelEnd && fCursorMeasure<prevMeasure) {
|
|
fSelEnd = fCursorMeasure;
|
|
[self setNeedsDisplay:YES];
|
|
} else if (prevMeasure == fSelStart && fCursorMeasure>prevMeasure) {
|
|
fSelStart = fCursorMeasure;
|
|
[self setNeedsDisplay:YES];
|
|
}
|
|
break;
|
|
default:
|
|
fCursorMeasure = prevMeasure;
|
|
break;
|
|
}
|
|
fCursorRegion = kRegionMeasure;
|
|
}
|
|
|
|
- (BOOL)validateMenuItem:(id) item
|
|
{
|
|
SEL action = [item action];
|
|
if (action == @selector(insertJumpToCoda:))
|
|
if (fSelStart == fSelEnd) {
|
|
[item setState:[self song]->fGoToCoda==fSelStart];
|
|
|
|
return YES;
|
|
} else
|
|
return NO;
|
|
else if (action == @selector(insertStartCoda:))
|
|
if (fSelStart == fSelEnd) {
|
|
[item setState:[self song]->fCoda==fSelStart];
|
|
|
|
return YES;
|
|
} else
|
|
return NO;
|
|
else if (action == @selector(insertBreak:))
|
|
if (fSelStart == fSelEnd && fSelStart > 0) {
|
|
VLSong * song = [self song];
|
|
[item setState:fSelStart < song->fMeasures.size()
|
|
&& song->fMeasures[fSelStart].fBreak == [item tag]];
|
|
|
|
return YES;
|
|
} else
|
|
return NO;
|
|
else
|
|
return [self validateUserInterfaceItem:item];
|
|
}
|
|
|
|
- (BOOL)validateUserInterfaceItem:(id) item
|
|
{
|
|
SEL action = [item action];
|
|
if (action == @selector(cut:)
|
|
|| action == @selector(copy:)
|
|
|| action == @selector(delete:)
|
|
)
|
|
return fSelStart < fSelEnd;
|
|
else if (action == @selector(editRepeat:))
|
|
return fSelEnd > fSelStart
|
|
&& [self song]->CanBeRepeat(fSelStart, fSelEnd);
|
|
else if (action == @selector(editRepeatEnding:))
|
|
return fSelEnd > fSelStart
|
|
&& [self song]->CanBeEnding(fSelStart, fSelEnd);
|
|
else if (action == @selector(paste:))
|
|
return fSelStart <= fSelEnd;
|
|
else
|
|
return YES;
|
|
}
|
|
|
|
- (IBAction)cut:(id)sender
|
|
{
|
|
[self copy:sender];
|
|
[self delete:sender];
|
|
}
|
|
|
|
- (IBAction)copy:(id)sender
|
|
{
|
|
NSPasteboard * pb = [NSPasteboard generalPasteboard];
|
|
NSString * pbType = [[NSBundle mainBundle] bundleIdentifier];
|
|
|
|
[pb declareTypes:[NSArray arrayWithObject:pbType] owner:nil];
|
|
if ([pb setString:@"whatever" forType:pbType])
|
|
sPasteboard = [self song]->CopyMeasures(fSelStart, fSelEnd);
|
|
}
|
|
|
|
- (IBAction)paste:(id)sender
|
|
{
|
|
NSPasteboard * pb = [NSPasteboard generalPasteboard];
|
|
NSString * pbType = [[NSBundle mainBundle] bundleIdentifier];
|
|
|
|
if ([pb availableTypeFromArray:[NSArray arrayWithObject:pbType]]) {
|
|
[[self document] willChangeSong];
|
|
if (![sender tag]) // Delete on paste, but not on overwrite
|
|
[self song]->DeleteMeasures(fSelStart, fSelEnd);
|
|
[self song]->PasteMeasures(fSelStart, sPasteboard, [sender tag]);
|
|
[[self document] didChangeSong];
|
|
[self setNeedsDisplay:YES];
|
|
}
|
|
}
|
|
|
|
- (IBAction)delete:(id)sender
|
|
{
|
|
[[self document] willChangeSong];
|
|
[self song]->DeleteMeasures(fSelStart, fSelEnd, [sender tag]);
|
|
[[self document] didChangeSong];
|
|
[self setNeedsDisplay:YES];
|
|
}
|
|
|
|
- (IBAction)editRepeat:(id)sender
|
|
{
|
|
int volta;
|
|
[self song]->CanBeRepeat(fSelStart, fSelEnd, &volta);
|
|
|
|
[fRepeatMsg setStringValue:
|
|
[NSString stringWithFormat:@"Repeat measures %d through %d",
|
|
fSelStart+1, fSelEnd]];
|
|
[NSApp beginSheet:fRepeatSheet modalForWindow:[self window]
|
|
modalDelegate:self
|
|
didEndSelector:@selector(didEndRepeatSheet:returnCode:contextInfo:)
|
|
contextInfo:nil];
|
|
}
|
|
|
|
- (void)didEndRepeatSheet:(NSWindow *)sheet returnCode:(int)returnCode
|
|
contextInfo:(void *)ctx
|
|
{
|
|
switch (returnCode) {
|
|
case NSAlertFirstButtonReturn:
|
|
[[self document] willChangeSong];
|
|
[self song]->AddRepeat(fSelStart, fSelEnd, [[self document] repeatVolta]);
|
|
[self setNeedsDisplay:YES];
|
|
[[self document] didChangeSong];
|
|
break;
|
|
case NSAlertThirdButtonReturn:
|
|
[[self document] willChangeSong];
|
|
[self song]->DelRepeat(fSelStart, fSelEnd);
|
|
[[self document] didChangeSong];
|
|
[self setNeedsDisplay:YES];
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
[sheet orderOut:self];
|
|
}
|
|
|
|
- (IBAction)editRepeatEnding:(id)sender
|
|
{
|
|
[self song]->CanBeEnding(fSelStart, fSelEnd, &fVolta, &fVoltaOK);
|
|
|
|
[fEndingMsg setStringValue:
|
|
[NSString stringWithFormat:@"Ending in measures %d through %d applies to repeats:",
|
|
fSelStart+1, fSelEnd]];
|
|
|
|
[NSApp beginSheet:fEndingSheet modalForWindow:[self window]
|
|
modalDelegate:self
|
|
didEndSelector:@selector(didEndEndingSheet:returnCode:contextInfo:)
|
|
contextInfo:nil];
|
|
}
|
|
|
|
- (void)didEndEndingSheet:(NSWindow *)sheet returnCode:(int)returnCode
|
|
contextInfo:(void *)ctx
|
|
{
|
|
switch (returnCode) {
|
|
case NSAlertFirstButtonReturn:
|
|
[[self document] willChangeSong];
|
|
[self song]->AddEnding(fSelStart, fSelEnd, fVolta);
|
|
[self setNeedsDisplay:YES];
|
|
[[self document] didChangeSong];
|
|
break;
|
|
case NSAlertThirdButtonReturn:
|
|
[[self document] willChangeSong];
|
|
[self song]->DelEnding(fSelStart, fSelEnd);
|
|
[[self document] didChangeSong];
|
|
[self setNeedsDisplay:YES];
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
[sheet orderOut:self];
|
|
}
|
|
|
|
//
|
|
// Data source for endings
|
|
//
|
|
- (int)numberOfRowsInTableView:(NSTableView *)aTableView
|
|
{
|
|
return 1;
|
|
}
|
|
|
|
- (id)tableView:(NSTableView*)tv objectValueForTableColumn:(NSTableColumn *)col
|
|
row:(int)rowIndex
|
|
{
|
|
int mask = [[col identifier] intValue];
|
|
return (fVoltaOK & mask) ? [NSNumber numberWithBool:(fVolta & mask)] : nil;
|
|
}
|
|
|
|
- (void)tableView:(NSTableView *)tv setObjectValue:(id)val forTableColumn:(NSTableColumn *)col row:(int)rowIndex
|
|
{
|
|
int mask = [[col identifier] intValue];
|
|
|
|
if ([val boolValue])
|
|
fVolta |= mask;
|
|
else
|
|
fVolta &= ~mask;
|
|
}
|
|
|
|
- (IBAction)insertJumpToCoda:(id)sender
|
|
{
|
|
[[self document] willChangeSong];
|
|
VLSong * song = [self song];
|
|
if (song->fGoToCoda == fSelStart)
|
|
song->fGoToCoda = -1;
|
|
else
|
|
song->fGoToCoda = fSelStart;
|
|
[self setNeedsDisplay:YES];
|
|
[[self document] didChangeSong];
|
|
}
|
|
|
|
- (IBAction)insertStartCoda:(id)sender
|
|
{
|
|
[[self document] willChangeSong];
|
|
VLSong * song = [self song];
|
|
if (song->fCoda == fSelStart)
|
|
song->fCoda = -1;
|
|
else
|
|
song->fCoda = fSelStart;
|
|
[self setNeedsDisplay:YES];
|
|
[[self document] didChangeSong];
|
|
}
|
|
|
|
- (IBAction)insertBreak:(id)sender
|
|
{
|
|
[[self document] willChangeSong];
|
|
VLSong * song = [self song];
|
|
VLMeasure & meas = song->fMeasures[fSelStart];
|
|
if (meas.fBreak == [sender tag])
|
|
meas.fBreak = 0;
|
|
else
|
|
meas.fBreak = [sender tag];
|
|
fNeedsRecalc = kRecalc;
|
|
[self setNeedsDisplay:YES];
|
|
[[self document] didChangeSong];
|
|
}
|
|
|
|
@end
|