0

我在 Mac 10.6 上使用 Cocoa。我试图让节拍器点击时间,但是当我运行代码时,只播放前两个声音,然后它是安静的,直到再次播放声音时结束。如果我用 NSBeep() 替换 [player play] 语句,它工作正常。有什么建议么?

#import <Cocoa/Cocoa.h>


@interface Metronome : NSObject {

NSInteger timeSig;
NSInteger noteValue;
NSInteger bpm;
NSUInteger beatNumber;
CGFloat duration;
NSSound *tickPlayer;
NSSound *tockPlayer;

}
@property(readwrite) NSInteger timeSig;
@property(readwrite) NSInteger noteValue;
@property(readwrite) NSInteger bpm;
@property(readwrite) float duration;
@property(readwrite) NSUInteger beatNumber;

-(id)init;
-(id)initWithTimeSig:(NSInteger)ts  andNoteValue:(NSInteger)nv andBpm:(NSInteger)bp;
-(void)BeginMetronome;
 -(void)PlaySound;

@end

#import "Metronome.h"

 #define defaultBpm 80
 #define defaultTimeSig 4
 #define defaultNoteValue 4

 @implementation Metronome

 @synthesize timeSig, noteValue, duration, bpm, beatNumber;
 -(id)init{
return [self initWithTimeSig:defaultTimeSig andNoteValue:defaultNoteValue       andBpm:defaultBpm];
 }
 -(id)initWithTimeSig:(NSInteger)ts andNoteValue:(NSInteger)nv andBpm:(NSInteger)bp {
[super init];
if(self){
    self.timeSig = ts;
    self.noteValue = nv;
    self.bpm = bp;
    self.duration = 60.0/bpm;
    tickPlayer = [NSSound soundNamed:@"Hat1"];
    tockPlayer = [NSSound soundNamed:@"Hat2"];
    self.beatNumber = 1;


}
return self;
 }
 -(void)BeginMetronome{
BOOL continuePlaying = YES;
NSInteger iter=0;

while (continuePlaying){
    if(iter > 12){
        continuePlaying = FALSE;
    }
    iter++;
    [self PlaySound];
    NSDate *curtainTime = [[NSDate alloc] initWithTimeIntervalSinceNow:self.duration];
    NSDate *currentTime = [[NSDate alloc] init];

    while (continuePlaying && ([currentTime compare:curtainTime] != NSOrderedDescending)) { 
        // if ([soundPlayerThread isCancelled] == YES) {
        //   continuePlaying = NO;
        //}
        //[NSThread sleepForTimeInterval:0.01];
        sleep(.01);
        //[tickPlayer stop];
        //[tockPlayer stop];
        [currentTime release];
        currentTime = [[NSDate alloc] init];
    }
    [curtainTime release];      
    [currentTime release];      

}
}
  - (void)PlaySound {          //***********Problem in this loop
if ([tickPlayer isPlaying])
    [tickPlayer stop];
if ([tockPlayer isPlaying])
    [tockPlayer stop];

if (beatNumber == 1) {
    [tockPlayer play];
}
else if (beatNumber == 2){
    [tickPlayer play];
}
else if (beatNumber == self.timeSig) {
    beatNumber = 0;
    [tickPlayer play];
}
else{
    //[tickPlayer stop];
    [tickPlayer play];
    NSBeep();
}


beatNumber++;
//NSLog(@" %d ", beatNumber);
 // NSBeep();
 }



 @end
4

1 回答 1

2

我相信您对 sleep() 的使用会阻止 NSSound 的播放。这是一种非常草率的方法,而是使用这样的计时器:

static NSInteger iter;

- (void) beginMetronome {

    iter = 0;

    NSTimer *timer = [NSTimer timerWithTimeInterval:self.duration target:self selector:@selector(continueMetronome:) userInfo:nil repeats:YES];
    [[NSRunLoop currentRunLoop] addTimer:timer forMode:NSRunLoopCommonModes];
    [timer fire];    
}

- (void) continueMetronome:(NSTimer *)theTimer {

    [self PlaySound];

    if (++iter > 12) {

        [theTimer invalidate];
    }
}

此外,您必须保留您计划在 init 方法中使用的声音。

tickPlayer = [[NSSound soundNamed:@"Hat1"] retain];
tockPlayer = [[NSSound soundNamed:@"Hat2"] retain];

Cocoa 方法应该像我写的那样以小写字母开头。我不太确定节拍器应该如何工作,因此您可能必须将 12 的值更改为其他值。

于 2011-06-09T13:14:54.847 回答