Hopefully fixed one last bug with the multisource playlists. Added some
[MenuTunes.git] / iTunesRemote.m
1 #import "iTunesRemote.h"
2
3 @implementation iTunesRemote
4
5 + (id)remote
6 {
7     return [[[iTunesRemote alloc] init] autorelease];
8 }
9
10 - (NSString *)remoteTitle
11 {
12     return @"iTunes Remote";
13 }
14
15 - (NSString *)remoteInformation
16 {
17     return @"Default MenuTunes plugin to control iTunes, by iThink Software.";
18 }
19
20 - (NSImage *)remoteIcon
21 {
22     return nil;
23 }
24
25 - (BOOL)begin
26 {
27     ITDebugLog(@"iTunesRemote begun");
28     savedPSN = [self iTunesPSN];
29     return YES;
30 }
31
32 - (BOOL)halt
33 {
34     ITDebugLog(@"iTunesRemote halted");
35     return YES;
36 }
37
38 - (NSString *)playerFullName
39 {
40     return @"iTunes";
41 }
42
43 - (NSString *)playerSimpleName
44 {
45     return @"iTunes";
46 }
47
48 - (NSDictionary *)capabilities
49 {
50     return [NSDictionary dictionaryWithObjectsAndKeys:
51                 [NSNumber numberWithBool: YES], @"Remote",
52                 [NSNumber numberWithBool: YES], @"Basic Track Control",
53                 [NSNumber numberWithBool: YES], @"Track Information",
54                 [NSNumber numberWithBool: YES], @"Track Navigation",
55                 [NSNumber numberWithBool: YES], @"Upcoming Songs",
56                 [NSNumber numberWithBool: YES], @"Playlists",
57                 [NSNumber numberWithBool: YES], @"Volume",
58                 [NSNumber numberWithBool: YES], @"Shuffle",
59                 [NSNumber numberWithBool: YES], @"Repeat Modes",
60                 [NSNumber numberWithBool: YES], @"Equalizer",
61                 [NSNumber numberWithBool: YES], @"Track Rating",
62                 nil];
63 }
64
65 - (BOOL)showPrimaryInterface
66 {
67     ITDebugLog(@"Showing player primary interface.");
68     
69     //If the window is closed
70     _winClosed = YES;
71     
72     //If not minimized and visible
73     if ( ([[ITAppleEventCenter sharedCenter] sendAEWithSendStringForNumber:@"'----':obj { form:'prop', want:type('prop'), seld:type('pMin'), from:obj { form:'indx', want:type('cBrW'), seld:1, from:'null'() } }" eventClass:@"core" eventID:@"getd" appPSN:savedPSN] == 0) &&
74          ([[ITAppleEventCenter sharedCenter] sendAEWithSendStringForNumber:@"'----':obj { form:'prop', want:type('prop'), seld:type('pvis'), from:obj { form:'indx', want:type('cBrW'), seld:1, from:'null'() } }" eventClass:@"core" eventID:@"getd" appPSN:savedPSN] != 0) &&
75          [[[[NSWorkspace sharedWorkspace] activeApplication] objectForKey:@"NSApplicationName"] isEqualToString:@"iTunes"] ) {
76         //set minimized of browser window 1 to true
77         [[ITAppleEventCenter sharedCenter] sendAEWithSendString:@"data:long(1), '----':obj { form:'prop', want:type('prop'), seld:type('pMin'), from:obj { form:'indx', want:type('cBrW'), seld:1, from:'null'() } }" eventClass:@"core" eventID:@"setd" appPSN:savedPSN];
78     } else {
79         //set minimized of browser window 1 to false
80         [[ITAppleEventCenter sharedCenter] sendAEWithSendString:@"data:long(0), '----':obj { form:'prop', want:type('prop'), seld:type('pMin'), from:obj { form:'indx', want:type('cBrW'), seld:1, from:'null'() } }" eventClass:@"core" eventID:@"setd" appPSN:savedPSN];
81     }
82     //set visible of browser window 1 to true
83     [[ITAppleEventCenter sharedCenter] sendAEWithSendString:@"data:long(1), '----':obj { form:'prop', want:type('prop'), seld:type('pvis'), from:obj { form:'indx', want:type('cBrW'), seld:1, from:'null'() } }" eventClass:@"core" eventID:@"setd" appPSN:savedPSN];
84     //active iTunes
85     [[ITAppleEventCenter sharedCenter] sendAEWithSendString:@"data:long(1), '----':obj { form:'prop', want:type('prop'), seld:type('pisf'), from:'null'() }" eventClass:@"core" eventID:@"setd" appPSN:savedPSN];
86     ITDebugLog(@"Done showing player primary interface.");
87     return YES;
88 }
89
90 - (ITMTRemotePlayerRunningState)playerRunningState
91 {
92     NSArray *apps = [[NSWorkspace sharedWorkspace] launchedApplications];
93     int i;
94     int count = [apps count];
95     
96     for (i = 0; i < count; i++) {
97         if ([[[apps objectAtIndex:i] objectForKey:@"NSApplicationName"] isEqualToString:@"iTunes"]) {
98             ITDebugLog(@"Player running state: 1");
99             return ITMTRemotePlayerRunning;
100         }
101     }
102     ITDebugLog(@"Player running state: 0");
103     return ITMTRemotePlayerNotRunning;
104 }
105
106 - (ITMTRemotePlayerPlayingState)playerPlayingState
107 {
108     long result;
109     
110     ITDebugLog(@"Getting player playing state");
111     
112     result = [[ITAppleEventCenter sharedCenter] sendAEWithSendStringForNumber:@"'----':obj { form:'prop', want:type('prop'), seld:type('pPlS'), from:'null'() }" eventClass:@"core" eventID:@"getd" appPSN:savedPSN];
113     
114     switch (result)
115     {
116         case 'kPSP':
117             ITDebugLog(@"Getting player playing state done. Player state: Playing");
118             return ITMTRemotePlayerPlaying;
119         case 'kPSp':
120             ITDebugLog(@"Getting player playing state done. Player state: Paused");
121             return ITMTRemotePlayerPaused;
122         case 'kPSR':
123             ITDebugLog(@"Getting player playing state done. Player state: Rewinding");
124             return ITMTRemotePlayerRewinding;
125         case 'kPSF':
126             ITDebugLog(@"Getting player playing state done. Player state: Forwarding");
127             return ITMTRemotePlayerForwarding;
128         case 'kPSS':
129         default:
130             ITDebugLog(@"Getting player playing state done. Player state: Stopped");
131             return ITMTRemotePlayerStopped;
132     }
133     ITDebugLog(@"Getting player playing state done. Player state: Stopped");
134     return ITMTRemotePlayerStopped;
135 }
136
137 /*- (NSArray *)playlists
138 {
139     long i = 0;
140     const signed long numPlaylists = [[ITAppleEventCenter sharedCenter] sendAEWithSendStringForNumber:@"kocl:type('cPly'), '----':()" eventClass:@"core" eventID:@"cnte" appPSN:savedPSN];
141     NSMutableArray *playlists = [[NSMutableArray alloc] initWithCapacity:numPlaylists];
142     
143     for (i = 1; i <= numPlaylists; i++) {
144         const long j = i;
145         NSString *sendStr = [NSString stringWithFormat:@"'----':obj { form:'prop', want:type('prop'), seld:type('pnam'), from:obj { form:'indx', want:type('cPly'), seld:long(%lu), from:'null'() } }",(unsigned long)j];
146         NSString *theObj = [[ITAppleEventCenter sharedCenter] sendAEWithSendString:sendStr eventClass:@"core" eventID:@"getd" appPSN:savedPSN];
147         [playlists addObject:theObj];
148     }
149     return [playlists autorelease];
150 }*/
151
152 //Full source awareness
153 - (NSArray *)playlists
154 {
155     unsigned long i, k;
156     const signed long numSources = [[ITAppleEventCenter sharedCenter] sendAEWithSendStringForNumber:@"kocl:type('cSrc'), '----':()" eventClass:@"core" eventID:@"cnte" appPSN:savedPSN];
157     NSMutableArray *allSources = [[NSMutableArray alloc] init];
158     
159     ITDebugLog(@"Getting playlists.");
160     if (numSources == 0) {
161         ITDebugLog(@"No sources.");
162         return nil;
163     }
164     
165     for (k = 1; k <= numSources ; k++) {
166         const signed long numPlaylists = [[ITAppleEventCenter sharedCenter] sendAEWithSendStringForNumber:[NSString stringWithFormat:@"kocl:type('cPly'), '----':obj { form:'indx', want:type('cSrc'), seld:long(%u), from:() }",k] eventClass:@"core" eventID:@"cnte" appPSN:savedPSN];
167         unsigned long fourcc = [[ITAppleEventCenter sharedCenter] sendAEWithSendStringForNumber:[NSString stringWithFormat:@"'----':obj { form:'prop', want:type('prop'), seld:type('pKnd'), from:obj { form:'indx', want:type('cSrc'), seld:long(%u), from:() } }",k] eventClass:@"core" eventID:@"getd" appPSN:savedPSN];
168         NSString *sourceName = [[ITAppleEventCenter sharedCenter] sendAEWithSendString:[NSString stringWithFormat:@"'----':obj { form:'prop', want:type('prop'), seld:type('pnam'), from:obj { form:'indx', want:type('cSrc'), seld:long(%u), from:() } }",k] eventClass:@"core" eventID:@"getd" appPSN:savedPSN];
169         unsigned long class;
170         NSMutableArray *aSource = [[NSMutableArray alloc] init];
171         [aSource addObject:[[sourceName copy] autorelease]];
172         switch (fourcc) {
173             case 'kTun':
174                 class = ITMTRemoteRadioSource;
175                 break;
176             case 'kDev':
177                 class = ITMTRemoteGenericDeviceSource;
178                 break;
179             case 'kPod':
180                 class = ITMTRemoteiPodSource;
181                 break;
182             case 'kMCD':
183             case 'kACD':
184                 class = ITMTRemoteCDSource;
185                 break;
186             case 'kShd':
187                 class = ITMTRemoteSharedLibrarySource;
188                 break;
189             case 'kUnk':
190             case 'kLib':
191             default:
192                 class = ITMTRemoteLibrarySource;
193                 break;
194         }
195         ITDebugLog(@"Adding source %@ of type %i", sourceName, class);
196         [aSource addObject:[NSNumber numberWithInt:class]];
197         for (i = 1; i <= numPlaylists; i++) {
198             NSString *sendStr = [NSString stringWithFormat:@"'----':obj { form:'prop', want:type('prop'), seld:type('pnam'), from:obj { form:'indx', want:type('cPly'), seld:long(%u), from:obj { form:'indx', want:type('cSrc'), seld:long(%u), from:() } } }",i,k];
199             NSString *theObj = [[ITAppleEventCenter sharedCenter] sendAEWithSendString:sendStr eventClass:@"core" eventID:@"getd" appPSN:savedPSN];
200             ITDebugLog(@" - Adding playlist %@", theObj);
201             [aSource addObject:[[theObj copy] autorelease]];
202         }
203         [allSources addObject:[aSource autorelease]];
204     }
205     ITDebugLog(@"Finished getting playlists.");
206     return [allSources autorelease];
207 }
208
209 - (int)numberOfSongsInPlaylistAtIndex:(int)index
210 {
211     int temp1;
212     ITDebugLog(@"Getting number of songs in playlist at index %i", index);
213     temp1 = [[ITAppleEventCenter sharedCenter] sendAEWithSendStringForNumber:[NSString stringWithFormat:@"kocl:type('cTrk'), '----':obj { form:'indx', want:type('cPly'), seld:long(%lu), from:obj { form:'prop', want:type('prop'), seld:type('ctnr'), from:obj { form:'prop', want:type('prop'), seld:type('pPla'), from:'null'() } } }",index] eventClass:@"core" eventID:@"cnte" appPSN:savedPSN];
214     ITDebugLog(@"Getting number of songs in playlist at index %i done", index);
215     return temp1;
216 }
217
218 - (ITMTRemotePlayerSource)currentSource
219 {
220     unsigned long fourcc;
221
222     ITDebugLog(@"Getting current source.");   
223     
224     fourcc = (unsigned long)[[ITAppleEventCenter sharedCenter] sendAEWithSendStringForNumber :[NSString stringWithFormat:@"'----':obj { form:'prop', want:type('prop'), seld:type('pKnd'), from:obj { form:'prop', want:type('prop'), seld:type('ctnr'), from:obj { form:'prop', want:type('prop'), seld:type('pPla'), from:'null'() } } }"] eventClass:@"core" eventID:@"getd" appPSN:savedPSN];
225     
226     switch (fourcc) {
227         case 'kTun':
228             ITDebugLog(@"Getting current source done. Source: Radio.");
229             return ITMTRemoteRadioSource;
230             break;
231         case 'kDev':
232             ITDebugLog(@"Getting current source done. Source: Generic Device.");
233             return ITMTRemoteGenericDeviceSource;
234         case 'kPod':
235             ITDebugLog(@"Getting current source done. Source: iPod.");
236             return ITMTRemoteiPodSource; //this is stupid
237             break;
238         case 'kMCD':
239         case 'kACD':
240             ITDebugLog(@"Getting current source done. Source: CD.");
241             return ITMTRemoteCDSource;
242             break;
243         case 'kShd':
244             ITDebugLog(@"Getting current source done. Source: Shared Library.");
245             return ITMTRemoteSharedLibrarySource;
246             break;
247         case 'kUnk':
248         case 'kLib':
249         default:
250             ITDebugLog(@"Getting current source done. Source: Library.");
251             return ITMTRemoteLibrarySource;
252             break;
253     }
254 }
255
256 - (int)currentSourceIndex
257 {
258     ITDebugLog(@"Getting current source.");   
259     return [[ITAppleEventCenter sharedCenter] sendAEWithSendStringForNumber:[NSString stringWithFormat:@"'----':obj { form:'prop', want:type('prop'), seld:type('pidx'), from:obj { form:'prop', want:type('prop'), seld:type('ctnr'), from:obj { form:'prop', want:type('prop'), seld:type('pPla'), from:'null'() } } }"] eventClass:@"core" eventID:@"getd" appPSN:savedPSN];
260 }
261
262 - (ITMTRemotePlayerPlaylistClass)currentPlaylistClass
263 {
264     int realResult;
265     ITDebugLog(@"Getting current playlist class");
266     realResult = [[ITAppleEventCenter sharedCenter] sendTwoTierAEWithRequestedKeyForNumber:@"pcls" fromObjectByKey:@"pPla" eventClass:@"core" eventID:@"getd" appPSN:savedPSN];
267     switch (realResult)
268            {
269            case 'cLiP':
270                ITDebugLog(@"Getting current playlist class done. Class: Library.");
271                return ITMTRemotePlayerLibraryPlaylist;
272                break;
273            case 'cRTP':
274                ITDebugLog(@"Getting current playlist class done. Class: Radio.");
275                return ITMTRemotePlayerRadioPlaylist;
276                break;
277            default:
278                ITDebugLog(@"Getting current playlist class done. Class: Standard playlist.");
279                return ITMTRemotePlayerPlaylist;
280            }
281 }
282
283 - (int)currentPlaylistIndex
284 {  
285     int temp1;
286     ITDebugLog(@"Getting current playlist index.");
287     temp1 = [[ITAppleEventCenter sharedCenter] sendTwoTierAEWithRequestedKeyForNumber:@"pidx" fromObjectByKey:@"pPla" eventClass:@"core" eventID:@"getd" appPSN:savedPSN];
288     ITDebugLog(@"Getting current playlist index done.");
289     return temp1;
290 }
291
292 - (NSString *)songTitleAtIndex:(int)index
293 {
294     NSString *temp1;
295     ITDebugLog(@"Getting song title at index %i.", index);
296     temp1 = [[ITAppleEventCenter sharedCenter] sendAEWithSendString:[NSString stringWithFormat:@"'----':obj { form:'prop', want:type('prop'), seld:type('pnam'), from:obj { form:'indx', want:type('cTrk'), seld:long(%lu), from:obj { form:'prop', want:type('prop'), seld:type('pPla'), from:'null'() } } }",index] eventClass:@"core" eventID:@"getd" appPSN:savedPSN];
297     ITDebugLog(@"Getting song title at index %i done.", index);
298     return ( ([temp1 length]) ? temp1 : nil ) ;
299 }
300
301 - (int)currentAlbumTrackCount
302 {
303     int temp1;
304     ITDebugLog(@"Getting current album track count.");
305     temp1 = [[ITAppleEventCenter sharedCenter] sendTwoTierAEWithRequestedKeyForNumber:@"pTrC" fromObjectByKey:@"pTrk" eventClass:@"core" eventID:@"getd" appPSN:savedPSN];
306     if ( [self currentPlaylistClass] == ITMTRemotePlayerRadioPlaylist ) { temp1 = 0; }
307     ITDebugLog(@"Getting current album track count done.");
308     return temp1;
309 }
310
311 - (int)currentSongTrack
312 {
313     int temp1;
314     ITDebugLog(@"Getting current song track.");
315     temp1 = [[ITAppleEventCenter sharedCenter] sendTwoTierAEWithRequestedKeyForNumber:@"pTrN" fromObjectByKey:@"pTrk" eventClass:@"core" eventID:@"getd" appPSN:savedPSN];
316     if ( [self currentPlaylistClass] == ITMTRemotePlayerRadioPlaylist ) { temp1 = 0; }
317     ITDebugLog(@"Getting current song track done.");
318     return temp1;
319 }
320
321 - (NSString *)playerStateUniqueIdentifier
322 {
323     NSString *temp1;
324     ITDebugLog(@"Getting current unique identifier.");
325     temp1 = [NSString stringWithFormat:@"%i-%i", [self currentPlaylistIndex], [[ITAppleEventCenter sharedCenter] sendTwoTierAEWithRequestedKeyForNumber:@"pDID" fromObjectByKey:@"pTrk" eventClass:@"core" eventID:@"getd" appPSN:savedPSN]];
326     ITDebugLog(@"Getting current unique identifier done.");
327     return ( ([temp1 length]) ? temp1 : nil ) ;
328 }
329
330 - (int)currentSongIndex
331 {
332     int temp1;
333     ITDebugLog(@"Getting current song index.");
334     temp1 = [[ITAppleEventCenter sharedCenter] sendTwoTierAEWithRequestedKeyForNumber:@"pidx" fromObjectByKey:@"pTrk" eventClass:@"core" eventID:@"getd" appPSN:savedPSN];
335     ITDebugLog(@"Getting current song index done.");
336     return temp1;
337 }
338
339 - (NSString *)currentSongTitle
340 {
341     NSString *temp1;
342     ITDebugLog(@"Getting current song title.");
343     
344     //If we're listening to the radio.
345     if ([[ITAppleEventCenter sharedCenter] sendTwoTierAEWithRequestedKeyForNumber:@"pcls" fromObjectByKey:@"pTrk" eventClass:@"core" eventID:@"getd" appPSN:savedPSN] == 'cURT') {
346         NSString *bad = [NSString stringWithUTF8String:"浳湧"];
347         temp1 = [[ITAppleEventCenter sharedCenter] sendAEWithRequestedKey:@"pStT" eventClass:@"core" eventID:@"getd" appPSN:savedPSN];
348         if ([temp1 isEqualToString:bad]) {
349             temp1 = [[ITAppleEventCenter sharedCenter] sendTwoTierAEWithRequestedKey:@"pnam" fromObjectByKey:@"pTrk" eventClass:@"core" eventID:@"getd" appPSN:savedPSN];
350         }
351         temp1 = [temp1 stringByAppendingString:@" (Stream)"];
352     } else {
353         temp1 = [[ITAppleEventCenter sharedCenter] sendTwoTierAEWithRequestedKey:@"pnam" fromObjectByKey:@"pTrk" eventClass:@"core" eventID:@"getd" appPSN:savedPSN];
354     }
355     ITDebugLog(@"Getting current song title done.");
356     return ( ([temp1 length]) ? temp1 : nil ) ;
357 }
358
359 - (NSString *)currentSongArtist
360 {
361     NSString *temp1;
362     ITDebugLog(@"Getting current song artist.");
363     if ( [self currentPlaylistClass] != ITMTRemotePlayerRadioPlaylist ) {
364         temp1 = [[ITAppleEventCenter sharedCenter] sendTwoTierAEWithRequestedKey:@"pArt" fromObjectByKey:@"pTrk" eventClass:@"core" eventID:@"getd" appPSN:savedPSN];
365     } else {
366         temp1 = @"";
367     }
368     ITDebugLog(@"Getting current song artist done.");
369     return ( ([temp1 length]) ? temp1 : nil ) ;
370 }
371
372 - (NSString *)currentSongAlbum
373 {
374     NSString *temp1;
375     ITDebugLog(@"Getting current song album.");
376     if ( [self currentPlaylistClass] != ITMTRemotePlayerRadioPlaylist ) {
377         temp1 = [[ITAppleEventCenter sharedCenter] sendTwoTierAEWithRequestedKey:@"pAlb" fromObjectByKey:@"pTrk" eventClass:@"core" eventID:@"getd" appPSN:savedPSN];
378     } else {
379         temp1 = @"";
380     }
381     ITDebugLog(@"Getting current song album done.");
382     return ( ([temp1 length]) ? temp1 : nil ) ;
383 }
384
385 - (NSString *)currentSongGenre
386 {
387     NSString *temp1;
388     ITDebugLog(@"Getting current song genre.");
389     temp1 = [[ITAppleEventCenter sharedCenter] sendTwoTierAEWithRequestedKey:@"pGen" fromObjectByKey:@"pTrk" eventClass:@"core" eventID:@"getd" appPSN:savedPSN];
390     ITDebugLog(@"Getting current song genre done.");
391     return ( ([temp1 length]) ? temp1 : nil ) ;
392 }
393
394 - (NSString *)currentSongLength
395 {
396     int temp1;
397     NSString *temp2;
398     ITDebugLog(@"Getting current song length.");
399     temp1 = [[ITAppleEventCenter sharedCenter] sendTwoTierAEWithRequestedKeyForNumber:@"pcls" fromObjectByKey:@"pTrk" eventClass:@"core" eventID:@"getd" appPSN:savedPSN];
400     temp2 = [[ITAppleEventCenter sharedCenter] sendTwoTierAEWithRequestedKey:@"pTim" fromObjectByKey:@"pTrk" eventClass:@"core" eventID:@"getd" appPSN:savedPSN];
401     if ( ([self currentPlaylistClass] == ITMTRemotePlayerRadioPlaylist) || (temp1 == 'cURT') ) { temp2 = @"Continuous"; }
402     ITDebugLog(@"Getting current song length done.");
403     return temp2;
404 }
405
406 - (NSString *)currentSongRemaining
407 {
408     long duration;
409     long current;
410     long final;
411     NSString *finalString;
412     
413     ITDebugLog(@"Getting current song remaining time.");
414     
415     duration = [[ITAppleEventCenter sharedCenter]
416                         sendTwoTierAEWithRequestedKeyForNumber:@"pDur" fromObjectByKey:@"pTrk" eventClass:@"core" eventID:@"getd" appPSN:savedPSN];
417     current = [[ITAppleEventCenter sharedCenter]
418                         sendAEWithRequestedKeyForNumber:@"pPos" eventClass:@"core" eventID:@"getd" appPSN:savedPSN];
419                         
420     final = duration - current;
421     finalString = [self formatTimeInSeconds:final];
422     
423     if ( [self currentPlaylistClass] == ITMTRemotePlayerRadioPlaylist ) { finalString = nil; }
424     
425     ITDebugLog(@"Getting current song remaining time done.");
426     
427     return finalString;
428 }
429
430 - (NSString *)currentSongElapsed
431 {
432     long final;
433     NSString *finalString;
434     
435     ITDebugLog(@"Getting current song elapsed time.");
436     
437     final = [[ITAppleEventCenter sharedCenter]
438                         sendAEWithRequestedKeyForNumber:@"pPos" eventClass:@"core" eventID:@"getd" appPSN:savedPSN];
439                         
440     finalString = [self formatTimeInSeconds:final];
441     ITDebugLog(@"Getting current song elapsed time done.");
442     return finalString;
443 }
444
445 - (NSImage *)currentSongAlbumArt
446 {
447     ITDebugLog(@"Getting current song album art.");
448     NSData *data = [[ITAppleEventCenter sharedCenter] sendAEWithSendStringForData:@"'----':obj { form:'prop', want:type('prop'), seld:type('pPCT'), from:obj { form:'indx', want:type('cArt'), seld:long(1), from:obj { form:'prop', want:type('prop'), seld:type('pTrk'), from:'null'() } } }" eventClass:@"core" eventID:@"getd" appPSN:savedPSN];
449     ITDebugLog(@"Getting current song album art done.");    
450     if (data) {
451         return [[[NSImage alloc] initWithData:data] autorelease];
452     } else {
453         return nil;
454     }
455 }
456
457 - (float)currentSongRating
458 {
459     float temp1;
460     ITDebugLog(@"Getting current song rating.");
461     temp1 = ((float)[[ITAppleEventCenter sharedCenter]
462                 sendTwoTierAEWithRequestedKeyForNumber:@"pRte" fromObjectByKey:@"pTrk" eventClass:@"core" eventID:@"getd" appPSN:savedPSN] / 100.0);
463     if ( [self currentPlaylistClass] == ITMTRemotePlayerRadioPlaylist ) { temp1 = -1.0; }
464     ITDebugLog(@"Getting current song rating done.");
465     return temp1;
466 }
467
468 - (BOOL)setCurrentSongRating:(float)rating
469 {
470     ITDebugLog(@"Setting current song rating to %f.", rating);
471     if ( [self currentPlaylistClass] == ITMTRemotePlayerRadioPlaylist ) { return NO; }
472     [[ITAppleEventCenter sharedCenter] sendAEWithSendString:[NSString stringWithFormat:@"data:long(%lu), '----':obj { form:'prop', want:type('prop'), seld:type('pRte'), from:obj { form:'indx', want:type('cTrk'), seld:long(%lu), from:obj { form:'prop', want:type('prop'), seld:type('pPla'), from:'null'() } } }",(long)(rating*100),[self currentSongIndex]] eventClass:@"core" eventID:@"setd" appPSN:savedPSN];
473     ITDebugLog(@"Setting current song rating to %f done.", rating);
474     return YES;
475 }
476
477 - (BOOL)equalizerEnabled
478 {
479     ITDebugLog(@"Getting equalizer enabled status.");
480     int thingy = [[ITAppleEventCenter sharedCenter] sendAEWithSendStringForNumber:@"'----':obj { form:type('prop'), want:type('prop'), seld:type('pEQ '), from:() }" eventClass:@"core" eventID:@"getd" appPSN:savedPSN];
481     ITDebugLog(@"Done getting equalizer enabled status.");
482     return thingy;
483 }
484
485 - (BOOL)setEqualizerEnabled:(BOOL)enabled
486 {
487     ITDebugLog(@"Setting equalizer enabled to %i.", enabled);
488     [[ITAppleEventCenter sharedCenter] sendAEWithSendString:[NSString stringWithFormat:@"data:long(%lu), '----':obj { form:'prop', want:type('prop'), seld:type('pEQ '), from:'null'() }",enabled] eventClass:@"core" eventID:@"setd" appPSN:savedPSN];
489     ITDebugLog(@"Done setting equalizer enabled to %i.", enabled);
490     return YES;
491 }
492
493 - (NSArray *)eqPresets
494 {
495     int i;
496     long numPresets = [[ITAppleEventCenter sharedCenter] sendAEWithSendStringForNumber:@"kocl:type('cEQP'), '----':(), &subj:()" eventClass:@"core" eventID:@"cnte" appPSN:savedPSN];
497     NSMutableArray *presets = [[NSMutableArray alloc] initWithCapacity:numPresets];
498     ITDebugLog(@"Getting EQ presets");
499     for (i = 1; i <= numPresets; i++) {
500         NSString *theObj = [[ITAppleEventCenter sharedCenter] sendAEWithSendString:[NSString stringWithFormat:@"'----':obj { form:'prop', want:type('prop'), seld:type('pnam'), from:obj { form:'indx', want:type('cEQP'), seld:long(%lu), from:'null'() } }",i] eventClass:@"core" eventID:@"getd" appPSN:savedPSN];
501         if (theObj) {
502             ITDebugLog(@"Adding preset %@", theObj);
503             [presets addObject:theObj];
504         }
505     }
506     ITDebugLog(@"Done getting EQ presets");
507     return [presets autorelease];
508 }
509
510 - (int)currentEQPresetIndex
511 {
512     int result;
513     ITDebugLog(@"Getting current EQ preset index.");
514     result = [[ITAppleEventCenter sharedCenter]
515                 sendTwoTierAEWithRequestedKeyForNumber:@"pidx" fromObjectByKey:@"pEQP" eventClass:@"core" eventID:@"getd" appPSN:savedPSN];
516     ITDebugLog(@"Getting current EQ preset index done.");
517     return result;
518 }
519
520 - (float)volume
521 {
522     ITDebugLog(@"Getting volume.");
523     ITDebugLog(@"Getting volume done.");
524     return (float)[[ITAppleEventCenter sharedCenter] sendAEWithRequestedKeyForNumber:@"pVol" eventClass:@"core" eventID:@"getd" appPSN:savedPSN] / 100;
525 }
526
527 - (BOOL)setVolume:(float)volume
528 {
529     ITDebugLog(@"Setting volume to %f.", volume);
530     [[ITAppleEventCenter sharedCenter] sendAEWithSendString:[NSString stringWithFormat:@"data:long(%lu), '----':obj { form:'prop', want:type('prop'), seld:type('pVol'), from:'null'() }",(long)(volume*100)] eventClass:@"core" eventID:@"setd" appPSN:savedPSN];
531     ITDebugLog(@"Setting volume to %f done.", volume);
532     return YES;
533 }
534
535 - (BOOL)shuffleEnabled
536 {
537     ITDebugLog(@"Getting shuffle enabled status.");
538     BOOL final;
539     int result = [[ITAppleEventCenter sharedCenter]
540                 sendTwoTierAEWithRequestedKeyForNumber:@"pShf" fromObjectByKey:@"pPla" eventClass:@"core" eventID:@"getd" appPSN:savedPSN];
541     if (result != 0) {
542         final = YES;
543     } else {
544         final = NO;
545     }
546     ITDebugLog(@"Getting shuffle enabled status done.");
547     return final;
548 }
549
550 - (BOOL)setShuffleEnabled:(BOOL)enabled
551 {
552     ITDebugLog(@"Set shuffle enabled to %i", enabled);
553     [[ITAppleEventCenter sharedCenter] sendAEWithSendString:[NSString stringWithFormat:@"data:long(%lu), '----':obj { form:'prop', want:type('prop'), seld:type('pShf'), from:obj { form:'prop', want:type('prop'), seld:type('pPla'), from:'null'() } }",(unsigned long)enabled] eventClass:@"core" eventID:@"setd" appPSN:savedPSN];
554     ITDebugLog(@"Set shuffle enabled to %i done", enabled);
555     return YES;
556 }
557
558 - (ITMTRemotePlayerRepeatMode)repeatMode
559 {
560     FourCharCode m00f = 0;
561     int result = 0;
562     m00f = [[ITAppleEventCenter sharedCenter]
563                 sendTwoTierAEWithRequestedKeyForNumber:@"pRpt" fromObjectByKey:@"pPla" eventClass:@"core" eventID:@"getd" appPSN:savedPSN];
564     ITDebugLog(@"Getting repeat mode.");
565     switch (m00f)
566     {
567         //case 'kRp0':
568         case 1800564815:
569             ITDebugLog(@"Repeat off");
570             result = ITMTRemotePlayerRepeatOff;
571             break;
572         case 'kRp1':
573             ITDebugLog(@"Repeat one");
574             result = ITMTRemotePlayerRepeatOne;
575             break;
576         case 'kRpA':
577             ITDebugLog(@"Repeat all");
578             result = ITMTRemotePlayerRepeatAll;
579             break;
580     }
581     ITDebugLog(@"Getting repeat mode done.");
582     return result;
583 }
584
585 - (BOOL)setRepeatMode:(ITMTRemotePlayerRepeatMode)repeatMode
586 {
587     char *m00f;
588     ITDebugLog(@"Setting repeat mode to %i", repeatMode);
589     switch (repeatMode)
590     {
591         case ITMTRemotePlayerRepeatOne:
592             m00f = "kRp1";
593             break;
594         case ITMTRemotePlayerRepeatAll:
595             m00f = "kRpA";
596             break;
597         case ITMTRemotePlayerRepeatOff:
598         default:
599             m00f = "kRp0";
600             break;
601     }
602     [[ITAppleEventCenter sharedCenter] sendAEWithSendString:[NSString stringWithFormat:@"data:'%s', '----':obj { form:'prop', want:type('prop'), seld:type('pRpt'), from:obj { form:'prop', want:type('prop'), seld:type('pPla'), from:() } }",m00f] eventClass:@"core" eventID:@"setd" appPSN:savedPSN];
603     ITDebugLog(@"Setting repeat mode to %c done", m00f);
604     return YES;
605 }
606
607 - (BOOL)play
608 {
609     ITDebugLog(@"Play");
610     [[ITAppleEventCenter sharedCenter] sendAEWithEventClass:@"hook" eventID:@"Play" appPSN:savedPSN];
611     ITDebugLog(@"Play done");
612     return YES;
613 }
614
615 - (BOOL)pause
616 {
617     ITDebugLog(@"Pause");
618     [[ITAppleEventCenter sharedCenter] sendAEWithEventClass:@"hook" eventID:@"Paus" appPSN:savedPSN];
619     ITDebugLog(@"Pause done");
620     return YES;
621 }
622
623 - (BOOL)goToNextSong
624 {
625     ITDebugLog(@"Go to next track");
626     [[ITAppleEventCenter sharedCenter] sendAEWithEventClass:@"hook" eventID:@"Next" appPSN:savedPSN];
627     ITDebugLog(@"Go to next track done");
628     return YES;
629 }
630
631 - (BOOL)goToPreviousSong
632 {
633     ITDebugLog(@"Go to previous track");
634     [[ITAppleEventCenter sharedCenter] sendAEWithEventClass:@"hook" eventID:@"Prev" appPSN:savedPSN];
635     ITDebugLog(@"Go to previous track done");
636     return YES;
637 }
638
639 - (BOOL)forward
640 {
641     ITDebugLog(@"Fast forward action");
642     [[ITAppleEventCenter sharedCenter] sendAEWithEventClass:@"hook" eventID:@"Fast" appPSN:savedPSN];
643     ITDebugLog(@"Fast forward action done");
644     return YES;
645 }
646
647 - (BOOL)rewind
648 {
649     ITDebugLog(@"Rewind action");
650     [[ITAppleEventCenter sharedCenter] sendAEWithEventClass:@"hook" eventID:@"Rwnd" appPSN:savedPSN];
651     ITDebugLog(@"Rewind action done");
652     return YES;
653 }
654
655 - (BOOL)switchToPlaylistAtIndex:(int)index
656 {
657     ITDebugLog(@"Switching to playlist at index %i", index);
658     [[ITAppleEventCenter sharedCenter] sendAEWithSendString:[NSString stringWithFormat:@"'----':obj { form:'indx', want:type('cPly'), seld:long(%lu), from:() }", index] eventClass:@"hook" eventID:@"Play" appPSN:savedPSN];
659     ITDebugLog(@"Done switching to playlist at index %i", index);
660     return YES;
661 }
662
663 - (BOOL)switchToPlaylistAtIndex:(int)index ofSourceAtIndex:(int)index2
664 {
665     ITDebugLog(@"Switching to playlist at index %i of source %i", index, index2);
666     [[ITAppleEventCenter sharedCenter] sendAEWithSendString:[NSString stringWithFormat:@"'----':obj { form:'indx', want:type('cPly'), seld:long(%lu), from: obj { form:'indx', want:type('cSrc'), seld:long(%lu), from:'null'() } }", index - 1, index2 + 1] eventClass:@"hook" eventID:@"Play" appPSN:savedPSN];
667     //{ form:'indx', want:type('cPly'), seld:long(%lu), from:obj { form:'indx', want:type('cSrc'), seld:long('%lu'), from:'null'() } } -- obj { form:'indx', want:type('cSrc'), seld:long(1), from:'null'() }
668     ITDebugLog(@"Done switching to playlist at index %i of source %i", index, index2);
669     return YES;
670 }
671
672 - (BOOL)switchToSongAtIndex:(int)index
673 {
674     ITDebugLog(@"Switching to track at index %i", index);
675     [[ITAppleEventCenter sharedCenter] sendAEWithSendString:[NSString stringWithFormat:@"'----':obj { form:'indx', want:type('cTrk'), seld:long(%lu), from:obj { form:'prop', want:type('prop'), seld:type('pPla'), from:() } }",index] eventClass:@"hook" eventID:@"Play" appPSN:savedPSN];
676     ITDebugLog(@"Done switching to track at index %i", index);
677     return YES;
678 }
679
680 - (BOOL)switchToEQAtIndex:(int)index
681 {
682     ITDebugLog(@"Switching to EQ preset at index %i", index);
683     // index should count from 0, but itunes counts from 1, so let's add 1.
684     [self setEqualizerEnabled:YES];
685     [[ITAppleEventCenter sharedCenter] sendAEWithSendString:[NSString stringWithFormat:@"'----':obj { form:'prop', want:type('prop'), seld:type('pEQP'), from:'null'() }, data:obj { form:'indx', want:type('cEQP'), seld:long(%lu), from:'null'() }",(index+1)] eventClass:@"core" eventID:@"setd" appPSN:savedPSN];
686     ITDebugLog(@"Done switching to EQ preset at index %i", index);
687     return YES;
688 }
689
690 - (ProcessSerialNumber)iTunesPSN
691 {
692     /*NSArray *apps = [[NSWorkspace sharedWorkspace] launchedApplications];
693     ProcessSerialNumber number;
694     int i;
695     int count = [apps count];
696     
697     number.highLongOfPSN = kNoProcess;
698     
699     for (i = 0; i < count; i++)
700     {
701         NSDictionary *curApp = [apps objectAtIndex:i];
702         
703         if ([[curApp objectForKey:@"NSApplicationName"] isEqualToString:@"iTunes"])
704         {
705             number.highLongOfPSN = [[curApp objectForKey:
706                 @"NSApplicationProcessSerialNumberHigh"] intValue];
707             number.lowLongOfPSN = [[curApp objectForKey:
708                 @"NSApplicationProcessSerialNumberLow"] intValue];
709         }
710     }
711     return number;*/
712     ProcessSerialNumber number;
713     number.highLongOfPSN = kNoProcess;
714     number.lowLongOfPSN = 0;
715     ITDebugLog(@"Getting iTunes' PSN.");
716     while ( (GetNextProcess(&number) == noErr) ) 
717     {
718         CFStringRef name;
719         if ( (CopyProcessName(&number, &name) == noErr) )
720         {
721             if ([(NSString *)name isEqualToString:@"iTunes"])
722             {
723                 ITDebugLog(@"iTunes' highLongOfPSN: %lu.", number.highLongOfPSN);
724                 ITDebugLog(@"iTunes' lowLongOfPSN: %lu.", number.lowLongOfPSN);
725                 ITDebugLog(@"Done getting iTunes' PSN.");
726                 return number;
727             }
728             [(NSString *)name release];
729         }
730     }
731     ITDebugLog(@"Failed getting iTunes' PSN.");
732     return number;
733 }
734
735 - (NSString*)formatTimeInSeconds:(long)seconds {
736     long final = seconds;
737     NSString *finalString;
738     if (final >= 60) {
739         if (final > 3600) {
740             finalString = [NSString stringWithFormat:@"%i:%@:%@",(final / 3600),[self zeroSixty:(int)((final % 3600) / 60)],[self zeroSixty:(int)((final % 3600) % 60)]];
741         } else {
742             finalString = [NSString stringWithFormat:@"%i:%@",(final / 60),[self zeroSixty:(int)(final % 60)]];
743         }
744     } else {
745         finalString = [NSString stringWithFormat:@"0:%@",[self zeroSixty:(int)final]];
746     }
747     return finalString;
748 }
749 - (NSString*)zeroSixty:(int)seconds {
750     if ( (seconds < 10) && (seconds > 0) ) {
751         return [NSString stringWithFormat:@"0%i",seconds];
752     } else if ( (seconds == 0) ) {
753         return [NSString stringWithFormat:@"00"];
754     } else {
755         return [NSString stringWithFormat:@"%i",seconds];
756     }
757 }
758
759 @end