/****************************************************************************** * $Id$ * * Copyright (c) 2005-2008 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 "PrefsController.h" #import "BlocklistDownloader.h" #import "NSApplicationAdditions.h" #import "NSStringAdditions.h" #import "UKKQueue.h" #define DOWNLOAD_FOLDER 0 #define DOWNLOAD_TORRENT 2 #define UPDATE_SECONDS 86400 #define TOOLBAR_GENERAL @"TOOLBAR_GENERAL" #define TOOLBAR_TRANSFERS @"TOOLBAR_TRANSFERS" #define TOOLBAR_BANDWIDTH @"TOOLBAR_BANDWIDTH" #define TOOLBAR_ADVANCED @"TOOLBAR_ADVANCED" @interface PrefsController (Private) - (void) setPrefView: (id) sender; - (void) folderSheetClosed: (NSOpenPanel *) openPanel returnCode: (int) code contextInfo: (void *) info; - (void) incompleteFolderSheetClosed: (NSOpenPanel *) openPanel returnCode: (int) code contextInfo: (void *) info; - (void) importFolderSheetClosed: (NSOpenPanel *) openPanel returnCode: (int) code contextInfo: (void *) info; @end @implementation PrefsController - (id) initWithHandle: (tr_handle *) handle { if ((self = [super initWithWindowNibName: @"PrefsWindow"])) { fDefaults = [NSUserDefaults standardUserDefaults]; fHandle = handle; //checks for old version speeds of -1 if ([fDefaults integerForKey: @"UploadLimit"] < 0) { [fDefaults setInteger: 20 forKey: @"UploadLimit"]; [fDefaults setBool: NO forKey: @"CheckUpload"]; } if ([fDefaults integerForKey: @"DownloadLimit"] < 0) { [fDefaults setInteger: 20 forKey: @"DownloadLimit"]; [fDefaults setBool: NO forKey: @"CheckDownload"]; } //check for old version download location NSString * choice; if ((choice = [fDefaults stringForKey: @"DownloadChoice"])) { [fDefaults setBool: [choice isEqualToString: @"Constant"] forKey: @"DownloadLocationConstant"]; [fDefaults setBool: YES forKey: @"DownloadAsk"]; [fDefaults removeObjectForKey: @"DownloadChoice"]; } //set check for update to right value [self setCheckForUpdate: nil]; //set auto import NSString * autoPath; if ([fDefaults boolForKey: @"AutoImport"] && (autoPath = [fDefaults stringForKey: @"AutoImportDirectory"])) [[UKKQueue sharedFileWatcher] addPath: [autoPath stringByExpandingTildeInPath]]; //set encryption [self setEncryptionMode: nil]; //actually set bandwidth limits [self applySpeedSettings: nil]; } return self; } - (void) dealloc { if (fPortStatusTimer) [fPortStatusTimer invalidate]; if (fPortChecker) { [fPortChecker cancelProbe]; [fPortChecker release]; } [super dealloc]; } - (void) awakeFromNib { fHasLoaded = YES; NSToolbar * toolbar = [[NSToolbar alloc] initWithIdentifier: @"Preferences Toolbar"]; [toolbar setDelegate: self]; [toolbar setAllowsUserCustomization: NO]; [toolbar setDisplayMode: NSToolbarDisplayModeIconAndLabel]; [toolbar setSizeMode: NSToolbarSizeModeRegular]; [toolbar setSelectedItemIdentifier: TOOLBAR_GENERAL]; [[self window] setToolbar: toolbar]; [toolbar release]; [self setPrefView: nil]; //set download folder [fFolderPopUp selectItemAtIndex: [fDefaults boolForKey: @"DownloadLocationConstant"] ? DOWNLOAD_FOLDER : DOWNLOAD_TORRENT]; //set stop ratio [self updateRatioStopField]; //set limits [self updateLimitFields]; //set speed limit [fSpeedLimitUploadField setIntValue: [fDefaults integerForKey: @"SpeedLimitUploadLimit"]]; [fSpeedLimitDownloadField setIntValue: [fDefaults integerForKey: @"SpeedLimitDownloadLimit"]]; //set port [fPortField setIntValue: [fDefaults integerForKey: @"BindPort"]]; fNatStatus = -1; [self updatePortStatus]; fPortStatusTimer = [NSTimer scheduledTimerWithTimeInterval: 5.0 target: self selector: @selector(updatePortStatus) userInfo: nil repeats: YES]; //set peer connections [fPeersGlobalField setIntValue: [fDefaults integerForKey: @"PeersTotal"]]; [fPeersTorrentField setIntValue: [fDefaults integerForKey: @"PeersTorrent"]]; //set queue values [fQueueDownloadField setIntValue: [fDefaults integerForKey: @"QueueDownloadNumber"]]; [fQueueSeedField setIntValue: [fDefaults integerForKey: @"QueueSeedNumber"]]; [fStalledField setIntValue: [fDefaults integerForKey: @"StalledMinutes"]]; //set blocklist [self updateBlocklistFields]; } - (void) setUpdater: (SUUpdater *) updater { fUpdater = updater; } - (NSToolbarItem *) toolbar: (NSToolbar *) toolbar itemForItemIdentifier: (NSString *) ident willBeInsertedIntoToolbar: (BOOL) flag { NSToolbarItem * item; item = [[NSToolbarItem alloc] initWithItemIdentifier: ident]; if ([ident isEqualToString: TOOLBAR_GENERAL]) { [item setLabel: NSLocalizedString(@"General", "Preferences -> General toolbar item title")]; [item setImage: [NSImage imageNamed: [NSApp isOnLeopardOrBetter] ? NSImageNamePreferencesGeneral : @"Preferences.png"]]; [item setTarget: self]; [item setAction: @selector(setPrefView:)]; [item setAutovalidates: NO]; } else if ([ident isEqualToString: TOOLBAR_TRANSFERS]) { [item setLabel: NSLocalizedString(@"Transfers", "Preferences -> Transfers toolbar item title")]; [item setImage: [NSImage imageNamed: @"Transfers.png"]]; [item setTarget: self]; [item setAction: @selector(setPrefView:)]; [item setAutovalidates: NO]; } else if ([ident isEqualToString: TOOLBAR_BANDWIDTH]) { [item setLabel: NSLocalizedString(@"Bandwidth", "Preferences -> Bandwidth toolbar item title")]; [item setImage: [NSImage imageNamed: @"Bandwidth.png"]]; [item setTarget: self]; [item setAction: @selector(setPrefView:)]; [item setAutovalidates: NO]; } else if ([ident isEqualToString: TOOLBAR_ADVANCED]) { [item setLabel: NSLocalizedString(@"Advanced", "Preferences -> Advanced toolbar item title")]; [item setImage: [NSImage imageNamed: [NSApp isOnLeopardOrBetter] ? NSImageNameAdvanced : @"Advanced.png"]]; [item setTarget: self]; [item setAction: @selector(setPrefView:)]; [item setAutovalidates: NO]; } else { [item release]; return nil; } return [item autorelease]; } - (NSArray *) toolbarSelectableItemIdentifiers: (NSToolbar *) toolbar { return [self toolbarDefaultItemIdentifiers: toolbar]; } - (NSArray *) toolbarDefaultItemIdentifiers: (NSToolbar *) toolbar { return [self toolbarAllowedItemIdentifiers: toolbar]; } - (NSArray *) toolbarAllowedItemIdentifiers: (NSToolbar *) toolbar { return [NSArray arrayWithObjects: TOOLBAR_GENERAL, TOOLBAR_TRANSFERS, TOOLBAR_BANDWIDTH, TOOLBAR_ADVANCED, nil]; } //used by ipc - (void) updatePortField { [fPortField setIntValue: [fDefaults integerForKey: @"BindPort"]]; } - (void) setPort: (id) sender { int port = [sender intValue]; [fDefaults setInteger: port forKey: @"BindPort"]; tr_setBindPort(fHandle, port); fPublicPort = -1; [self updatePortStatus]; } - (void) setNat: (id) sender { tr_natTraversalEnable(fHandle, [fDefaults boolForKey: @"NatTraversal"]); fNatStatus = -1; [self updatePortStatus]; } - (void) updatePortStatus { const tr_handle_status * stat = tr_handleStatus(fHandle); if (fNatStatus != stat->natTraversalStatus || fPublicPort != stat->publicPort) { fNatStatus = stat->natTraversalStatus; fPublicPort = stat->publicPort; [fPortStatusField setStringValue: [NSLocalizedString(@"Checking port status", "Preferences -> Advanced -> port status") stringByAppendingEllipsis]]; [fPortStatusImage setImage: nil]; [fPortStatusProgress startAnimation: self]; if (fPortChecker) { [fPortChecker cancelProbe]; [fPortChecker release]; } fPortChecker = [[PortChecker alloc] initForPort: fPublicPort withDelegate: self]; } } - (void) portCheckerDidFinishProbing: (PortChecker *) portChecker { [fPortStatusProgress stopAnimation: self]; switch ([fPortChecker status]) { case PORT_STATUS_OPEN: [fPortStatusField setStringValue: NSLocalizedString(@"Port is open", "Preferences -> Advanced -> port status")]; [fPortStatusImage setImage: [NSImage imageNamed: @"GreenDot.png"]]; break; case PORT_STATUS_CLOSED: [fPortStatusField setStringValue: NSLocalizedString(@"Port is closed", "Preferences -> Advanced -> port status")]; [fPortStatusImage setImage: [NSImage imageNamed: @"RedDot.png"]]; break; case PORT_STATUS_STEALTH: [fPortStatusField setStringValue: NSLocalizedString(@"Port is stealth", "Preferences -> Advanced -> port status")]; [fPortStatusImage setImage: [NSImage imageNamed: @"RedDot.png"]]; break; case PORT_STATUS_ERROR: [fPortStatusField setStringValue: NSLocalizedString(@"Unable to check port status", "Preferences -> Advanced -> port status")]; [fPortStatusImage setImage: [NSImage imageNamed: @"YellowDot.png"]]; break; } [fPortChecker release]; fPortChecker = nil; } - (NSArray *) sounds { NSMutableArray * sounds = [NSMutableArray array]; //until Apple can fix soundNamed to not crash on invalid sound files, don't use custom sounds NSArray * directories = [NSArray arrayWithObjects: @"/System/Library/Sounds", @"/Library/Sounds", /*[NSHomeDirectory() stringByAppendingPathComponent: @"Library/Sounds"],*/ nil]; BOOL isDirectory; NSEnumerator * soundEnumerator; NSString * sound; NSString * directory; NSEnumerator * enumerator = [directories objectEnumerator]; while ((directory = [enumerator nextObject])) if ([[NSFileManager defaultManager] fileExistsAtPath: directory isDirectory: &isDirectory] && isDirectory) { soundEnumerator = [[[NSFileManager defaultManager] directoryContentsAtPath: directory] objectEnumerator]; while ((sound = [soundEnumerator nextObject])) { sound = [sound stringByDeletingPathExtension]; if ([NSSound soundNamed: sound]) [sounds addObject: sound]; } } return sounds; } - (void) setSound: (id) sender { //play sound when selecting NSSound * sound; if ((sound = [NSSound soundNamed: [sender titleOfSelectedItem]])) [sound play]; } - (void) setPeersGlobal: (id) sender { int count = [sender intValue]; [fDefaults setInteger: count forKey: @"PeersTotal"]; tr_setGlobalPeerLimit(fHandle, count); } - (void) setPeersTorrent: (id) sender { int count = [sender intValue]; [fDefaults setInteger: count forKey: @"PeersTorrent"]; } - (void) setPEX: (id) sender { tr_setPexEnabled(fHandle, [fDefaults boolForKey: @"PEXGlobal"]); } - (void) setEncryptionMode: (id) sender { tr_setEncryptionMode(fHandle, [fDefaults boolForKey: @"EncryptionPrefer"] ? ([fDefaults boolForKey: @"EncryptionRequire"] ? TR_ENCRYPTION_REQUIRED : TR_ENCRYPTION_PREFERRED) : TR_PLAINTEXT_PREFERRED); } - (void) setBlocklistEnabled: (id) sender { BOOL enable = [sender state] == NSOnState; [fDefaults setBool: enable forKey: @"Blocklist"]; tr_blocklistSetEnabled(fHandle, enable); } - (void) updateBlocklist: (id) sender { [BlocklistDownloader downloadWithPrefsController: self withHandle: fHandle]; } - (void) updateBlocklistFields { BOOL exists = tr_blocklistExists(fHandle); [fBlocklistMessageField setStringValue: exists ? NSLocalizedString(@"Blocklist is loaded", "Prefs -> blocklist -> message") : NSLocalizedString(@"A blocklist must first be downloaded", "Prefs -> blocklist -> message")]; [fBlocklistEnableCheck setEnabled: exists]; [fBlocklistEnableCheck setState: exists && [fDefaults boolForKey: @"Blocklist"]]; } - (void) applySpeedSettings: (id) sender { if ([fDefaults boolForKey: @"SpeedLimit"]) { tr_setUseGlobalSpeedLimit(fHandle, TR_UP, 1); tr_setGlobalSpeedLimit(fHandle, TR_UP, [fDefaults integerForKey: @"SpeedLimitUploadLimit"]); tr_setUseGlobalSpeedLimit(fHandle, TR_DOWN, 1); tr_setGlobalSpeedLimit(fHandle, TR_DOWN, [fDefaults integerForKey: @"SpeedLimitDownloadLimit"]); } else { tr_setUseGlobalSpeedLimit(fHandle, TR_UP, [fDefaults boolForKey: @"CheckUpload"]); tr_setGlobalSpeedLimit(fHandle, TR_UP, [fDefaults integerForKey: @"UploadLimit"]); tr_setUseGlobalSpeedLimit(fHandle, TR_DOWN, [fDefaults boolForKey: @"CheckDownload"]); tr_setGlobalSpeedLimit(fHandle, TR_DOWN, [fDefaults integerForKey: @"DownloadLimit"]); } } - (void) applyRatioSetting: (id) sender { [[NSNotificationCenter defaultCenter] postNotificationName: @"UpdateUI" object: nil]; } - (void) updateRatioStopField { if (!fHasLoaded) return; [fRatioStopField setFloatValue: [fDefaults floatForKey: @"RatioLimit"]]; [self applyRatioSetting: nil]; } - (void) setRatioStop: (id) sender { [fDefaults setFloat: [sender floatValue] forKey: @"RatioLimit"]; [self applyRatioSetting: nil]; } - (void) updateLimitFields { if (!fHasLoaded) return; [fUploadField setIntValue: [fDefaults integerForKey: @"UploadLimit"]]; [fDownloadField setIntValue: [fDefaults integerForKey: @"DownloadLimit"]]; } - (void) setGlobalLimit: (id) sender { [fDefaults setInteger: [sender intValue] forKey: sender == fUploadField ? @"UploadLimit" : @"DownloadLimit"]; [self applySpeedSettings: self]; } - (void) setSpeedLimit: (id) sender { [fDefaults setInteger: [sender intValue] forKey: sender == fSpeedLimitUploadField ? @"SpeedLimitUploadLimit" : @"SpeedLimitDownloadLimit"]; [self applySpeedSettings: self]; } - (void) setAutoSpeedLimit: (id) sender { [[NSNotificationCenter defaultCenter] postNotificationName: @"AutoSpeedLimitChange" object: self]; } - (BOOL) control: (NSControl *) control textShouldBeginEditing: (NSText *) fieldEditor { [fInitialString release]; fInitialString = [[control stringValue] retain]; return YES; } - (BOOL) control: (NSControl *) control didFailToFormatString: (NSString *) string errorDescription: (NSString *) error { NSBeep(); if (fInitialString) { [control setStringValue: fInitialString]; [fInitialString release]; fInitialString = nil; } return NO; } - (void) setBadge: (id) sender { [[NSNotificationCenter defaultCenter] postNotificationName: @"DockBadgeChange" object: self]; } - (void) resetWarnings: (id) sender { [fDefaults setBool: YES forKey: @"WarningDuplicate"]; [fDefaults setBool: YES forKey: @"WarningRemainingSpace"]; [fDefaults setBool: YES forKey: @"WarningFolderDataSameName"]; } - (void) setCheckForUpdate: (id) sender { NSTimeInterval seconds = [fDefaults boolForKey: @"CheckForUpdates"] ? UPDATE_SECONDS : 0; [fDefaults setInteger: seconds forKey: @"SUScheduledCheckInterval"]; if (fUpdater) [fUpdater scheduleCheckWithInterval: seconds]; } - (void) setQueue: (id) sender { [[NSNotificationCenter defaultCenter] postNotificationName: @"UpdateQueue" object: self]; } - (void) setQueueNumber: (id) sender { [fDefaults setInteger: [sender intValue] forKey: sender == fQueueDownloadField ? @"QueueDownloadNumber" : @"QueueSeedNumber"]; [self setQueue: nil]; } - (void) setStalled: (id) sender { [[NSNotificationCenter defaultCenter] postNotificationName: @"UpdateQueue" object: self]; } - (void) setStalledMinutes: (id) sender { [fDefaults setInteger: [sender intValue] forKey: @"StalledMinutes"]; [self setStalled: nil]; } - (void) setDownloadLocation: (id) sender { [fDefaults setBool: [fFolderPopUp indexOfSelectedItem] == DOWNLOAD_FOLDER forKey: @"DownloadLocationConstant"]; } - (void) folderSheetShow: (id) sender { NSOpenPanel * panel = [NSOpenPanel openPanel]; [panel setPrompt: NSLocalizedString(@"Select", "Preferences -> Open panel prompt")]; [panel setAllowsMultipleSelection: NO]; [panel setCanChooseFiles: NO]; [panel setCanChooseDirectories: YES]; [panel setCanCreateDirectories: YES]; [panel beginSheetForDirectory: nil file: nil types: nil modalForWindow: [self window] modalDelegate: self didEndSelector: @selector(folderSheetClosed:returnCode:contextInfo:) contextInfo: nil]; } - (void) incompleteFolderSheetShow: (id) sender { NSOpenPanel * panel = [NSOpenPanel openPanel]; [panel setPrompt: NSLocalizedString(@"Select", "Preferences -> Open panel prompt")]; [panel setAllowsMultipleSelection: NO]; [panel setCanChooseFiles: NO]; [panel setCanChooseDirectories: YES]; [panel setCanCreateDirectories: YES]; [panel beginSheetForDirectory: nil file: nil types: nil modalForWindow: [self window] modalDelegate: self didEndSelector: @selector(incompleteFolderSheetClosed:returnCode:contextInfo:) contextInfo: nil]; } - (void) setAutoImport: (id) sender { NSString * path; if ((path = [fDefaults stringForKey: @"AutoImportDirectory"])) { path = [path stringByExpandingTildeInPath]; if ([fDefaults boolForKey: @"AutoImport"]) [[UKKQueue sharedFileWatcher] addPath: path]; else [[UKKQueue sharedFileWatcher] removePathFromQueue: path]; [[NSNotificationCenter defaultCenter] postNotificationName: @"AutoImportSettingChange" object: self]; } else [self importFolderSheetShow: nil]; } - (void) importFolderSheetShow: (id) sender { NSOpenPanel * panel = [NSOpenPanel openPanel]; [panel setPrompt: NSLocalizedString(@"Select", "Preferences -> Open panel prompt")]; [panel setAllowsMultipleSelection: NO]; [panel setCanChooseFiles: NO]; [panel setCanChooseDirectories: YES]; [panel setCanCreateDirectories: YES]; [panel beginSheetForDirectory: nil file: nil types: nil modalForWindow: [self window] modalDelegate: self didEndSelector: @selector(importFolderSheetClosed:returnCode:contextInfo:) contextInfo: nil]; } - (void) setAutoSize: (id) sender { [[NSNotificationCenter defaultCenter] postNotificationName: @"AutoSizeSettingChange" object: self]; } - (void) helpForNetwork: (id) sender { [[NSHelpManager sharedHelpManager] openHelpAnchor: @"AdvancedPrefs" inBook: [[NSBundle mainBundle] objectForInfoDictionaryKey: @"CFBundleHelpBookName"]]; } @end @implementation PrefsController (Private) - (void) setPrefView: (id) sender { NSView * view = fGeneralView; if (sender) { NSString * identifier = [sender itemIdentifier]; if ([identifier isEqualToString: TOOLBAR_TRANSFERS]) view = fTransfersView; else if ([identifier isEqualToString: TOOLBAR_BANDWIDTH]) view = fBandwidthView; else if ([identifier isEqualToString: TOOLBAR_ADVANCED]) view = fAdvancedView; else; } NSWindow * window = [self window]; if ([window contentView] == view) return; NSRect windowRect = [window frame]; float difference = ([view frame].size.height - [[window contentView] frame].size.height) * [window userSpaceScaleFactor]; windowRect.origin.y -= difference; windowRect.size.height += difference; [view setHidden: YES]; [window setContentView: view]; [window setFrame: windowRect display: YES animate: YES]; [view setHidden: NO]; //set title label if (sender) [window setTitle: [sender label]]; else { NSToolbar * toolbar = [window toolbar]; NSString * itemIdentifier = [toolbar selectedItemIdentifier]; NSEnumerator * enumerator = [[toolbar items] objectEnumerator]; NSToolbarItem * item; while ((item = [enumerator nextObject])) if ([[item itemIdentifier] isEqualToString: itemIdentifier]) { [window setTitle: [item label]]; break; } } //for advanced view make sure progress indicator hides itself if (view == fAdvancedView && [fPortStatusImage image]) [fPortStatusProgress setDisplayedWhenStopped: NO]; } - (void) folderSheetClosed: (NSOpenPanel *) openPanel returnCode: (int) code contextInfo: (void *) info { if (code == NSOKButton) { [fFolderPopUp selectItemAtIndex: DOWNLOAD_FOLDER]; [fDefaults setObject: [[openPanel filenames] objectAtIndex: 0] forKey: @"DownloadFolder"]; [fDefaults setObject: @"Constant" forKey: @"DownloadChoice"]; } else { //reset if cancelled [fFolderPopUp selectItemAtIndex: [fDefaults boolForKey: @"DownloadLocationConstant"] ? DOWNLOAD_FOLDER : DOWNLOAD_TORRENT]; } } - (void) incompleteFolderSheetClosed: (NSOpenPanel *) openPanel returnCode: (int) code contextInfo: (void *) info { if (code == NSOKButton) [fDefaults setObject: [[openPanel filenames] objectAtIndex: 0] forKey: @"IncompleteDownloadFolder"]; [fIncompleteFolderPopUp selectItemAtIndex: 0]; } - (void) importFolderSheetClosed: (NSOpenPanel *) openPanel returnCode: (int) code contextInfo: (void *) info { NSString * path = [fDefaults stringForKey: @"AutoImportDirectory"]; if (code == NSOKButton) { UKKQueue * sharedQueue = [UKKQueue sharedFileWatcher]; if (path) [sharedQueue removePathFromQueue: [path stringByExpandingTildeInPath]]; path = [[openPanel filenames] objectAtIndex: 0]; [fDefaults setObject: path forKey: @"AutoImportDirectory"]; [sharedQueue addPath: [path stringByExpandingTildeInPath]]; [[NSNotificationCenter defaultCenter] postNotificationName: @"AutoImportSettingChange" object: self]; } else if (!path) [fDefaults setBool: NO forKey: @"AutoImport"]; [fImportFolderPopUp selectItemAtIndex: 0]; } @end