cog/Application/PlaybackController.m

382 lines
8.1 KiB
Matlab
Raw Normal View History

2006-01-20 15:41:31 +00:00
#import "PlaybackController.h"
#import "PlaylistView.h"
#import "DBLog.h"
#import "CogAudio/Status.h"
2006-01-20 15:41:31 +00:00
#import "PlaylistController.h"
#import "PlaylistEntry.h"
2006-01-20 15:41:31 +00:00
@implementation PlaybackController
- (id)init
{
self = [super init];
if (self)
{
2007-02-26 05:26:48 +00:00
[self initDefaults];
audioPlayer = [[AudioPlayer alloc] init];
[audioPlayer setDelegate:self];
2006-01-20 15:41:31 +00:00
playbackStatus = kCogStatusStopped;
showTimeRemaining = NO;
scrobbler = [[AudioScrobbler alloc] init];
[GrowlApplicationBridge setGrowlDelegate:self];
2006-01-20 15:41:31 +00:00
}
return self;
}
2007-02-26 05:26:48 +00:00
- (void)initDefaults
{
NSDictionary *defaultsDictionary = [NSDictionary dictionaryWithObjectsAndKeys:
[NSNumber numberWithBool:YES], @"enableAudioScrobbler",
[NSNumber numberWithBool:NO], @"automaticallyLaunchLastFM",
nil];
[[NSUserDefaults standardUserDefaults] registerDefaults:defaultsDictionary];
}
- (NSDictionary *) registrationDictionaryForGrowl
{
NSArray *notifications = [NSArray arrayWithObjects:@"Stream Changed", nil];
return [NSDictionary dictionaryWithObjectsAndKeys:
@"Cog", GROWL_APP_NAME,
notifications, GROWL_NOTIFICATIONS_ALL,
notifications, GROWL_NOTIFICATIONS_DEFAULT,
nil];
}
2006-01-20 15:41:31 +00:00
- (void)awakeFromNib
{
2006-04-04 01:08:21 +00:00
currentVolume = 100.0;
2006-05-13 13:37:32 +00:00
[volumeSlider setDoubleValue:pow(10.0, log10(0.5)/4.0)*[volumeSlider maxValue]];
2006-01-20 15:41:31 +00:00
}
2006-01-20 15:41:31 +00:00
- (IBAction)playPauseResume:(id)sender
{
2006-04-04 01:08:21 +00:00
DBLog(@"PLAYING");
2006-01-20 15:41:31 +00:00
if (playbackStatus == kCogStatusStopped)
[self play:self];
else
[self pauseResume:self];
}
- (IBAction)pauseResume:(id)sender
{
// DBLog(@"Pause/Resume Sent!");
if (playbackStatus == kCogStatusPaused)
[self resume:self];
else
[self pause:self];
}
- (IBAction)pause:(id)sender
{
// DBLog(@"Pause Sent!");
[audioPlayer pause];
if([[NSUserDefaults standardUserDefaults] boolForKey:@"enableAudioScrobbler"]) {
[scrobbler pause];
}
2006-01-20 15:41:31 +00:00
}
- (IBAction)resume:(id)sender
{
// DBLog(@"Resume Sent!");
[audioPlayer resume];
if([[NSUserDefaults standardUserDefaults] boolForKey:@"enableAudioScrobbler"]) {
[scrobbler resume];
}
2006-01-20 15:41:31 +00:00
}
- (IBAction)stop:(id)sender
{
// DBLog(@"Stop Sent!");
[audioPlayer stop];
if([[NSUserDefaults standardUserDefaults] boolForKey:@"enableAudioScrobbler"]) {
[scrobbler stop];
}
2006-01-20 15:41:31 +00:00
}
//called by double-clicking on table
- (void)playEntryAtIndex:(int)i
{
PlaylistEntry *pe = [[playlistController arrangedObjects] objectAtIndex:i];
[self playEntry:pe];
}
- (IBAction)play:(id)sender
{
if ([playlistView selectedRow] == -1)
[playlistView selectRow:0 byExtendingSelection:NO];
[self playEntryAtIndex:[playlistView selectedRow]];
}
- (void)playEntry:(PlaylistEntry *)pe
2006-01-20 15:41:31 +00:00
{
// DBLog(@"PlayEntry: %@ Sent!", [pe filename]);
2006-01-29 14:57:48 +00:00
if (playbackStatus != kCogStatusStopped)
[self stop:self];
[playlistController setCurrentEntry:pe];
[positionSlider setDoubleValue:0.0f];
[self updateTimeField:0.0f];
[audioPlayer play:[pe url] withUserInfo:pe];
[audioPlayer setVolume:currentVolume];
if([[NSUserDefaults standardUserDefaults] boolForKey:@"enableAudioScrobbler"]) {
[scrobbler start:pe];
}
[GrowlApplicationBridge notifyWithTitle:[pe title]
description:[pe artist]
notificationName:@"Stream Changed"
iconData:nil
priority:0
isSticky:NO
clickContext:nil];
2006-01-20 15:41:31 +00:00
}
- (IBAction)next:(id)sender
{
2006-04-04 01:08:21 +00:00
DBLog(@"CALLING: %i %i", playbackStatus, kCogStatusStopped);
2006-01-29 14:57:48 +00:00
if ([playlistController next] == NO)
2006-01-20 15:41:31 +00:00
return;
2006-01-29 14:57:48 +00:00
if (playbackStatus != kCogStatusStopped)
{
2006-04-04 01:08:21 +00:00
DBLog(@"STOPPING");
2006-01-29 14:57:48 +00:00
[self stop:self];
[self playEntry:[playlistController currentEntry]];
}
2006-01-20 15:41:31 +00:00
}
- (IBAction)prev:(id)sender
{
2006-04-04 01:08:21 +00:00
DBLog(@"CALLING");
2006-01-20 15:41:31 +00:00
if ([playlistController prev] == nil)
return;
2006-01-29 14:57:48 +00:00
if (playbackStatus != kCogStatusStopped)
{
[self stop:self];
[self playEntry:[playlistController currentEntry]];
}
2006-01-20 15:41:31 +00:00
}
- (IBAction)seek:(id)sender
{
// DBLog(@"SEEKING?");
double time;
time = [positionSlider doubleValue];
2006-04-18 17:00:29 +00:00
2006-04-04 01:08:21 +00:00
if ([sender tracking] == NO) // check if user stopped sliding before playing audio
[audioPlayer seekToTime:time];
2006-01-20 15:41:31 +00:00
[self updateTimeField:time];
}
- (void)changePlayButtonImage:(NSString *)name
{
NSImage *img = [NSImage imageNamed:[name stringByAppendingString:@"_gray"]];
NSImage *alt = [NSImage imageNamed:[name stringByAppendingString:@"_blue"]];
[img retain];
[alt retain];
if (img == nil)
{
DBLog(@"NIL IMAGE!!!");
}
if (alt == nil)
{
DBLog(@"NIL ALT");
}
[playButton setImage:img];
[playButton setAlternateImage:alt];
}
- (IBAction)changeVolume:(id)sender
{
double percent;
2006-04-18 17:00:29 +00:00
2006-05-13 13:37:32 +00:00
//Approximated log
percent = (float)[sender doubleValue]/[sender maxValue];
percent = percent * percent * percent * percent;
//gravitates at the 100% mark
float v = [sender frame].size.width - ([sender frame].size.width*(percent*[sender maxValue])/100.0);
2006-04-18 17:00:29 +00:00
if (fabs(v) < 10.0)
{
2006-05-13 13:37:32 +00:00
percent = 0.5;
v = pow(10.0, log10(percent)/4.0);
[sender setDoubleValue:v*[sender maxValue]];
2006-04-18 17:00:29 +00:00
}
2006-05-13 13:37:32 +00:00
currentVolume = percent * [sender maxValue];
[audioPlayer setVolume:currentVolume];
2006-01-20 15:41:31 +00:00
}
- (IBAction)volumeDown:(id)sender
{
double percent;
[volumeSlider setDoubleValue:([volumeSlider doubleValue] - 5)];
percent = (float)[volumeSlider doubleValue]/[volumeSlider maxValue];
percent = percent * percent * percent * percent;
currentVolume = percent * [volumeSlider maxValue];
[audioPlayer setVolume:currentVolume];
}
- (IBAction)volumeUp:(id)sender
{
double percent;
[volumeSlider setDoubleValue:([volumeSlider doubleValue] + 5)];
percent = (float)[volumeSlider doubleValue]/[volumeSlider maxValue];
percent = percent * percent * percent * percent;
currentVolume = percent * [volumeSlider maxValue];
[audioPlayer setVolume:currentVolume];
}
2006-01-20 15:41:31 +00:00
- (void)updateTimeField:(double)pos
{
NSString *text;
if (showTimeRemaining == NO)
{
int sec = (int)(pos/1000.0);
text = [NSString stringWithFormat:NSLocalizedString(@"TimeElapsed", @""), sec/60, sec%60];
}
else
{
int sec = (int)(([positionSlider maxValue] - pos)/1000.0);
2006-05-13 16:50:52 +00:00
if (sec < 0)
sec = 0;
2006-01-20 15:41:31 +00:00
text = [NSString stringWithFormat:NSLocalizedString(@"TimeRemaining", @""), sec/60, sec%60];
}
[timeField setStringValue:text];
}
- (IBAction)toggleShowTimeRemaining:(id)sender
{
showTimeRemaining = !showTimeRemaining;
[self updateTimeField:[positionSlider doubleValue]];
}
- (void)audioPlayer:(AudioPlayer *)player requestNextStream:(id)userInfo
2006-01-20 15:41:31 +00:00
{
PlaylistEntry *curEntry = (PlaylistEntry *)userInfo;
2006-01-20 15:41:31 +00:00
PlaylistEntry *pe;
2006-05-12 18:12:31 +00:00
if ([playlistController shuffle] == YES)
{
pe = [playlistController entryAtIndex:[[curEntry shuffleIndex] intValue] + 1];
2006-05-12 18:12:31 +00:00
}
else
{
pe = [playlistController entryAtIndex:[[curEntry index] intValue] + 1];
2006-05-12 18:12:31 +00:00
}
2006-01-20 15:41:31 +00:00
if (pe == nil)
[player setNextStream:nil];
2006-01-20 15:41:31 +00:00
else
{
DBLog(@"NEXT SONG: %@", [pe url]);
[player setNextStream:[pe url] withUserInfo:pe];
}
2006-01-20 15:41:31 +00:00
}
- (void)audioPlayer:(AudioPlayer *)player streamChanged:(id)userInfo
2006-01-20 15:41:31 +00:00
{
PlaylistEntry *pe = (PlaylistEntry *)userInfo;
2006-04-13 02:51:22 +00:00
[playlistController setCurrentEntry:pe];
2006-01-20 15:41:31 +00:00
[positionSlider setDoubleValue:0.0f];
2006-01-20 15:41:31 +00:00
[self updateTimeField:0.0f];
2006-01-20 15:41:31 +00:00
if([[NSUserDefaults standardUserDefaults] boolForKey:@"enableAudioScrobbler"]) {
[scrobbler start:pe];
}
[GrowlApplicationBridge notifyWithTitle:[pe title]
description:[pe artist]
notificationName:@"Stream Changed"
iconData:nil
priority:0
isSticky:NO
clickContext:nil];
2006-01-20 15:41:31 +00:00
}
- (void)updatePosition:(id)sender
2006-01-20 15:41:31 +00:00
{
double pos = [audioPlayer amountPlayed];
2006-01-20 15:41:31 +00:00
if ([positionSlider tracking] == NO)
{
// DBLog(@"Received pos update: %f", pos);
[positionSlider setDoubleValue:pos];
[self updateTimeField:pos];
}
}
- (void)audioPlayer:(AudioPlayer *)player statusChanged:(id)s
2006-01-20 15:41:31 +00:00
{
2006-01-29 14:57:48 +00:00
int status = [s intValue];
2006-01-20 15:41:31 +00:00
if (status == kCogStatusStopped || status == kCogStatusPaused)
{
2006-04-04 01:08:21 +00:00
DBLog(@"INVALIDATING");
if (positionTimer)
{
[positionTimer invalidate];
positionTimer = NULL;
}
if (status == kCogStatusStopped)
{
[positionSlider setDoubleValue:0.0f];
[self updateTimeField:0.0f];
}
2006-01-20 15:41:31 +00:00
//Show play image
[self changePlayButtonImage:@"play"];
}
else if (status == kCogStatusPlaying)
{
if (!positionTimer)
positionTimer = [NSTimer scheduledTimerWithTimeInterval:1.00 target:self selector:@selector(updatePosition:) userInfo:nil repeats:YES];
2006-01-20 15:41:31 +00:00
//Show pause
[self changePlayButtonImage:@"pause"];
}
playbackStatus = status;
}
@end