447 lines
15 KiB
Plaintext
447 lines
15 KiB
Plaintext
// 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"
|
|
|
|
static NSTimeInterval const kUpdateSeconds = 1.0;
|
|
|
|
typedef NS_ENUM(NSUInteger, PopupPriority) {
|
|
PopupPriorityHigh = 0,
|
|
PopupPriorityNormal = 1,
|
|
PopupPriorityLow = 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;
|
|
|
|
@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;
|
|
|
|
//disable fullscreen support
|
|
self.window.collectionBehavior = NSWindowCollectionBehaviorFullScreenNone;
|
|
|
|
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];
|
|
|
|
PopupPriority priorityIndex;
|
|
switch (self.torrent.priority)
|
|
{
|
|
case TR_PRI_HIGH:
|
|
priorityIndex = PopupPriorityHigh;
|
|
break;
|
|
case TR_PRI_NORMAL:
|
|
priorityIndex = PopupPriorityNormal;
|
|
break;
|
|
case TR_PRI_LOW:
|
|
priorityIndex = PopupPriorityLow;
|
|
break;
|
|
default:
|
|
NSAssert1(NO, @"Unknown priority for adding torrent: %d", self.torrent.priority);
|
|
priorityIndex = PopupPriorityNormal;
|
|
}
|
|
[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:kUpdateSeconds 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 PopupPriorityHigh:
|
|
priority = TR_PRI_HIGH;
|
|
break;
|
|
case PopupPriorityNormal:
|
|
priority = TR_PRI_NORMAL;
|
|
break;
|
|
case PopupPriorityLow:
|
|
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 localizedStringWithFormat: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
|