// This file Copyright © 2008-2022 Transmission authors and contributors. // It may be used under the MIT (SPDX: MIT) license. // License text can be found in the licenses/ folder. #import "AddWindowController.h" #import "Controller.h" #import "ExpandedPathToIconTransformer.h" #import "FileOutlineController.h" #import "GroupsController.h" #import "NSStringAdditions.h" #import "Torrent.h" #define UPDATE_SECONDS 1.0 #define POPUP_PRIORITY_HIGH 0 #define POPUP_PRIORITY_NORMAL 1 #define POPUP_PRIORITY_LOW 2 @interface AddWindowController () @property(nonatomic) IBOutlet NSImageView* fIconView; @property(nonatomic) IBOutlet NSImageView* fLocationImageView; @property(nonatomic) IBOutlet NSTextField* fNameField; @property(nonatomic) IBOutlet NSTextField* fStatusField; @property(nonatomic) IBOutlet NSTextField* fLocationField; @property(nonatomic) IBOutlet NSButton* fStartCheck; @property(nonatomic) IBOutlet NSButton* fDeleteCheck; @property(nonatomic) IBOutlet NSPopUpButton* fGroupPopUp; @property(nonatomic) IBOutlet NSPopUpButton* fPriorityPopUp; @property(nonatomic) IBOutlet NSProgressIndicator* fVerifyIndicator; @property(nonatomic) IBOutlet NSTextField* fFileFilterField; @property(nonatomic) IBOutlet NSButton* fCheckAllButton; @property(nonatomic) IBOutlet NSButton* fUncheckAllButton; @property(nonatomic) IBOutlet FileOutlineController* fFileController; @property(nonatomic) IBOutlet NSScrollView* fFileScrollView; @property(nonatomic, readonly) Controller* fController; @property(nonatomic, copy) NSString* fDestination; @property(nonatomic, readonly) NSString* fTorrentFile; @property(nonatomic) BOOL fLockDestination; @property(nonatomic, readonly) BOOL fDeleteTorrentEnableInitially; @property(nonatomic, readonly) BOOL fCanToggleDelete; @property(nonatomic) NSInteger fGroupValue; @property(nonatomic, weak) NSTimer* fTimer; @property(nonatomic) TorrentDeterminationType fGroupValueDetermination; - (void)updateFiles; - (void)confirmAdd; - (void)setDestinationPath:(NSString*)destination determinationType:(TorrentDeterminationType)determinationType; - (void)setGroupsMenu; - (void)changeGroupValue:(id)sender; @end @implementation AddWindowController - (instancetype)initWithTorrent:(Torrent*)torrent destination:(NSString*)path lockDestination:(BOOL)lockDestination controller:(Controller*)controller torrentFile:(NSString*)torrentFile deleteTorrentCheckEnableInitially:(BOOL)deleteTorrent canToggleDelete:(BOOL)canToggleDelete { if ((self = [super initWithWindowNibName:@"AddWindow"])) { _torrent = torrent; _fDestination = path.stringByExpandingTildeInPath; _fLockDestination = lockDestination; _fController = controller; _fTorrentFile = torrentFile.stringByExpandingTildeInPath; _fDeleteTorrentEnableInitially = deleteTorrent; _fCanToggleDelete = canToggleDelete; _fGroupValue = torrent.groupValue; _fGroupValueDetermination = TorrentDeterminationAutomatic; _fVerifyIndicator.usesThreadedAnimation = YES; } return self; } - (void)awakeFromNib { [NSNotificationCenter.defaultCenter addObserver:self selector:@selector(updateCheckButtons:) name:@"TorrentFileCheckChange" object:self.torrent]; [NSNotificationCenter.defaultCenter addObserver:self selector:@selector(updateGroupMenu:) name:@"UpdateGroups" object:nil]; self.fFileController.torrent = self.torrent; NSString* name = self.torrent.name; self.window.title = name; self.fNameField.stringValue = name; self.fNameField.toolTip = name; self.fIconView.image = self.torrent.icon; if (!self.torrent.folder) { self.fFileFilterField.hidden = YES; self.fCheckAllButton.hidden = YES; self.fUncheckAllButton.hidden = YES; NSRect scrollFrame = self.fFileScrollView.frame; CGFloat const diff = NSMinY(self.fFileScrollView.frame) - NSMinY(self.fFileFilterField.frame); scrollFrame.origin.y -= diff; scrollFrame.size.height += diff; self.fFileScrollView.frame = scrollFrame; } else { [self updateCheckButtons:nil]; } [self setGroupsMenu]; [self.fGroupPopUp selectItemWithTag:self.fGroupValue]; NSInteger priorityIndex; switch (self.torrent.priority) { case TR_PRI_HIGH: priorityIndex = POPUP_PRIORITY_HIGH; break; case TR_PRI_NORMAL: priorityIndex = POPUP_PRIORITY_NORMAL; break; case TR_PRI_LOW: priorityIndex = POPUP_PRIORITY_LOW; break; default: NSAssert1(NO, @"Unknown priority for adding torrent: %d", self.torrent.priority); priorityIndex = POPUP_PRIORITY_NORMAL; } [self.fPriorityPopUp selectItemAtIndex:priorityIndex]; self.fStartCheck.state = [NSUserDefaults.standardUserDefaults boolForKey:@"AutoStartDownload"] ? NSControlStateValueOn : NSControlStateValueOff; self.fDeleteCheck.state = self.fDeleteTorrentEnableInitially ? NSControlStateValueOn : NSControlStateValueOff; self.fDeleteCheck.enabled = self.fCanToggleDelete; if (self.fDestination) { [self setDestinationPath:self.fDestination determinationType:(self.fLockDestination ? TorrentDeterminationUserSpecified : TorrentDeterminationAutomatic)]; } else { self.fLocationField.stringValue = @""; self.fLocationImageView.image = nil; } self.fTimer = [NSTimer scheduledTimerWithTimeInterval:UPDATE_SECONDS target:self selector:@selector(updateFiles) userInfo:nil repeats:YES]; [self updateFiles]; } - (void)windowDidLoad { //if there is no destination, prompt for one right away if (!self.fDestination) { [self setDestination:nil]; } } - (void)dealloc { [NSNotificationCenter.defaultCenter removeObserver:self]; [_fTimer invalidate]; } - (void)setDestination:(id)sender { NSOpenPanel* panel = [NSOpenPanel openPanel]; panel.prompt = NSLocalizedString(@"Select", "Open torrent -> prompt"); panel.allowsMultipleSelection = NO; panel.canChooseFiles = NO; panel.canChooseDirectories = YES; panel.canCreateDirectories = YES; panel.message = [NSString stringWithFormat:NSLocalizedString(@"Select the download folder for \"%@\"", "Add -> select destination folder"), self.torrent.name]; [panel beginSheetModalForWindow:self.window completionHandler:^(NSInteger result) { if (result == NSModalResponseOK) { self.fLockDestination = YES; [self setDestinationPath:panel.URLs[0].path determinationType:TorrentDeterminationUserSpecified]; } else { if (!self.fDestination) { [self performSelectorOnMainThread:@selector(cancelAdd:) withObject:nil waitUntilDone:NO]; } } }]; } - (void)add:(id)sender { if ([self.fDestination.lastPathComponent isEqualToString:self.torrent.name] && [NSUserDefaults.standardUserDefaults boolForKey:@"WarningFolderDataSameName"]) { NSAlert* alert = [[NSAlert alloc] init]; alert.messageText = NSLocalizedString(@"The destination directory and root data directory have the same name.", "Add torrent -> same name -> title"); alert.informativeText = NSLocalizedString( @"If you are attempting to use already existing data," " the root data directory should be inside the destination directory.", "Add torrent -> same name -> message"); alert.alertStyle = NSAlertStyleWarning; [alert addButtonWithTitle:NSLocalizedString(@"Cancel", "Add torrent -> same name -> button")]; [alert addButtonWithTitle:NSLocalizedString(@"Add", "Add torrent -> same name -> button")]; alert.showsSuppressionButton = YES; [alert beginSheetModalForWindow:[self window] completionHandler:^(NSModalResponse returnCode) { if (alert.suppressionButton.state == NSControlStateValueOn) { [NSUserDefaults.standardUserDefaults setBool:NO forKey:@"WarningFolderDataSameName"]; } if (returnCode == NSAlertSecondButtonReturn) { [self performSelectorOnMainThread:@selector(confirmAdd) withObject:nil waitUntilDone:NO]; } }]; } else { [self confirmAdd]; } } - (void)cancelAdd:(id)sender { [self.window performClose:sender]; } //only called on cancel - (BOOL)windowShouldClose:(id)window { [self.fTimer invalidate]; self.fTimer = nil; self.fFileController.torrent = nil; //avoid a crash when window tries to update [self.fController askOpenConfirmed:self add:NO]; return YES; } - (void)setFileFilterText:(id)sender { self.fFileController.filterText = [sender stringValue]; } - (IBAction)checkAll:(id)sender { [self.fFileController checkAll]; } - (IBAction)uncheckAll:(id)sender { [self.fFileController uncheckAll]; } - (void)verifyLocalData:(id)sender { [self.torrent resetCache]; [self updateFiles]; } - (void)changePriority:(id)sender { tr_priority_t priority; switch ([sender indexOfSelectedItem]) { case POPUP_PRIORITY_HIGH: priority = TR_PRI_HIGH; break; case POPUP_PRIORITY_NORMAL: priority = TR_PRI_NORMAL; break; case POPUP_PRIORITY_LOW: priority = TR_PRI_LOW; break; default: NSAssert1(NO, @"Unknown priority tag for adding torrent: %ld", [sender tag]); priority = TR_PRI_NORMAL; } self.torrent.priority = priority; } - (void)updateCheckButtons:(NSNotification*)notification { NSString* statusString = [NSString stringForFileSize:self.torrent.size]; if (self.torrent.folder) { //check buttons //keep synced with identical code in InfoFileViewController.m NSInteger const filesCheckState = [self.torrent checkForFiles:[NSIndexSet indexSetWithIndexesInRange:NSMakeRange(0, self.torrent.fileCount)]]; self.fCheckAllButton.enabled = filesCheckState != NSControlStateValueOn; //if anything is unchecked self.fUncheckAllButton.enabled = !self.torrent.allDownloaded; //if there are any checked files that aren't finished //status field NSString* fileString; NSUInteger count = self.torrent.fileCount; if (count != 1) { fileString = [NSString stringWithFormat:NSLocalizedString(@"%lu files", "Add torrent -> info"), count]; } else { fileString = NSLocalizedString(@"1 file", "Add torrent -> info"); } NSString* selectedString = [NSString stringWithFormat:NSLocalizedString(@"%@ selected", "Add torrent -> info"), [NSString stringForFileSize:self.torrent.totalSizeSelected]]; statusString = [NSString stringWithFormat:@"%@, %@ (%@)", fileString, statusString, selectedString]; } self.fStatusField.stringValue = statusString; } - (void)updateGroupMenu:(NSNotification*)notification { [self setGroupsMenu]; if (![self.fGroupPopUp selectItemWithTag:self.fGroupValue]) { self.fGroupValue = -1; self.fGroupValueDetermination = TorrentDeterminationAutomatic; [self.fGroupPopUp selectItemWithTag:self.fGroupValue]; } } #pragma mark - Private - (void)updateFiles { [self.torrent update]; [self.fFileController refresh]; [self updateCheckButtons:nil]; //call in case button state changed by checking if (self.torrent.checking) { BOOL const waiting = self.torrent.checkingWaiting; self.fVerifyIndicator.indeterminate = waiting; if (waiting) { [self.fVerifyIndicator startAnimation:self]; } else { self.fVerifyIndicator.doubleValue = self.torrent.checkingProgress; } } else { self.fVerifyIndicator.indeterminate = YES; //we want to hide when stopped, which only applies when indeterminate [self.fVerifyIndicator stopAnimation:self]; } } - (void)confirmAdd { [self.fTimer invalidate]; self.fTimer = nil; [self.torrent setGroupValue:self.fGroupValue determinationType:self.fGroupValueDetermination]; if (self.fTorrentFile && self.fCanToggleDelete && self.fDeleteCheck.state == NSControlStateValueOn) { [Torrent trashFile:self.fTorrentFile error:nil]; } if (self.fStartCheck.state == NSControlStateValueOn) { [self.torrent startTransfer]; } self.fFileController.torrent = nil; //avoid a crash when window tries to update [self close]; [self.fController askOpenConfirmed:self add:YES]; } - (void)setDestinationPath:(NSString*)destination determinationType:(TorrentDeterminationType)determinationType { destination = destination.stringByExpandingTildeInPath; if (!self.fDestination || ![self.fDestination isEqualToString:destination]) { self.fDestination = destination; [self.torrent changeDownloadFolderBeforeUsing:self.fDestination determinationType:determinationType]; } self.fLocationField.stringValue = self.fDestination.stringByAbbreviatingWithTildeInPath; self.fLocationField.toolTip = self.fDestination; ExpandedPathToIconTransformer* iconTransformer = [[ExpandedPathToIconTransformer alloc] init]; self.fLocationImageView.image = [iconTransformer transformedValue:self.fDestination]; } - (void)setGroupsMenu { NSMenu* groupMenu = [GroupsController.groups groupMenuWithTarget:self action:@selector(changeGroupValue:) isSmall:NO]; self.fGroupPopUp.menu = groupMenu; } - (void)changeGroupValue:(id)sender { NSInteger previousGroup = self.fGroupValue; self.fGroupValue = [sender tag]; self.fGroupValueDetermination = TorrentDeterminationUserSpecified; if (!self.fLockDestination) { if ([GroupsController.groups usesCustomDownloadLocationForIndex:self.fGroupValue]) { [self setDestinationPath:[GroupsController.groups customDownloadLocationForIndex:self.fGroupValue] determinationType:TorrentDeterminationAutomatic]; } else if ([self.fDestination isEqualToString:[GroupsController.groups customDownloadLocationForIndex:previousGroup]]) { [self setDestinationPath:[NSUserDefaults.standardUserDefaults stringForKey:@"DownloadFolder"] determinationType:TorrentDeterminationAutomatic]; } } } @end