Reverted back to the lame single source style playlist menu because I
[MenuTunes.git] / MenuController.m
1 //
2 //  MenuController.m
3 //  MenuTunes
4 //
5 //  Created by Joseph Spiros on Wed Apr 30 2003.
6 //  Copyright (c) 2003 iThink Software. All rights reserved.
7 //
8
9 #import "MenuController.h"
10 #import "MainController.h"
11 #import "ITMTRemote.h"
12 #import <ITFoundation/ITDebug.h>
13 #import <ITKit/ITHotKeyCenter.h>
14 #import <ITKit/ITHotKey.h>
15 #import <ITKit/ITKeyCombo.h>
16 #import <ITKit/ITCategory-NSMenu.h>
17
18 @interface MenuController (SubmenuMethods)
19 - (NSMenu *)ratingMenu;
20 - (NSMenu *)upcomingSongsMenu;
21 - (NSMenu *)playlistsMenu;
22 - (NSMenu *)eqMenu;
23 - (void)setKeyEquivalentForCode:(short)code andModifiers:(long)modifiers
24         onItem:(NSMenuItem *)item;
25 - (BOOL)iPodAtPathAutomaticallyUpdates:(NSString *)name;
26 @end
27
28 @implementation MenuController
29
30 - (id)init
31 {
32     if ( (self = [super init]) ) {
33         _menuLayout = [[NSMutableArray alloc] initWithCapacity:0];
34     }
35     return self;
36 }
37
38 - (NSMenu *)menu
39 {
40     NSMenu *menu = [[NSMenu alloc] initWithTitle:@""];
41     NSUserDefaults *defaults = [NSUserDefaults standardUserDefaults];
42     NSArray *menuArray = [defaults arrayForKey:@"menu"];
43     NSEnumerator *enumerator = [menuArray objectEnumerator];
44     NSString *nextObject;
45     NSMenuItem *tempItem;
46     NSEnumerator *itemEnum;
47     ITHotKey *hotKey;
48     NSArray *hotKeys = [[ITHotKeyCenter sharedCenter] allHotKeys];
49     int currentSongRating;
50     
51     //Get the information
52     NS_DURING
53         _currentPlaylist = [[[MainController sharedController] currentRemote] currentPlaylistIndex];
54         _playingRadio = ([[[MainController sharedController] currentRemote] currentPlaylistClass] == ITMTRemotePlayerRadioPlaylist);
55         currentSongRating = ( [[[MainController sharedController] currentRemote] currentSongRating] != -1 );
56     NS_HANDLER
57         [[MainController sharedController] networkError:localException];
58     NS_ENDHANDLER
59     
60     ITDebugLog(@"Reset menu if required.");
61     
62     //Kill the old submenu items
63     if ( (tempItem = [_currentMenu itemWithTag:1]) ) {
64         ITDebugLog(@"Removing \"Song Rating\" submenu.");
65         [tempItem setSubmenu:nil];
66     }
67     
68     if ( (tempItem = [_currentMenu itemWithTag:2]) ) {
69         ITDebugLog(@"Removing \"Upcoming Songs\" submenu.");
70         [tempItem setSubmenu:nil];
71     }
72     
73     if ( (tempItem = [_currentMenu itemWithTag:3]) ) {
74         ITDebugLog(@"Removing \"Playlists\" submenu.");
75         [tempItem setSubmenu:nil];
76     }
77     
78     if ( (tempItem = [_currentMenu itemWithTag:4]) ) {
79         ITDebugLog(@"Removing \"EQ Presets\" submenu.");
80         [tempItem setSubmenu:nil];
81     }
82     
83     ITDebugLog(@"Begin building menu.");
84     
85     //create our menu
86     while ( (nextObject = [enumerator nextObject]) ) {
87         //Main menu items
88         if ([nextObject isEqualToString:@"playPause"]) {
89             ITDebugLog(@"Add \"Play\"/\"Pause\" menu item.");
90             tempItem = [menu addItemWithTitle:NSLocalizedString(@"play", @"Play")
91                     action:@selector(performMainMenuAction:)
92                     keyEquivalent:@""];
93             [tempItem setTag:MTMenuPlayPauseItem];
94             [tempItem setTarget:self];
95             
96             itemEnum = [hotKeys objectEnumerator];
97             while ( (hotKey = [itemEnum nextObject]) ) {
98                 if ([[hotKey name] isEqualToString:@"PlayPause"]) {
99                     ITKeyCombo *combo = [hotKey keyCombo];
100                     [self setKeyEquivalentForCode:[combo keyCode]
101                           andModifiers:[combo modifiers]
102                           onItem:tempItem];
103                 }
104             }
105             
106             ITDebugLog(@"Set \"Play\"/\"Pause\" menu item's title to correct state.");
107             NS_DURING
108                 switch ([[[MainController sharedController] currentRemote] playerPlayingState]) {
109                     case ITMTRemotePlayerPlaying:
110                         [tempItem setTitle:NSLocalizedString(@"pause", @"Pause")];
111                     break;
112                     case ITMTRemotePlayerRewinding:
113                     case ITMTRemotePlayerForwarding:
114                         [tempItem setTitle:NSLocalizedString(@"resume", @"Resume")];
115                     break;
116                     default:
117                     break;
118                 }
119             NS_HANDLER
120                 [[MainController sharedController] networkError:localException];
121             NS_ENDHANDLER
122         } else if ([nextObject isEqualToString:@"nextTrack"]) {
123             ITDebugLog(@"Add \"Next Track\" menu item.");
124             tempItem = [menu addItemWithTitle:NSLocalizedString(@"nextTrack", @"Next Track")
125                     action:@selector(performMainMenuAction:)
126                     keyEquivalent:@""];
127             
128             itemEnum = [hotKeys objectEnumerator];
129             while ( (hotKey = [itemEnum nextObject]) ) {
130                 if ([[hotKey name] isEqualToString:@"NextTrack"]) {
131                     ITKeyCombo *combo = [hotKey keyCombo];
132                     [self setKeyEquivalentForCode:[combo keyCode]
133                           andModifiers:[combo modifiers]
134                           onItem:tempItem];
135                 }
136             }
137             
138             if (_currentPlaylist) {
139                 [tempItem setTag:MTMenuNextTrackItem];
140                 [tempItem setTarget:self];
141             }
142         } else if ([nextObject isEqualToString:@"prevTrack"]) {
143             ITDebugLog(@"Add \"Previous Track\" menu item.");
144             tempItem = [menu addItemWithTitle:NSLocalizedString(@"prevTrack", @"Previous Track")
145                     action:@selector(performMainMenuAction:)
146                     keyEquivalent:@""];
147             
148             itemEnum = [hotKeys objectEnumerator];
149             while ( (hotKey = [itemEnum nextObject]) ) {
150                 if ([[hotKey name] isEqualToString:@"PrevTrack"]) {
151                     ITKeyCombo *combo = [hotKey keyCombo];
152                     [self setKeyEquivalentForCode:[combo keyCode]
153                           andModifiers:[combo modifiers]
154                           onItem:tempItem];
155                 }
156             }
157             
158             if (_currentPlaylist) {
159                 [tempItem setTag:MTMenuPreviousTrackItem];
160                 [tempItem setTarget:self];
161             }
162         } else if ([nextObject isEqualToString:@"fastForward"]) {
163             ITDebugLog(@"Add \"Fast Forward\" menu item.");
164             tempItem = [menu addItemWithTitle:NSLocalizedString(@"fastForward", @"Fast Forward")
165                     action:@selector(performMainMenuAction:)
166                     keyEquivalent:@""];
167             if (_currentPlaylist) {
168                 [tempItem setTag:MTMenuFastForwardItem];
169                 [tempItem setTarget:self];
170             }
171         } else if ([nextObject isEqualToString:@"rewind"]) {
172             ITDebugLog(@"Add \"Rewind\" menu item.");
173             tempItem = [menu addItemWithTitle:NSLocalizedString(@"rewind", @"Rewind")
174                     action:@selector(performMainMenuAction:)
175                     keyEquivalent:@""];
176             if (_currentPlaylist) {
177                 [tempItem setTag:MTMenuRewindItem];
178                 [tempItem setTarget:self];
179             }
180         } else if ([nextObject isEqualToString:@"showPlayer"]) {
181             ITDebugLog(@"Add \"Show Player\" menu item.");
182             NS_DURING
183                 tempItem = [menu addItemWithTitle:[NSString stringWithFormat:@"%@ %@",
184                                 NSLocalizedString(@"show", @"Show"),
185                                     [[[MainController sharedController] currentRemote] playerSimpleName]]
186                                 action:@selector(performMainMenuAction:)
187                                 keyEquivalent:@""];
188             NS_HANDLER
189                 [[MainController sharedController] networkError:localException];
190             NS_ENDHANDLER
191             
192             itemEnum = [hotKeys objectEnumerator];
193             while ( (hotKey = [itemEnum nextObject]) ) {
194                 if ([[hotKey name] isEqualToString:@"ShowPlayer"]) {
195                     ITKeyCombo *combo = [hotKey keyCombo];
196                     [self setKeyEquivalentForCode:[combo keyCode]
197                           andModifiers:[combo modifiers]
198                           onItem:tempItem];
199                 }
200             }
201             
202             [tempItem setTarget:self];
203             [tempItem setTag:MTMenuShowPlayerItem];
204         } else if ([nextObject isEqualToString:@"preferences"]) {
205             ITDebugLog(@"Add \"Preferences...\" menu item.");
206             tempItem = [menu addItemWithTitle:NSLocalizedString(@"preferences", @"Preferences...")
207                     action:@selector(performMainMenuAction:)
208                     keyEquivalent:@""];
209             [tempItem setTag:MTMenuPreferencesItem];
210             [tempItem setTarget:self];
211         } else if ([nextObject isEqualToString:@"quit"]) {
212             if ([[MainController sharedController] blingBling] == NO) {
213                 ITDebugLog(@"Add \"Register MenuTunes...\" menu item.");
214                 tempItem = [menu addItemWithTitle:NSLocalizedString(@"register", @"Register MenuTunes...") action:@selector(performMainMenuAction:) keyEquivalent:@""];
215                 [tempItem setTag:MTMenuRegisterItem];
216                 [tempItem setTarget:self];
217             }
218             ITDebugLog(@"Add \"Quit\" menu item.");
219             tempItem = [menu addItemWithTitle:NSLocalizedString(@"quit", @"Quit")
220                     action:@selector(performMainMenuAction:)
221                     keyEquivalent:@""];
222             [tempItem setTag:MTMenuQuitItem];
223             [tempItem setTarget:self];
224         } else if ([nextObject isEqualToString:@"trackInfo"]) {
225             ITDebugLog(@"Check to see if a Track is playing...");
226             //Handle playing radio too
227             if (_currentPlaylist) {
228                 NSString *title;
229                 NS_DURING
230                     title = [[[MainController sharedController] currentRemote] currentSongTitle];
231                 NS_HANDLER
232                     [[MainController sharedController] networkError:localException];
233                 NS_ENDHANDLER
234                 ITDebugLog(@"A Track is Playing, Add \"Track Info\" menu items.");
235                 ITDebugLog(@"Add \"Now Playing\" menu item.");
236                 [menu addItemWithTitle:NSLocalizedString(@"nowPlaying", @"Now Playing") action:NULL keyEquivalent:@""];
237                 
238                 if ([title length] > 0) {
239                     ITDebugLog(@"Add Track Title (\"%@\") menu item.", title);
240                     [menu indentItem:
241                         [menu addItemWithTitle:title action:nil keyEquivalent:@""]];
242                 }
243                 
244                 if (!_playingRadio) {
245                     if ([defaults boolForKey:@"showAlbum"]) {
246                         NSString *curAlbum;
247                         NS_DURING
248                             curAlbum = [[[MainController sharedController] currentRemote] currentSongAlbum];
249                         NS_HANDLER
250                             [[MainController sharedController] networkError:localException];
251                         NS_ENDHANDLER
252                         ITDebugLog(@"Add Track Album (\"%@\") menu item.", curAlbum);
253                         if ( curAlbum ) {
254                             [menu indentItem:
255                                 [menu addItemWithTitle:curAlbum action:nil keyEquivalent:@""]];
256                         }
257                     }
258                     
259                     if ([defaults boolForKey:@"showArtist"]) {
260                         NSString *curArtist;
261                         NS_DURING
262                             curArtist = [[[MainController sharedController] currentRemote] currentSongArtist];
263                         NS_HANDLER
264                             [[MainController sharedController] networkError:localException];
265                         NS_ENDHANDLER
266                         ITDebugLog(@"Add Track Artist (\"%@\") menu item.", curArtist);
267                         if ( curArtist ) {
268                             [menu indentItem:
269                                 [menu addItemWithTitle:curArtist action:nil keyEquivalent:@""]];
270                         }
271                     }
272                     
273                     if ([defaults boolForKey:@"showTrackNumber"]) {
274                         int track;
275                         NS_DURING
276                             track = [[[MainController sharedController] currentRemote] currentSongTrack];
277                         NS_HANDLER
278                             [[MainController sharedController] networkError:localException];
279                         NS_ENDHANDLER
280                         ITDebugLog(@"Add Track Number (\"Track %i\") menu item.", track);
281                         if ( track > 0 ) {
282                             [menu indentItem:
283                                 [menu addItemWithTitle:[NSString stringWithFormat:@"%@ %i", NSLocalizedString(@"track", @"Track"), track] action:nil keyEquivalent:@""]];
284                         }
285                     }
286                 }
287                 
288                 NS_DURING
289                     if ([defaults boolForKey:@"showTime"] && ( ([[[MainController sharedController] currentRemote] currentSongElapsed] != nil) || ([[[MainController sharedController] currentRemote] currentSongLength] != nil) )) {
290                         ITDebugLog(@"Add Track Elapsed (\"%@/%@\") menu item.", [[[MainController sharedController] currentRemote] currentSongElapsed], [[[MainController sharedController] currentRemote] currentSongLength]);
291                         [menu indentItem:[menu addItemWithTitle:[NSString stringWithFormat:@"%@/%@", [[[MainController sharedController] currentRemote] currentSongElapsed], [[[MainController sharedController] currentRemote] currentSongLength]] action:nil keyEquivalent:@""]];
292                     }
293                 NS_HANDLER
294                     [[MainController sharedController] networkError:localException];
295                 NS_ENDHANDLER
296                 
297                 if (!_playingRadio) {
298                     NS_DURING
299                         if ([defaults boolForKey:@"showTrackRating"] && ( [[[MainController sharedController] currentRemote] currentSongRating] != -1.0 )) {
300                             NSString *string = nil;
301                             switch ((int)([[[MainController sharedController] currentRemote] currentSongRating] * 5)) {
302                                 case 0:
303                                     string = [NSString stringWithUTF8String:"☆☆☆☆☆"];
304                                 break;
305                                 case 1:
306                                     string = [NSString stringWithUTF8String:"★☆☆☆☆"];
307                                 break;
308                                 case 2:
309                                     string = [NSString stringWithUTF8String:"★★☆☆☆"];
310                                 break;
311                                 case 3:
312                                     string = [NSString stringWithUTF8String:"★★★☆☆"];
313                                 break;
314                                 case 4:
315                                     string = [NSString stringWithUTF8String:"★★★★☆"];
316                                 break;
317                                 case 5:
318                                     string = [NSString stringWithUTF8String:"★★★★★"];
319                                 break;
320                             }
321                             ITDebugLog(@"Add Track Rating (\"%@\") menu item.", string);
322                             [menu indentItem:[menu addItemWithTitle:string action:nil keyEquivalent:@""]];
323                         }
324                     NS_HANDLER
325                         [[MainController sharedController] networkError:localException];
326                     NS_ENDHANDLER
327                 }
328             } else {
329                 ITDebugLog(@"No Track is Playing, Add \"No Song\" menu item.");
330                 [menu addItemWithTitle:NSLocalizedString(@"noSong", @"No Song") action:NULL keyEquivalent:@""];
331             }
332         } else if ([nextObject isEqualToString:@"separator"]) {
333             ITDebugLog(@"Add a separator menu item.");
334             [menu addItem:[NSMenuItem separatorItem]];
335         //Submenu items
336         } else if ([nextObject isEqualToString:@"playlists"]) {
337             ITDebugLog(@"Add \"Playlists\" submenu.");
338             tempItem = [menu addItemWithTitle:NSLocalizedString(@"playlists", @"Playlists")
339                     action:nil
340                     keyEquivalent:@""];
341             [tempItem setSubmenu:_playlistsMenu];
342             [tempItem setTag:3];
343         } else if ([nextObject isEqualToString:@"eqPresets"]) {
344             ITDebugLog(@"Add \"EQ Presets\" submenu.");
345             tempItem = [menu addItemWithTitle:NSLocalizedString(@"eqPresets", @"EQ Presets")
346                     action:nil
347                     keyEquivalent:@""];
348             [tempItem setSubmenu:_eqMenu];
349             [tempItem setTag:4];
350             
351             itemEnum = [[_eqMenu itemArray] objectEnumerator];
352             while ( (tempItem = [itemEnum nextObject]) ) {
353                 [tempItem setState:NSOffState];
354             }
355             NS_DURING
356                 [[_eqMenu itemAtIndex:([[[MainController sharedController] currentRemote] currentEQPresetIndex] - 1)] setState:NSOnState];
357             NS_HANDLER
358                 [[MainController sharedController] networkError:localException];
359             NS_ENDHANDLER
360         } else if ([nextObject isEqualToString:@"songRating"] && currentSongRating) {
361                 ITDebugLog(@"Add \"Song Rating\" submenu.");
362                 tempItem = [menu addItemWithTitle:NSLocalizedString(@"songRating", @"Song Rating")
363                         action:nil
364                         keyEquivalent:@""];
365                 [tempItem setSubmenu:_ratingMenu];
366                 [tempItem setTag:1];
367                 if (_playingRadio || !_currentPlaylist) {
368                     [tempItem setEnabled:NO];
369                 }
370                 
371                 itemEnum = [[_ratingMenu itemArray] objectEnumerator];
372                 while ( (tempItem = [itemEnum nextObject]) ) {
373                     [tempItem setState:NSOffState];
374                 }
375                 
376                 NS_DURING
377                     [[_ratingMenu itemAtIndex:([[[MainController sharedController] currentRemote] currentSongRating] * 5)] setState:NSOnState];
378                 NS_HANDLER
379                     [[MainController sharedController] networkError:localException];
380                 NS_ENDHANDLER
381             } else if ([nextObject isEqualToString:@"upcomingSongs"]) {
382                 ITDebugLog(@"Add \"Upcoming Songs\" submenu.");
383                 tempItem = [menu addItemWithTitle:NSLocalizedString(@"upcomingSongs", @"Upcoming Songs")
384                         action:nil
385                         keyEquivalent:@""];
386                 [tempItem setSubmenu:_upcomingSongsMenu];
387                 [tempItem setTag:2];
388                 if (_playingRadio || !_currentPlaylist) {
389                     [tempItem setEnabled:NO];
390                 }
391             }
392         }
393     ITDebugLog(@"Finished building menu.");
394     [_currentMenu release];
395     _currentMenu = menu;
396     return _currentMenu;
397 }
398
399 - (NSMenu *)menuForNoPlayer
400 {
401     NSMenu *menu = [[NSMenu alloc] initWithTitle:@""];
402     NSMenuItem *tempItem;
403     ITDebugLog(@"Creating menu for when player isn't running.");
404     NS_DURING
405         ITDebugLog(@"Add \"Open %@\" menu item.", [[[MainController sharedController] currentRemote] playerSimpleName]);
406         tempItem = [menu addItemWithTitle:[NSString stringWithFormat:@"%@ %@", NSLocalizedString(@"open", @"Open"), [[[MainController sharedController] currentRemote] playerSimpleName]] action:@selector(performMainMenuAction:) keyEquivalent:@""];
407     NS_HANDLER
408         [[MainController sharedController] networkError:localException];
409     NS_ENDHANDLER
410     [tempItem setTag:MTMenuShowPlayerItem];
411     [tempItem setTarget:self];
412     ITDebugLog(@"Add a separator menu item.");
413     [menu addItem:[NSMenuItem separatorItem]];
414     ITDebugLog(@"Add \"Preferences...\" menu item.");
415     tempItem = [menu addItemWithTitle:NSLocalizedString(@"preferences", @"Preferences...") action:@selector(performMainMenuAction:) keyEquivalent:@""];
416     [tempItem setTag:MTMenuPreferencesItem];
417     [tempItem setTarget:self];
418     if ([[MainController sharedController] blingBling] == NO) {
419         ITDebugLog(@"Add \"Register MenuTunes...\" menu item.");
420         tempItem = [menu addItemWithTitle:NSLocalizedString(@"register", @"Register MenuTunes...") action:@selector(performMainMenuAction:) keyEquivalent:@""];
421         [tempItem setTag:MTMenuRegisterItem];
422         [tempItem setTarget:self];
423     }
424     ITDebugLog(@"Add \"Quit\" menu item.");
425     tempItem = [menu addItemWithTitle:NSLocalizedString(@"quit", @"Quit") action:@selector(performMainMenuAction:) keyEquivalent:@""];
426     [tempItem setTag:MTMenuQuitItem];
427     [tempItem setTarget:self];
428     return [menu autorelease];
429 }
430
431 - (void)rebuildSubmenus
432 {
433     ITDebugLog(@"Rebuilding all of the submenus.");
434     NS_DURING
435         _currentPlaylist = [[[MainController sharedController] currentRemote] currentPlaylistIndex];
436         _currentTrack = [[[MainController sharedController] currentRemote] currentSongIndex];
437         _playingRadio = ([[[MainController sharedController] currentRemote] currentPlaylistClass] == ITMTRemotePlayerRadioPlaylist);
438     NS_HANDLER
439         [[MainController sharedController] networkError:localException];
440     NS_ENDHANDLER
441     [_ratingMenu release];
442     [_upcomingSongsMenu release];
443     [_playlistsMenu release];
444     [_eqMenu release];
445     ITDebugLog(@"Beginning Rebuild of \"Song Rating\" submenu.");
446     _ratingMenu = [self ratingMenu];
447     ITDebugLog(@"Beginning Rebuild of \"Upcoming Songs\" submenu.");
448     _upcomingSongsMenu = [self upcomingSongsMenu];
449     ITDebugLog(@"Beginning Rebuild of \"Playlists\" submenu.");
450     _playlistsMenu = [self playlistsMenu];
451     ITDebugLog(@"Beginning Rebuild of \"EQ Presets\" submenu.");
452     _eqMenu = [self eqMenu];
453     ITDebugLog(@"Done rebuilding all of the submenus.");
454 }
455
456 - (NSMenu *)ratingMenu
457 {
458     NSMenu *ratingMenu = [[NSMenu alloc] initWithTitle:@""];
459     NSEnumerator *itemEnum;
460     id  anItem;
461     int itemTag = 0;
462     SEL itemSelector = @selector(performRatingMenuAction:);
463     
464     ITDebugLog(@"Building \"Song Rating\" menu.");
465     
466     [ratingMenu addItemWithTitle:[NSString stringWithUTF8String:"☆☆☆☆☆"] action:nil keyEquivalent:@""];
467     [ratingMenu addItemWithTitle:[NSString stringWithUTF8String:"★☆☆☆☆"] action:nil keyEquivalent:@""];
468     [ratingMenu addItemWithTitle:[NSString stringWithUTF8String:"★★☆☆☆"] action:nil keyEquivalent:@""];
469     [ratingMenu addItemWithTitle:[NSString stringWithUTF8String:"★★★☆☆"] action:nil keyEquivalent:@""];
470     [ratingMenu addItemWithTitle:[NSString stringWithUTF8String:"★★★★☆"] action:nil keyEquivalent:@""];
471     [ratingMenu addItemWithTitle:[NSString stringWithUTF8String:"★★★★★"] action:nil keyEquivalent:@""];
472     
473     itemEnum = [[ratingMenu itemArray] objectEnumerator];
474     while ( (anItem = [itemEnum nextObject]) ) {
475         ITDebugLog(@"Setting up \"%@\" menu item.", [anItem title]);
476         [anItem setAction:itemSelector];
477         [anItem setTarget:self];
478         [anItem setTag:itemTag];
479         itemTag += 20;
480     }
481     ITDebugLog(@"Done Building \"Song Rating\" menu.");
482     return ratingMenu;
483 }
484
485 - (NSMenu *)upcomingSongsMenu
486 {
487     NSMenu *upcomingSongsMenu = [[NSMenu alloc] initWithTitle:@""];
488     int numSongs, numSongsInAdvance = [[NSUserDefaults standardUserDefaults] integerForKey:@"SongsInAdvance"];
489     
490     NS_DURING
491         numSongs = [[[MainController sharedController] currentRemote] numberOfSongsInPlaylistAtIndex:_currentPlaylist];
492     NS_HANDLER
493         [[MainController sharedController] networkError:localException];
494     NS_ENDHANDLER
495     
496     ITDebugLog(@"Building \"Upcoming Songs\" menu.");
497     if (_currentPlaylist && !_playingRadio) {
498         if (numSongs > 0) {
499             int i;
500
501             for (i = _currentTrack + 1; i <= _currentTrack + numSongsInAdvance; i++) {
502                 if (i <= numSongs) {
503                     NSString *curSong;
504                     NS_DURING
505                         curSong = [[[MainController sharedController] currentRemote] songTitleAtIndex:i];
506                     NS_HANDLER
507                         [[MainController sharedController] networkError:localException];
508                     NS_ENDHANDLER
509                     NSMenuItem *songItem;
510                     ITDebugLog(@"Adding song: %@", curSong);
511                     songItem = [upcomingSongsMenu addItemWithTitle:curSong action:@selector(performUpcomingSongsMenuAction:) keyEquivalent:@""];
512                     [songItem setTag:i];
513                     [songItem setTarget:self];
514                 } else {
515                     break;
516                 }
517             }
518         }
519         
520         if ([upcomingSongsMenu numberOfItems] == 0) {
521             [upcomingSongsMenu addItemWithTitle:NSLocalizedString(@"noUpcomingSongs", @"No upcoming songs.") action:NULL keyEquivalent:@""];
522         }
523     }
524     ITDebugLog(@"Done Building \"Upcoming Songs\" menu.");
525     return upcomingSongsMenu;
526 }
527
528 - (NSMenu *)playlistsMenu
529 {
530     NSMenu *playlistsMenu = [[NSMenu alloc] initWithTitle:@""];
531     NSArray *playlists;
532     NSMenuItem *tempItem;
533     ITMTRemotePlayerSource source = [[[MainController sharedController] currentRemote] currentSource];
534     int i;
535     NS_DURING
536         playlists = [[[MainController sharedController] currentRemote] playlists];
537     NS_HANDLER
538         [[MainController sharedController] networkError:localException];
539     NS_ENDHANDLER
540     
541     ITDebugLog(@"Building \"Playlists\" menu.");
542     
543     for (i = 0; i < [playlists count]; i++) {
544         NSString *curPlaylist = [playlists objectAtIndex:i];
545         ITDebugLog(@"Adding playlist: %@", curPlaylist);
546         tempItem = [playlistsMenu addItemWithTitle:curPlaylist action:@selector(performPlaylistMenuAction:) keyEquivalent:@""];
547         [tempItem setTag:i + 1];
548         [tempItem setTarget:self];
549     }
550     
551     if (source == ITMTRemoteRadioSource) {
552         [[playlistsMenu addItemWithTitle:NSLocalizedString(@"radio", @"Radio") action:NULL keyEquivalent:@""] setState:NSOnState];
553     } else if (source == ITMTRemoteGenericDeviceSource) {
554         [[playlistsMenu addItemWithTitle:NSLocalizedString(@"genericDevice", @"Generic Device") action:NULL keyEquivalent:@""] setState:NSOnState];
555     } else if (source == ITMTRemoteiPodSource) {
556         [[playlistsMenu addItemWithTitle:NSLocalizedString(@"iPod", @"iPod") action:NULL keyEquivalent:@""] setState:NSOnState];
557     } else if (source == ITMTRemoteCDSource) {
558         [[playlistsMenu addItemWithTitle:NSLocalizedString(@"cd", @"CD") action:NULL keyEquivalent:@""] setState:NSOnState];
559     } else if (source == ITMTRemoteSharedLibrarySource) {
560         [[playlistsMenu addItemWithTitle:NSLocalizedString(@"sharedLibrary", @"Shared Library") action:NULL keyEquivalent:@""] setState:NSOnState];
561     } else if (source == ITMTRemoteLibrarySource && _currentPlaylist) {
562         [[playlistsMenu itemAtIndex:_currentPlaylist - 1] setState:NSOnState];
563     }
564     ITDebugLog(@"Done Building \"Playlists\" menu");
565     return playlistsMenu;
566 }
567
568
569 /*- (NSMenu *)playlistsMenu
570 {
571     NSMenu *playlistsMenu = [[NSMenu alloc] initWithTitle:@""];
572     NSArray *playlists;
573     NSMenuItem *tempItem;
574     ITMTRemotePlayerSource source = [[[MainController sharedController] currentRemote] currentSource];
575     int i, j;
576     NS_DURING
577         playlists = [[[MainController sharedController] currentRemote] playlists];
578     NS_HANDLER
579         [[MainController sharedController] networkError:localException];
580     NS_ENDHANDLER
581     ITDebugLog(@"Building \"Playlists\" menu.");
582     {
583         NSArray *curPlaylist = [playlists objectAtIndex:0];
584         NSString *name = [curPlaylist objectAtIndex:0];
585         ITDebugLog(@"Adding main source: %@", name);
586         for (i = 2; i < [curPlaylist count]; i++) {
587             ITDebugLog(@"Adding playlist: %@", [curPlaylist objectAtIndex:i]);
588             tempItem = [playlistsMenu addItemWithTitle:[curPlaylist objectAtIndex:i] action:@selector(performPlaylistMenuAction:) keyEquivalent:@""];
589             [tempItem setTag:i];
590             [tempItem setTarget:self];
591         }
592     }
593     
594     if ( (source == ITMTRemoteRadioSource) || ([playlists count] - 2 > 0) ) {
595         [playlistsMenu addItem:[NSMenuItem separatorItem]];
596     }
597     
598     if (source == ITMTRemoteRadioSource) {
599         [[playlistsMenu addItemWithTitle:NSLocalizedString(@"radio", @"Radio") action:@selector(performPlaylistMenuAction:) keyEquivalent:@""] setState:NSOnState];
600     }
601     
602     for (i = 2; i < [playlists count]; i++) {
603         NSArray *curPlaylist = [playlists objectAtIndex:i];
604         NSString *name = [curPlaylist objectAtIndex:0];
605         NSMenu *submenu = [[NSMenu alloc] init];
606         ITDebugLog(@"Adding source: %@", name);
607         
608         if ( ([[curPlaylist objectAtIndex:i] intValue] == ITMTRemoteiPodSource) && (![self iPodAtPathAutomaticallyUpdates:[curPlaylist objectAtIndex:j]]) ) {
609             ITDebugLog(@"Invalid iPod source.");
610         } else {
611             for (j = 2; j < [curPlaylist count]; j++) {
612                 ITDebugLog(@"Adding playlist: %@", [curPlaylist objectAtIndex:j]);
613                 tempItem = [submenu addItemWithTitle:[curPlaylist objectAtIndex:j] action:@selector(performPlaylistMenuAction:) keyEquivalent:@""];
614                 [tempItem setTag:(i * 1000) + j];
615                 [tempItem setTarget:self];
616             }
617             [[playlistsMenu addItemWithTitle:name action:NULL keyEquivalent:@""] setSubmenu:[submenu autorelease]];
618         }
619     }
620     
621     if ( (source == ITMTRemoteSharedLibrarySource) || (source == ITMTRemoteiPodSource) || (source == ITMTRemoteGenericDeviceSource) || (source == ITMTRemoteCDSource) ){
622         tempItem = [playlistsMenu itemAtIndex:(int)[[[MainController sharedController] currentRemote] currentSourceIndex] + [playlistsMenu numberOfItems] - 5];
623         [tempItem setState:NSOnState];
624         [[[tempItem submenu] itemAtIndex:_currentPlaylist - 1] setState:NSOnState];
625     } else if (source == ITMTRemoteLibrarySource && _currentPlaylist) {
626         [[playlistsMenu itemAtIndex:_currentPlaylist - 1] setState:NSOnState];
627     }
628     ITDebugLog(@"Done Building \"Playlists\" menu");
629     return playlistsMenu;
630 }*/
631
632 - (NSMenu *)eqMenu
633 {
634     NSMenu *eqMenu = [[NSMenu alloc] initWithTitle:@""];
635     NSArray *eqPresets;
636     NSMenuItem *tempItem;
637     int i;
638     
639     NS_DURING
640         eqPresets = [[[MainController sharedController] currentRemote] eqPresets];
641     NS_HANDLER
642         [[MainController sharedController] networkError:localException];
643     NS_ENDHANDLER
644     
645     ITDebugLog(@"Building \"EQ Presets\" menu.");
646     
647     for (i = 0; i < [eqPresets count]; i++) {
648         NSString *name;
649            if ( ( name = [eqPresets objectAtIndex:i] ) ) {
650             ITDebugLog(@"Adding EQ Preset: %@", name);
651             tempItem = [eqMenu addItemWithTitle:name
652                     action:@selector(performEqualizerMenuAction:)
653                     keyEquivalent:@""];
654             [tempItem setTag:i];
655             [tempItem setTarget:self];
656            }
657     }
658     ITDebugLog(@"Done Building \"EQ Presets\" menu");
659     return eqMenu;
660 }
661
662 - (void)performMainMenuAction:(id)sender
663 {
664     switch ( [sender tag] )
665     {
666         case MTMenuPlayPauseItem:
667             ITDebugLog(@"Performing Menu Action: Play/Pause");
668             [[MainController sharedController] playPause];
669             break;
670         case MTMenuFastForwardItem:
671             ITDebugLog(@"Performing Menu Action: Fast Forward");
672             [[MainController sharedController] fastForward];
673             break;
674         case MTMenuRewindItem:
675             ITDebugLog(@"Performing Menu Action: Rewind");
676             [[MainController sharedController] rewind];
677             break;
678         case MTMenuPreviousTrackItem:
679             ITDebugLog(@"Performing Menu Action: Previous Track");
680             [[MainController sharedController] prevSong];
681             break;
682         case MTMenuNextTrackItem:
683             ITDebugLog(@"Performing Menu Action: Next Track");
684             [[MainController sharedController] nextSong];
685             break;
686         case MTMenuShowPlayerItem:
687             ITDebugLog(@"Performing Menu Action: Show Main Interface");
688             [[MainController sharedController] showPlayer];
689             break;
690         case MTMenuPreferencesItem:
691             ITDebugLog(@"Performing Menu Action: Preferences...");
692             [[MainController sharedController] showPreferences];
693             break;
694         case MTMenuQuitItem:
695             ITDebugLog(@"Performing Menu Action: Quit");
696             [[MainController sharedController] quitMenuTunes];
697             break;
698         case MTMenuRegisterItem:
699             ITDebugLog(@"Performing Menu Action: Register");
700             [[MainController sharedController] blingNow];
701             break;
702         default:
703             ITDebugLog(@"Performing Menu Action: Unimplemented Menu Item OR Child-bearing Menu Item");
704             break;
705     }
706 }
707
708 - (void)performRatingMenuAction:(id)sender
709 {
710     ITDebugLog(@"Rating action selected on item with tag %i", [sender tag]);
711     [[MainController sharedController] selectSongRating:[sender tag]];
712 }
713
714 - (void)performPlaylistMenuAction:(id)sender
715 {
716     ITDebugLog(@"Playlist action selected on item with tag %i", [sender tag]);
717     [[MainController sharedController] selectPlaylistAtIndex:[sender tag]];
718 }
719
720 - (void)performEqualizerMenuAction:(id)sender
721 {
722     ITDebugLog(@"EQ action selected on item with tag %i", [sender tag]);
723     [[MainController sharedController] selectEQPresetAtIndex:[sender tag]];
724 }
725
726 - (void)performUpcomingSongsMenuAction:(id)sender
727 {
728     ITDebugLog(@"Song action selected on item with tag %i", [sender tag]);
729     [[MainController sharedController] selectSongAtIndex:[sender tag]];
730 }
731
732 - (void)updateMenu
733 {
734     ITDebugLog(@"Update Menu");
735     [_currentMenu update];
736 }
737
738 - (BOOL)validateMenuItem:(id <NSMenuItem>)menuItem
739 {
740     return YES;
741 }
742
743 //This is never used I know, keep it though
744 - (NSString *)systemUIColor
745 {
746     NSDictionary *tmpDict;
747     NSNumber *tmpNumber;
748     if ( (tmpDict = [NSDictionary dictionaryWithContentsOfFile:[@"~/Library/Preferences/.GlobalPreferences.plist" stringByExpandingTildeInPath]]) ) {
749         if ( (tmpNumber = [tmpDict objectForKey:@"AppleAquaColorVariant"]) ) {
750             if ( ([tmpNumber intValue] == 1) ) {
751                 return @"Aqua";
752             } else {
753                 return @"Graphite";
754             }
755         } else {
756             return @"Aqua";
757         }
758     } else {
759         return @"Aqua";
760     }
761 }
762
763 - (void)setKeyEquivalentForCode:(short)code andModifiers:(long)modifiers
764         onItem:(NSMenuItem *)item
765 {
766     unichar charcode = 'a';
767     int i;
768     long cocoaModifiers = 0;
769     static long carbonToCocoa[6][2] = 
770     {
771         { cmdKey, NSCommandKeyMask },
772         { optionKey, NSAlternateKeyMask },
773         { controlKey, NSControlKeyMask },
774         { shiftKey, NSShiftKeyMask },
775     };
776     
777     ITDebugLog(@"Setting Key Equivelent on menu item \"%@\".", [item title]);
778     
779     for (i = 0; i < 6; i++) {
780         if (modifiers & carbonToCocoa[i][0]) {
781             cocoaModifiers += carbonToCocoa[i][1];
782         }
783     }
784     [item setKeyEquivalentModifierMask:cocoaModifiers];
785     
786     //Missing key combos for some keys. Must find them later.
787     switch (code)
788     {
789         case 36:
790             ITDebugLog(@"Keycode for menu item \"%@\": 36 (Return)", [item title]);
791             charcode = '\r';
792         break;
793         
794         case 48:
795             ITDebugLog(@"Keycode for menu item \"%@\": 48 (Tab)", [item title]);
796             charcode = '\t';
797         break;
798         
799         //Space -- ARGH!
800         case 49:
801         {
802             ITDebugLog(@"Keycode for menu item \"%@\": 49 (Space)", [item title]);
803             // Haven't tested this, though it should work.
804             // This doesn't work. :'(
805             //unichar buffer;
806             //[[NSString stringWithString:@"Space"] getCharacters:&buffer];
807             //charcode = buffer;
808             /*MenuRef menuRef = _NSGetCarbonMenu([item menu]);
809             ITDebugLog(@"%@", menuRef);
810             SetMenuItemCommandKey(menuRef, 0, NO, 49);
811             SetMenuItemModifiers(menuRef, 0, kMenuNoCommandModifier);
812             SetMenuItemKeyGlyph(menuRef, 0, kMenuBlankGlyph);
813             charcode = 'b';*/
814             unichar buffer;
815             [[NSString stringWithString:@" "] getCharacters:&buffer]; // this will have to do for now :(
816             charcode = buffer;
817         }
818         break;
819         
820         case 51:
821             ITDebugLog(@"Keycode for menu item \"%@\": 51 (Delete)", [item title]);
822             charcode = NSDeleteFunctionKey;
823         break;
824         
825         case 53:
826             ITDebugLog(@"Keycode for menu item \"%@\": 53 (Escape)", [item title]);
827             charcode = '\e';
828         break;
829         
830         case 71:
831             ITDebugLog(@"Keycode for menu item \"%@\": 71 (Escape)", [item title]);
832             charcode = '\e';
833         break;
834         
835         case 76:
836             ITDebugLog(@"Keycode for menu item \"%@\": 76 (Return)", [item title]);
837             charcode = '\r';
838         break;
839         
840         case 96:
841             ITDebugLog(@"Keycode for menu item \"%@\": 96 (F5)", [item title]);
842             charcode = NSF5FunctionKey;
843         break;
844         
845         case 97:
846             ITDebugLog(@"Keycode for menu item \"%@\": 97 (F6)", [item title]);
847             charcode = NSF6FunctionKey;
848         break;
849         
850         case 98:
851             ITDebugLog(@"Keycode for menu item \"%@\": 98 (F7)", [item title]);
852             charcode = NSF7FunctionKey;
853         break;
854         
855         case 99:
856             ITDebugLog(@"Keycode for menu item \"%@\": 99 (F3)", [item title]);
857             charcode = NSF3FunctionKey;
858         break;
859         
860         case 100:
861             ITDebugLog(@"Keycode for menu item \"%@\": 100 (F8)", [item title]);
862             charcode = NSF8FunctionKey;
863         break;
864         
865         case 101:
866             ITDebugLog(@"Keycode for menu item \"%@\": 101 (F9)", [item title]);
867             charcode = NSF9FunctionKey;
868         break;
869         
870         case 103:
871             ITDebugLog(@"Keycode for menu item \"%@\": 103 (F11)", [item title]);
872             charcode = NSF11FunctionKey;
873         break;
874         
875         case 105:
876             ITDebugLog(@"Keycode for menu item \"%@\": 105 (F13)", [item title]);
877             charcode = NSF13FunctionKey;
878         break;
879         
880         case 107:
881             ITDebugLog(@"Keycode for menu item \"%@\": 107 (F14)", [item title]);
882             charcode = NSF14FunctionKey;
883         break;
884         
885         case 109:
886             ITDebugLog(@"Keycode for menu item \"%@\": 109 (F10)", [item title]);
887             charcode = NSF10FunctionKey;
888         break;
889         
890         case 111:
891             ITDebugLog(@"Keycode for menu item \"%@\": 111 (F12)", [item title]);
892             charcode = NSF12FunctionKey;
893         break;
894         
895         case 113:
896             ITDebugLog(@"Keycode for menu item \"%@\": 113 (F13)", [item title]);
897             charcode = NSF13FunctionKey;
898         break;
899         
900         case 114:
901             ITDebugLog(@"Keycode for menu item \"%@\": 114 (Insert)", [item title]);
902             charcode = NSInsertFunctionKey;
903         break;
904         
905         case 115:
906             ITDebugLog(@"Keycode for menu item \"%@\": 115 (Home)", [item title]);
907             charcode = NSHomeFunctionKey;
908         break;
909         
910         case 116:
911             ITDebugLog(@"Keycode for menu item \"%@\": 116 (PgUp)", [item title]);
912             charcode = NSPageUpFunctionKey;
913         break;
914         
915         case 117:
916             ITDebugLog(@"Keycode for menu item \"%@\": 117 (Delete)", [item title]);
917             charcode = NSDeleteFunctionKey;
918         break;
919         
920         case 118:
921             ITDebugLog(@"Keycode for menu item \"%@\": 118 (F4)", [item title]);
922             charcode = NSF4FunctionKey;
923         break;
924         
925         case 119:
926             ITDebugLog(@"Keycode for menu item \"%@\": 119 (End)", [item title]);
927             charcode = NSEndFunctionKey;
928         break;
929         
930         case 120:
931             ITDebugLog(@"Keycode for menu item \"%@\": 120 (F2)", [item title]);
932             charcode = NSF2FunctionKey;
933         break;
934         
935         case 121:
936             ITDebugLog(@"Keycode for menu item \"%@\": 121 (PgDown)", [item title]);
937             charcode = NSPageDownFunctionKey;
938         break;
939         
940         case 122:
941             ITDebugLog(@"Keycode for menu item \"%@\": 122 (F1)", [item title]);
942             charcode = NSF1FunctionKey;
943         break;
944         
945         case 123:
946             ITDebugLog(@"Keycode for menu item \"%@\": 123 (Left Arrow)", [item title]);
947             charcode = NSLeftArrowFunctionKey;
948         break;
949         
950         case 124:
951             ITDebugLog(@"Keycode for menu item \"%@\": 124 (Right Arrow)", [item title]);
952             charcode = NSRightArrowFunctionKey;
953         break;
954         
955         case 125:
956             ITDebugLog(@"Keycode for menu item \"%@\": 125 (Down Arrow)", [item title]);
957             charcode = NSDownArrowFunctionKey;
958         break;
959         
960         case 126:
961             ITDebugLog(@"Keycode for menu item \"%@\": 126 (Up Arrow)", [item title]);
962             charcode = NSUpArrowFunctionKey;
963         break;
964     }
965     
966     if (charcode == 'a') {
967         unsigned long state;
968         long keyTrans;
969         char charCode;
970         Ptr kchr;
971         state = 0;
972         kchr = (Ptr) GetScriptVariable(smCurrentScript, smKCHRCache);
973         keyTrans = KeyTranslate(kchr, code, &state);
974         charCode = keyTrans;
975         ITDebugLog(@"Keycode for menu item \"%@\": %i (%c)", [item title], code, charCode);
976         [item setKeyEquivalent:[NSString stringWithCString:&charCode length:1]];
977     } else if (charcode != 'b') {
978         [item setKeyEquivalent:[NSString stringWithCharacters:&charcode length:1]];
979     }
980     ITDebugLog(@"Done setting key equivalent on menu item: %@", [item title]);
981 }
982
983 - (BOOL)iPodAtPathAutomaticallyUpdates:(NSString *)name
984 {
985     NSArray *volumes = [[NSWorkspace sharedWorkspace] mountedLocalVolumePaths];
986     NSEnumerator *volEnum = [volumes objectEnumerator];
987     NSString *nextVolume;
988     
989     while ( (nextVolume = [volEnum nextObject]) ) {
990         if ([nextVolume rangeOfString:name options:nil range:NSMakeRange(0, [name length] - 1)].location != NSNotFound) {
991             NSFileHandle *handle;
992             NSData *data;
993             NSString *path = [nextVolume stringByAppendingPathComponent:@"/iPod_Control/iTunes/iTunesPrefs"];
994             if ( ![[NSFileManager defaultManager] fileExistsAtPath:path] ) {
995                 ITDebugLog(@"Error, path isn't an iPod! %@", path);
996                 return NO;
997             }
998             handle = [NSFileHandle fileHandleForReadingAtPath:name];
999             [handle seekToFileOffset:10];
1000             data = [handle readDataOfLength:1];
1001             if ( (*((unsigned char*)[data bytes]) == 0x00) ) {
1002                 ITDebugLog(@"iPod is manually updated. %@", path);
1003                 return NO;
1004             } else if ( ( *((unsigned char*)[data bytes]) == 0x01 ) ) {
1005                 ITDebugLog(@"iPod is automatically updated. %@", path);
1006                 return YES;
1007             } else {
1008                 ITDebugLog(@"Error! Value: %h  Desc: %@ Path: %@", *((unsigned char*)[data bytes]), [data description], path);
1009                 return NO;
1010             }
1011         }
1012     }
1013 }
1014
1015 @end