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