/****************************************************************************** * Copyright (c) 2005-2006 Transmission authors and contributors * * Permission is hereby granted, free of charge, to any person obtaining a * copy of this software and associated documentation files (the "Software"), * to deal in the Software without restriction, including without limitation * the rights to use, copy, modify, merge, publish, distribute, sublicense, * and/or sell copies of the Software, and to permit persons to whom the * Software is furnished to do so, subject to the following conditions: * * The above copyright notice and this permission notice shall be included in * all copies or substantial portions of the Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER * DEALINGS IN THE SOFTWARE. *****************************************************************************/ #import #import "Controller.h" #import "Torrent.h" #import "TorrentCell.h" #import "StringAdditions.h" #import "Utils.h" #import "TorrentTableView.h" #import "PrefsController.h" #define TOOLBAR_OPEN @"Toolbar Open" #define TOOLBAR_REMOVE @"Toolbar Remove" #define TOOLBAR_INFO @"Toolbar Info" #define TOOLBAR_PAUSE_ALL @"Toolbar Pause All" #define TOOLBAR_RESUME_ALL @"Toolbar Resume All" #define WEBSITE_URL @"http://transmission.m0k.org/" #define FORUM_URL @"http://transmission.m0k.org/forum/" #define VERSION_PLIST_URL @"http://transmission.m0k.org/version.plist" #define GROWL_PATH @"/Library/PreferencePanes/Growl.prefPane/Contents/Resources/GrowlHelperApp.app" static void sleepCallBack( void * controller, io_service_t y, natural_t messageType, void * messageArgument ) { Controller * c = controller; [c sleepCallBack: messageType argument: messageArgument]; } @implementation Controller - (id) init { fLib = tr_init(); fTorrents = [[NSMutableArray alloc] initWithCapacity: 10]; return self; } - (void) dealloc { [fTorrents release]; tr_close( fLib ); [fAppIcon release]; [super dealloc]; } - (void) awakeFromNib { fAppIcon = [[NSApp applicationIconImage] copy]; [fPrefsController setPrefsWindow: fLib]; fDefaults = [NSUserDefaults standardUserDefaults]; //check advanced bar menu item [fAdvancedBarItem setState: [fDefaults boolForKey:@"UseAdvancedBar"] ? NSOnState : NSOffState]; fToolbar = [[NSToolbar alloc] initWithIdentifier: @"Transmission Toolbar"]; [fToolbar setDelegate: self]; [fToolbar setAllowsUserCustomization: YES]; [fToolbar setAutosavesConfiguration: YES]; [fWindow setToolbar: fToolbar]; [fWindow setDelegate: self]; [fTableView setTorrents: fTorrents]; [[fTableView tableColumnWithIdentifier: @"Torrent"] setDataCell: [[TorrentCell alloc] init]]; [fTableView registerForDraggedTypes: [NSArray arrayWithObjects: NSFilenamesPboardType, NULL]]; //Register for sleep notifications IONotificationPortRef notify; io_object_t anIterator; fRootPort = IORegisterForSystemPower( self, & notify, sleepCallBack, & anIterator); if (fRootPort) { CFRunLoopAddSource( CFRunLoopGetCurrent(), IONotificationPortGetRunLoopSource( notify ), kCFRunLoopCommonModes ); } else printf( "Could not IORegisterForSystemPower\n" ); NSString * torrentPath, * downloadFolder, * paused; NSDictionary * dic; Torrent * torrent; NSEnumerator * enumerator = [[fDefaults arrayForKey: @"History"] objectEnumerator]; while ((dic = [enumerator nextObject])) { torrentPath = [dic objectForKey: @"TorrentPath"]; downloadFolder = [dic objectForKey: @"DownloadFolder"]; paused = [dic objectForKey: @"Paused"]; if (!torrentPath || !downloadFolder || !paused) continue; torrent = [[Torrent alloc] initWithPath: torrentPath lib: fLib]; if( !torrent ) continue; [fTorrents addObject: torrent]; [torrent setFolder: downloadFolder]; if ([paused isEqualToString: @"NO"]) [torrent start]; } //check and register Growl if it is installed for this user or all users NSFileManager * manager = [NSFileManager defaultManager]; fHasGrowl = [manager fileExistsAtPath: GROWL_PATH] || [manager fileExistsAtPath: [[NSString stringWithFormat: @"~%@", GROWL_PATH] stringByExpandingTildeInPath]]; [self growlRegister: self]; //initialize badging fBadger = [[Badger alloc] init]; //update the interface every 500 ms fDownloading = 0; fSeeding = 0; fCompleted = 0; [self updateUI: nil]; fTimer = [NSTimer scheduledTimerWithTimeInterval: 1.0 target: self selector: @selector( updateUI: ) userInfo: nil repeats: YES]; [[NSRunLoop currentRunLoop] addTimer: fTimer forMode: NSModalPanelRunLoopMode]; [[NSRunLoop currentRunLoop] addTimer: fTimer forMode: NSEventTrackingRunLoopMode]; [self checkForUpdateTimer: nil]; fUpdateTimer = [NSTimer scheduledTimerWithTimeInterval: 60.0 target: self selector: @selector( checkForUpdateTimer: ) userInfo: nil repeats: YES]; } - (void) windowDidBecomeKey: (NSNotification *) n { /* Reset the number of recently completed downloads */ fCompleted = 0; } - (void) windowDidResize: (NSNotification *) n { [fTableView sizeToFit]; } - (BOOL) windowShouldClose: (id) sender { [fWindow orderOut: nil]; return NO; } - (BOOL) applicationShouldHandleReopen: (NSApplication *) app hasVisibleWindows: (BOOL) flag { [self showMainWindow: nil]; return NO; } - (NSApplicationTerminateReply)applicationShouldTerminate:(NSApplication *)sender { int active = 0; Torrent * torrent; NSEnumerator * enumerator = [fTorrents objectEnumerator]; while( ( torrent = [enumerator nextObject] ) ) { if( [torrent isActive] ) { active++; } } if (active > 0 && [fDefaults boolForKey: @"CheckQuit"]) { NSString * message = active == 1 ? @"There is an active torrent. Do you really want to quit?" : [NSString stringWithFormat: @"There are %d active torrents. Do you really want to quit?", active]; NSBeginAlertSheet(@"Confirm Quit", @"Quit", @"Cancel", nil, fWindow, self, @selector(quitSheetDidEnd:returnCode:contextInfo:), nil, nil, message); return NSTerminateLater; } return NSTerminateNow; } - (void) quitSheetDidEnd:(NSWindow *)sheet returnCode:(int)returnCode contextInfo:(void *)contextInfo { [NSApp stopModal]; [NSApp replyToApplicationShouldTerminate: (returnCode == NSAlertDefaultReturn)]; } - (void) applicationWillTerminate: (NSNotification *) notification { NSEnumerator * enumerator; Torrent * torrent; // Stop updating the interface [fTimer invalidate]; [fUpdateTimer invalidate]; //clear badge [fBadger clearBadge]; [fBadger release]; // Save history [self updateTorrentHistory]; // Stop running torrents enumerator = [fTorrents objectEnumerator]; while( ( torrent = [enumerator nextObject] ) ) { [torrent stop]; } // Wait for torrents to stop (5 seconds timeout) NSDate * start = [NSDate date]; while( [fTorrents count] ) { torrent = [fTorrents objectAtIndex: 0]; while( [[NSDate date] timeIntervalSinceDate: start] < 5 && ![torrent isPaused] ) { usleep( 100000 ); [torrent update]; } [fTorrents removeObject: torrent]; [torrent release]; } } - (void) showPreferenceWindow: (id) sender { //place the window if not open if (![fPrefsWindow isVisible]) { [fPrefsWindow center]; } [fPrefsWindow makeKeyAndOrderFront:NULL]; } - (void) folderChoiceClosed: (NSOpenPanel *) s returnCode: (int) code contextInfo: (Torrent *) torrent { if (code == NSOKButton) { [fTorrents addObject: torrent]; [torrent setFolder: [[s filenames] objectAtIndex: 0]]; [torrent start]; } else { [torrent release]; } [NSApp stopModal]; } - (void) application: (NSApplication *) sender openFiles: (NSArray *) filenames { NSString * downloadChoice, * downloadFolder, * torrentPath; Torrent * torrent; downloadChoice = [fDefaults stringForKey: @"DownloadChoice"]; downloadFolder = [fDefaults stringForKey: @"DownloadFolder"]; NSEnumerator * enumerator = [filenames objectEnumerator]; while ((torrentPath = [enumerator nextObject])) { torrent = [[Torrent alloc] initWithPath: torrentPath lib: fLib]; if( !torrent ) continue; /* Add it to the "File > Open Recent" menu */ [[NSDocumentController sharedDocumentController] noteNewRecentDocumentURL: [NSURL fileURLWithPath: torrentPath]]; if( [downloadChoice isEqualToString: @"Constant"] ) { [fTorrents addObject: torrent]; [torrent setFolder: [downloadFolder stringByExpandingTildeInPath]]; [torrent start]; } else if( [downloadChoice isEqualToString: @"Torrent"] ) { [fTorrents addObject: torrent]; [torrent setFolder: [torrentPath stringByDeletingLastPathComponent]]; [torrent start]; } else { NSOpenPanel * panel = [NSOpenPanel openPanel]; [panel setPrompt: @"Select Download Folder"]; [panel setAllowsMultipleSelection: NO]; [panel setCanChooseFiles: NO]; [panel setCanChooseDirectories: YES]; if( [panel respondsToSelector: @selector(setMessage:)] ) /* >= 10.3 */ [panel setMessage: [NSString stringWithFormat: @"Select the download folder for %@", [torrentPath lastPathComponent]]]; [panel beginSheetForDirectory: NULL file: NULL types: NULL modalForWindow: fWindow modalDelegate: self didEndSelector: @selector( folderChoiceClosed:returnCode:contextInfo: ) contextInfo: torrent]; [NSApp runModalForWindow: panel]; } } [self updateUI: nil]; [self updateTorrentHistory]; } - (void) advancedChanged: (id) sender { [fAdvancedBarItem setState: ![fAdvancedBarItem state]]; [fDefaults setObject: [fAdvancedBarItem state] == NSOffState ? @"NO" : @"YES" forKey:@"UseAdvancedBar"]; [fTableView display]; } //called on by applescript - (void) open: (NSArray *) files { fFilenames = [files retain]; [self performSelectorOnMainThread: @selector(cantFindAName:) withObject: NULL waitUntilDone: NO]; } - (void) openShowSheet: (id) sender { NSOpenPanel * panel; NSArray * fileTypes; panel = [NSOpenPanel openPanel]; fileTypes = [NSArray arrayWithObject: @"torrent"]; [panel setAllowsMultipleSelection: YES]; [panel setCanChooseFiles: YES]; [panel setCanChooseDirectories: NO]; [panel beginSheetForDirectory: NULL file: NULL types: fileTypes modalForWindow: fWindow modalDelegate: self didEndSelector: @selector( openSheetClosed:returnCode:contextInfo: ) contextInfo: NULL]; } - (void) cantFindAName: (id) sender { [self application: NSApp openFiles: fFilenames]; [fFilenames release]; } - (void) openSheetClosed: (NSOpenPanel *) s returnCode: (int) code contextInfo: (void *) info { if( code != NSOKButton ) { return; } fFilenames = [[s filenames] retain]; [self performSelectorOnMainThread: @selector(cantFindAName:) withObject: NULL waitUntilDone: NO]; } - (void) resumeTorrent: (id) sender { [self resumeTorrentWithIndex: [fTableView selectedRow]]; } - (void) resumeAllTorrents: (id) sender { Torrent * torrent; NSEnumerator * enumerator = [fTorrents objectEnumerator]; while( ( torrent = [enumerator nextObject] ) ) { [torrent start]; } [self updateUI: nil]; [self updateTorrentHistory]; } - (void) resumeTorrentWithIndex: (int) idx { Torrent * torrent = [fTorrents objectAtIndex: idx]; [torrent start]; [self updateUI: nil]; [self updateTorrentHistory]; } - (void) stopTorrent: (id) sender { [self stopTorrentWithIndex: [fTableView selectedRow]]; } - (void) stopAllTorrents: (id) sender { Torrent * torrent; NSEnumerator * enumerator = [fTorrents objectEnumerator]; while( ( torrent = [enumerator nextObject] ) ) { [torrent stop]; } [self updateUI: nil]; [self updateTorrentHistory]; } - (void) stopTorrentWithIndex: (int) idx { Torrent * torrent = [fTorrents objectAtIndex: idx]; [torrent stop]; [self updateUI: nil]; [self updateTorrentHistory]; } - (void) removeTorrentWithIndex: (int) idx deleteTorrent: (BOOL) deleteTorrent deleteData: (BOOL) deleteData { Torrent * torrent = [fTorrents objectAtIndex: idx]; if( [torrent isActive] && [fDefaults boolForKey: @"CheckRemove"] ) { NSDictionary * dict = [NSDictionary dictionaryWithObjectsAndKeys: [NSString stringWithInt: idx], @"Index", [NSString stringWithInt: deleteTorrent], @"DeleteTorrent", [NSString stringWithInt: deleteData], @"DeleteData", nil]; [dict retain]; NSBeginAlertSheet(@"Confirm Remove", @"Remove", @"Cancel", nil, fWindow, self, @selector(removeSheetDidEnd:returnCode:contextInfo:), NULL, dict, @"This torrent is active. Do you really want to remove it?"); } else { [self confirmRemoveTorrentWithIndex: idx deleteTorrent: deleteTorrent deleteData: deleteData]; } } - (void) removeSheetDidEnd:(NSWindow *)sheet returnCode:(int)returnCode contextInfo:(NSDictionary *)dict { [NSApp stopModal]; if( returnCode == NSAlertDefaultReturn ) { int idx = [[dict objectForKey:@"Index"] intValue]; [self confirmRemoveTorrentWithIndex: idx deleteTorrent: [[dict objectForKey:@"DeleteTorrent"] intValue] deleteData: [[dict objectForKey:@"DeleteData"] intValue]]; } [dict release]; } - (void) confirmRemoveTorrentWithIndex: (int) idx deleteTorrent: (BOOL) deleteTorrent deleteData: (BOOL) deleteData { Torrent * torrent = [fTorrents objectAtIndex: idx]; /* Pause if not paused already */ [torrent stop]; if( deleteData ) { [torrent trashData]; } if( deleteTorrent ) { [torrent trashTorrent]; } [fTorrents removeObject: torrent]; [torrent release]; [self updateUI: nil]; [self updateTorrentHistory]; } - (void) removeTorrent: (id) sender { [self removeTorrentWithIndex: [fTableView selectedRow] deleteTorrent: NO deleteData: NO ]; } - (void) removeTorrentDeleteFile: (id) sender { [self removeTorrentWithIndex: [fTableView selectedRow] deleteTorrent: YES deleteData: NO]; } - (void) removeTorrentDeleteData: (id) sender { [self removeTorrentWithIndex: [fTableView selectedRow] deleteTorrent: NO deleteData: YES]; } - (void) removeTorrentDeleteBoth: (id) sender { [self removeTorrentWithIndex: [fTableView selectedRow] deleteTorrent: YES deleteData: YES]; } - (void) showInfo: (id) sender { if( [fInfoPanel isVisible] ) { [fInfoPanel close]; } else { [fInfoPanel orderFront: sender]; } } - (void) updateInfoPanel { int row = [fTableView selectedRow]; if( row < 0 ) { [fInfoImageView setImage: fAppIcon]; [fInfoName setStringValue: @"No Torrent Selected"]; [fInfoSize setStringValue: @""]; [fInfoTracker setStringValue: @""]; [fInfoAnnounce setStringValue: @""]; [fInfoPieceSize setStringValue: @""]; [fInfoPieces setStringValue: @""]; [fInfoHash1 setStringValue: @""]; [fInfoHash2 setStringValue: @""]; [fInfoSeeders setStringValue: @""]; [fInfoLeechers setStringValue: @""]; [fInfoDownloaded setStringValue: @""]; [fInfoUploaded setStringValue: @""]; return; } Torrent * torrent = [fTorrents objectAtIndex: row]; [fInfoImageView setImage: [torrent iconNonFlipped]]; [fInfoName setStringValue: [torrent name]]; [fInfoSize setStringValue: [NSString stringForFileSize: [torrent size]]]; [fInfoTracker setStringValue: [torrent tracker]]; [fInfoAnnounce setStringValue: [torrent announce]]; [fInfoPieceSize setStringValue: [NSString stringForFileSize: [torrent pieceSize]]]; [fInfoPieces setIntValue: [torrent pieceCount]]; [fInfoHash1 setStringValue: [torrent hash1]]; [fInfoHash2 setStringValue: [torrent hash2]]; int seeders = [torrent seeders], leechers = [torrent leechers]; [fInfoSeeders setStringValue: seeders < 0 ? @"?" : [NSString stringWithInt: seeders]]; [fInfoLeechers setStringValue: leechers < 0 ? @"?" : [NSString stringWithInt: leechers]]; [fInfoDownloaded setStringValue: [NSString stringForFileSize: [torrent downloaded]]]; [fInfoUploaded setStringValue: [NSString stringForFileSize: [torrent uploaded]]]; } - (void) updateUI: (NSTimer *) t { float dl, ul; NSEnumerator * enumerator; Torrent * torrent; /* Update the TableView */ enumerator = [fTorrents objectEnumerator]; while( ( torrent = [enumerator nextObject] ) ) { [torrent update]; if( [torrent justFinished] ) { /* Notifications */ [self notifyGrowl: [torrent name]]; if( ![fWindow isKeyWindow] ) { fCompleted++; } } } [fTableView reloadData]; //Update the global DL/UL rates tr_torrentRates( fLib, &dl, &ul ); NSString * downloadRate = [NSString stringForSpeed: dl]; NSString * uploadRate = [NSString stringForSpeed: ul]; [fTotalDLField setStringValue: downloadRate]; [fTotalULField setStringValue: uploadRate]; [self updateInfoPanel]; //badge dock [fBadger updateBadgeWithCompleted: fCompleted uploadRate: ul >= 0.1 && [fDefaults boolForKey: @"BadgeUploadRate"] ? [NSString stringForSpeedAbbrev: ul] : nil downloadRate: dl >= 0.1 && [fDefaults boolForKey: @"BadgeDownloadRate"] ? [NSString stringForSpeedAbbrev: dl] : nil]; } - (void) updateTorrentHistory { NSMutableArray * history = [NSMutableArray arrayWithCapacity: [fTorrents count]]; NSEnumerator * enumerator = [fTorrents objectEnumerator]; Torrent * torrent; while( ( torrent = [enumerator nextObject] ) ) { [history addObject: [NSDictionary dictionaryWithObjectsAndKeys: [torrent path], @"TorrentPath", [torrent getFolder], @"DownloadFolder", [torrent isActive] ? @"NO" : @"YES", @"Paused", nil]]; } [fDefaults setObject: history forKey: @"History"]; } - (int) numberOfRowsInTableView: (NSTableView *) t { return [fTorrents count]; } - (id) tableView: (NSTableView *) t objectValueForTableColumn: (NSTableColumn *) tableColumn row: (int) rowIndex { return nil; } - (void) tableView: (NSTableView *) t willDisplayCell: (id) cell forTableColumn: (NSTableColumn *) tableColumn row: (int) rowIndex { [cell setTorrent: [fTorrents objectAtIndex: rowIndex]]; if( OSX_VERSION >= 10.3 && [fWindow isKeyWindow] && rowIndex == [fTableView selectedRow] ) [cell setTextColor: [NSColor whiteColor]]; else [cell setTextColor: [NSColor blackColor]]; } - (BOOL) tableView: (NSTableView *) t acceptDrop: (id ) info row: (int) row dropOperation: (NSTableViewDropOperation) operation { [self application: NSApp openFiles: [[[info draggingPasteboard] propertyListForType: NSFilenamesPboardType] pathsMatchingExtensions: [NSArray arrayWithObject: @"torrent"]]]; return YES; } - (NSDragOperation) tableView: (NSTableView *) t validateDrop: (id ) info proposedRow: (int) row proposedDropOperation: (NSTableViewDropOperation) operation { NSPasteboard * pasteboard = [info draggingPasteboard]; if (![[pasteboard types] containsObject: NSFilenamesPboardType] || [[[pasteboard propertyListForType: NSFilenamesPboardType] pathsMatchingExtensions: [NSArray arrayWithObject: @"torrent"]] count] == 0) return NSDragOperationNone; [fTableView setDropRow: [fTorrents count] dropOperation: NSTableViewDropAbove]; return NSDragOperationGeneric; } - (void) tableViewSelectionDidChange: (NSNotification *) n { [self updateInfoPanel]; } - (NSToolbarItem *) toolbar: (NSToolbar *) t itemForItemIdentifier: (NSString *) ident willBeInsertedIntoToolbar: (BOOL) flag { NSToolbarItem * item = [[NSToolbarItem alloc] initWithItemIdentifier: ident]; if( [ident isEqualToString: TOOLBAR_OPEN] ) { [item setLabel: @"Open"]; [item setPaletteLabel: [item label]]; [item setToolTip: @"Open a torrent"]; [item setImage: [NSImage imageNamed: @"Open.png"]]; [item setTarget: self]; [item setAction: @selector( openShowSheet: )]; } else if( [ident isEqualToString: TOOLBAR_REMOVE] ) { [item setLabel: @"Remove"]; [item setPaletteLabel: [item label]]; [item setToolTip: @"Remove torrent from list"]; [item setImage: [NSImage imageNamed: @"Remove.png"]]; [item setTarget: self]; [item setAction: @selector( removeTorrent: )]; } else if( [ident isEqualToString: TOOLBAR_INFO] ) { [item setLabel: @"Info"]; [item setPaletteLabel: [item label]]; [item setToolTip: @"Information"]; [item setImage: [NSImage imageNamed: @"Info.png"]]; [item setTarget: self]; [item setAction: @selector( showInfo: )]; } else if( [ident isEqualToString: TOOLBAR_RESUME_ALL] ) { [item setLabel: @"Resume All"]; [item setPaletteLabel: [item label]]; [item setToolTip: @"Resume all torrents"]; [item setImage: [NSImage imageNamed: @"ResumeAll.png"]]; [item setTarget: self]; [item setAction: @selector( resumeAllTorrents: )]; } else if( [ident isEqualToString: TOOLBAR_PAUSE_ALL] ) { [item setLabel: @"Pause All"]; [item setPaletteLabel: [item label]]; [item setToolTip: @"Pause all torrents"]; [item setImage: [NSImage imageNamed: @"PauseAll.png"]]; [item setTarget: self]; [item setAction: @selector( stopAllTorrents: )]; } else { [item release]; return NULL; } return item; } - (NSArray *) toolbarAllowedItemIdentifiers: (NSToolbar *) t { return [NSArray arrayWithObjects: TOOLBAR_OPEN, TOOLBAR_REMOVE, TOOLBAR_RESUME_ALL, TOOLBAR_PAUSE_ALL, TOOLBAR_INFO, NSToolbarSeparatorItemIdentifier, NSToolbarSpaceItemIdentifier, NSToolbarFlexibleSpaceItemIdentifier, NSToolbarCustomizeToolbarItemIdentifier, NULL]; } - (NSArray *) toolbarDefaultItemIdentifiers: (NSToolbar *) t { return [NSArray arrayWithObjects: TOOLBAR_OPEN, TOOLBAR_REMOVE, NSToolbarSeparatorItemIdentifier, TOOLBAR_RESUME_ALL, TOOLBAR_PAUSE_ALL, NSToolbarFlexibleSpaceItemIdentifier, TOOLBAR_INFO, NULL]; } - (void) runCustomizationPalette: (id) sender { [fToolbar runCustomizationPalette:sender]; } - (void) showHideToolbar: (id) sender { [fWindow toggleToolbarShown:sender]; } - (BOOL)validateToolbarItem:(NSToolbarItem *)toolbarItem { SEL action = [toolbarItem action]; //enable remove item if (action == @selector(removeTorrent:)) return [fTableView selectedRow] >= 0; Torrent * torrent; NSEnumerator * enumerator; //enable resume all item if (action == @selector(resumeAllTorrents:)) { enumerator = [fTorrents objectEnumerator]; while( ( torrent = [enumerator nextObject] ) ) if( [torrent isPaused] ) return YES; return NO; } //enable pause all item if (action == @selector(stopAllTorrents:)) { enumerator = [fTorrents objectEnumerator]; while( ( torrent = [enumerator nextObject] ) ) if( [torrent isActive] ) return YES; return NO; } return YES; } - (BOOL)validateMenuItem:(NSMenuItem *)menuItem { SEL action = [menuItem action]; //disable menus if customize sheet is active if ([fToolbar customizationPaletteIsRunning]) return NO; //enable customize toolbar item if (action == @selector(showHideToolbar:)) { [menuItem setTitle: [fToolbar isVisible] ? @"Hide Toolbar" : @"Show Toolbar"]; return YES; } //enable show info if (action == @selector(showInfo:)) { [menuItem setTitle: [fInfoPanel isVisible] ? @"Hide Info" : @"Show Info"]; return YES; } Torrent * torrent; NSEnumerator * enumerator; //enable resume all item if (action == @selector(resumeAllTorrents:)) { enumerator = [fTorrents objectEnumerator]; while( ( torrent = [enumerator nextObject] ) ) if( [torrent isPaused] ) return YES; return NO; } //enable pause all item if (action == @selector(stopAllTorrents:)) { enumerator = [fTorrents objectEnumerator]; while( ( torrent = [enumerator nextObject] ) ) if( [torrent isActive] ) return YES; return NO; } int row = [fTableView selectedRow]; torrent = ( row < 0 ) ? nil : [fTorrents objectAtIndex: row]; if (action == @selector(revealFromMenu:)) { return ( torrent != nil ); } //enable remove items if (action == @selector(removeTorrent:) || action == @selector(removeTorrentDeleteFile:) || action == @selector(removeTorrentDeleteData:) || action == @selector(removeTorrentDeleteBoth:)) { //append or remove ellipsis when needed if( torrent && [torrent isActive] && [fDefaults boolForKey: @"CheckRemove"] ) { if (![[menuItem title] hasSuffix:NS_ELLIPSIS]) [menuItem setTitle:[[menuItem title] stringByAppendingString:NS_ELLIPSIS]]; } else { if ([[menuItem title] hasSuffix:NS_ELLIPSIS]) [menuItem setTitle:[[menuItem title] substringToIndex:[[menuItem title] length]-[NS_ELLIPSIS length]]]; } return ( torrent != nil ); } //enable and change pause / remove item if( action == @selector(resumeTorrent:) || action == @selector(stopTorrent:) ) { if( !torrent || [torrent isActive] ) { [menuItem setTitle: @"Pause"]; [menuItem setAction: @selector( stopTorrent: )]; } else { [menuItem setTitle: @"Resume"]; [menuItem setAction: @selector( resumeTorrent: )]; } return ( torrent != nil ); } return YES; } - (void) sleepCallBack: (natural_t) messageType argument: (void *) messageArgument { NSEnumerator * enumerator;; Torrent * torrent; switch( messageType ) { case kIOMessageSystemWillSleep: /* Close all connections before going to sleep and remember we should resume when we wake up */ enumerator = [fTorrents objectEnumerator]; while( ( torrent = [enumerator nextObject] ) ) { [torrent sleep]; } /* Wait for torrents to stop (5 seconds timeout) */ NSDate * start = [NSDate date]; enumerator = [fTorrents objectEnumerator]; while( ( torrent = [enumerator nextObject] ) ) { while( [[NSDate date] timeIntervalSinceDate: start] < 5 && ![torrent isPaused] ) { usleep( 100000 ); [torrent update]; } } IOAllowPowerChange( fRootPort, (long) messageArgument ); break; case kIOMessageCanSystemSleep: /* Prevent idle sleep unless all paused */ if (fDownloading > 0 || fSeeding > 0) IOCancelPowerChange( fRootPort, (long) messageArgument ); else IOAllowPowerChange( fRootPort, (long) messageArgument ); break; case kIOMessageSystemHasPoweredOn: /* Resume download after we wake up */ enumerator = [fTorrents objectEnumerator]; while( ( torrent = [enumerator nextObject] ) ) { [torrent wakeUp]; } break; } } - (NSRect) windowWillUseStandardFrame: (NSWindow *) w defaultFrame: (NSRect) defaultFrame { NSRect rectWin, rectView; float foo; rectWin = [fWindow frame]; rectView = [fScrollView frame]; foo = 10.0 - rectView.size.height + MAX( 1U, [fTorrents count] ) * ( [fTableView rowHeight] + [fTableView intercellSpacing].height ); rectWin.size.height += foo; rectWin.origin.y -= foo; return rectWin; } - (void) showMainWindow: (id) sender { [fWindow makeKeyAndOrderFront: nil]; } - (void) linkHomepage: (id) sender { [[NSWorkspace sharedWorkspace] openURL: [NSURL URLWithString: WEBSITE_URL]]; } - (void) linkForums: (id) sender { [[NSWorkspace sharedWorkspace] openURL: [NSURL URLWithString: FORUM_URL]]; } - (void) notifyGrowl: (NSString * ) file { NSString * growlScript; NSAppleScript * appleScript; NSDictionary * error; if( !fHasGrowl ) return; growlScript = [NSString stringWithFormat: @"tell application \"System Events\"\n" " if exists application process \"GrowlHelperApp\" then\n" " tell application \"GrowlHelperApp\"\n " " notify with name \"Download Complete\"" " title \"Download Complete\"" " description \"%@\"" " application name \"Transmission\"\n" " end tell\n" " end if\n" "end tell", file]; appleScript = [[NSAppleScript alloc] initWithSource: growlScript]; if( ![appleScript executeAndReturnError: &error] ) { printf( "Growl notify failed\n" ); } [appleScript release]; } - (void) growlRegister: (id) sender { NSString * growlScript; NSAppleScript * appleScript; NSDictionary * error; if( !fHasGrowl ) return; growlScript = [NSString stringWithFormat: @"tell application \"System Events\"\n" " if exists application process \"GrowlHelperApp\" then\n" " tell application \"GrowlHelperApp\"\n" " register as application \"Transmission\" " " all notifications {\"Download Complete\"}" " default notifications {\"Download Complete\"}" " icon of application \"Transmission\"\n" " end tell\n" " end if\n" "end tell"]; appleScript = [[NSAppleScript alloc] initWithSource: growlScript]; if( ![appleScript executeAndReturnError: &error] ) { printf( "Growl registration failed\n" ); } [appleScript release]; } - (void) revealFromMenu: (id) sender { Torrent * torrent; torrent = [fTorrents objectAtIndex: [fTableView selectedRow]]; [torrent reveal]; } - (void) checkForUpdate: (id) sender { [self checkForUpdateAuto: NO]; } - (void) checkForUpdateTimer: (NSTimer *) timer { NSString * check = [fDefaults stringForKey: @"VersionCheck"]; if( [check isEqualToString: @"Never"] ) return; NSTimeInterval interval; if( [check isEqualToString: @"Daily"] ) interval = 24 * 3600; else if( [check isEqualToString: @"Weekly"] ) interval = 7 * 24 * 3600; else return; id lastObject = [fDefaults objectForKey: @"VersionCheckLast"]; NSDate * lastDate = [lastObject isKindOfClass: [NSDate class]] ? lastObject : nil; if( lastDate ) { NSTimeInterval actualInterval = [[NSDate date] timeIntervalSinceDate: lastDate]; if( actualInterval > 0 && actualInterval < interval ) { return; } } [self checkForUpdateAuto: YES]; [fDefaults setObject: [NSDate date] forKey: @"VersionCheckLast"]; } - (void) checkForUpdateAuto: (BOOL) automatic { fCheckIsAutomatic = automatic; [[NSURL URLWithString: VERSION_PLIST_URL] loadResourceDataNotifyingClient: self usingCache: NO]; } - (void) URLResourceDidFinishLoading: (NSURL *) sender { NSDictionary * dict = [NSPropertyListSerialization propertyListFromData: [sender resourceDataUsingCache: NO] mutabilityOption: NSPropertyListImmutable format: nil errorDescription: nil]; //check if plist was actually found and contains a version NSString * webVersion = nil; if (dict) webVersion = [dict objectForKey: @"Version"]; if (!webVersion) { if (!fCheckIsAutomatic) { if( OSX_VERSION >= 10.3 ) { NSAlert * dialog = [[NSAlert alloc] init]; [dialog addButtonWithTitle: @"OK"]; [dialog setMessageText: @"Error checking for updates."]; [dialog setInformativeText: @"Transmission was not able to check the latest version available."]; [dialog setAlertStyle: NSInformationalAlertStyle]; [dialog runModal]; [dialog release]; } else /* XXX */; } return; } NSString * currentVersion = [[[NSBundle mainBundle] infoDictionary] objectForKey: (NSString *)kCFBundleVersionKey]; NSEnumerator * webEnum = [[webVersion componentsSeparatedByString: @"."] objectEnumerator], * currentEnum = [[currentVersion componentsSeparatedByString: @"."] objectEnumerator]; NSString * webSub, * currentSub; BOOL webGreater = NO; NSComparisonResult result; while ((webSub = [webEnum nextObject])) { if (!(currentSub = [currentEnum nextObject])) { webGreater = YES; break; } if( OSX_VERSION >= 10.3 ) { result = [currentSub compare: webSub options: NSNumericSearch]; if (result != NSOrderedSame) { if (result == NSOrderedAscending) webGreater = YES; break; } } else /* XXX */; } if (webGreater) { if( OSX_VERSION >= 10.3 ) { NSAlert * dialog = [[NSAlert alloc] init]; [dialog addButtonWithTitle: @"Go to Website"]; [dialog addButtonWithTitle:@"Cancel"]; [dialog setMessageText: @"New version is available!"]; [dialog setInformativeText: [NSString stringWithFormat: @"A newer version (%@) is available for download from the Transmission website.", webVersion]]; [dialog setAlertStyle: NSInformationalAlertStyle]; if ([dialog runModal] == NSAlertFirstButtonReturn) [self linkHomepage: nil]; [dialog release]; } else /* XXX */; } else if (!fCheckIsAutomatic) { if( OSX_VERSION >= 10.3 ) { NSAlert * dialog = [[NSAlert alloc] init]; [dialog addButtonWithTitle: @"OK"]; [dialog setMessageText: @"No new versions are available."]; [dialog setInformativeText: [NSString stringWithFormat: @"You are running the most current version of Transmission (%@).", currentVersion]]; [dialog setAlertStyle: NSInformationalAlertStyle]; [dialog runModal]; [dialog release]; } else /* XXX */; } else; } @end